mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-04 21:47:40 +03:00
The authentication provider can now provide multiple authorization identities associated with credentials. Protocols that support that (e.g. JMAP, SASL) can let the client select the wanted identity.
105 lines
2.3 KiB
Go
105 lines
2.3 KiB
Go
// +build !windows
|
|
|
|
package shadow
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"github.com/foxcpp/maddy/internal/address"
|
|
"github.com/foxcpp/maddy/internal/auth/external"
|
|
"github.com/foxcpp/maddy/internal/config"
|
|
"github.com/foxcpp/maddy/internal/log"
|
|
"github.com/foxcpp/maddy/internal/module"
|
|
)
|
|
|
|
type Auth struct {
|
|
instName string
|
|
useHelper bool
|
|
helperPath string
|
|
|
|
Log log.Logger
|
|
}
|
|
|
|
func New(modName, instName string, _, inlineArgs []string) (module.Module, error) {
|
|
if len(inlineArgs) != 0 {
|
|
return nil, errors.New("shadow: inline arguments are not used")
|
|
}
|
|
return &Auth{
|
|
instName: instName,
|
|
Log: log.Logger{Name: modName},
|
|
}, nil
|
|
}
|
|
|
|
func (a *Auth) Name() string {
|
|
return "shadow"
|
|
}
|
|
|
|
func (a *Auth) InstanceName() string {
|
|
return a.instName
|
|
}
|
|
|
|
func (a *Auth) Init(cfg *config.Map) error {
|
|
cfg.Bool("debug", true, false, &a.Log.Debug)
|
|
cfg.Bool("use_helper", false, false, &a.useHelper)
|
|
if _, err := cfg.Process(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if a.useHelper {
|
|
a.helperPath = filepath.Join(config.LibexecDirectory, "maddy-shadow-helper")
|
|
if _, err := os.Stat(a.helperPath); err != nil {
|
|
return fmt.Errorf("shadow: no helper binary (maddy-shadow-helper) found in %s", config.LibexecDirectory)
|
|
}
|
|
} else {
|
|
f, err := os.Open("/etc/shadow")
|
|
if err != nil {
|
|
if os.IsPermission(err) {
|
|
return fmt.Errorf("shadow: can't read /etc/shadow due to permission error, use helper binary or run maddy as a privileged user")
|
|
}
|
|
return fmt.Errorf("shadow: can't read /etc/shadow: %v", err)
|
|
}
|
|
f.Close()
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (a *Auth) AuthPlain(username, password string) ([]string, error) {
|
|
accountName, _, err := address.Split(username)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if a.useHelper {
|
|
return []string{username}, external.AuthUsingHelper(a.helperPath, accountName, password)
|
|
}
|
|
|
|
ent, err := Lookup(accountName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !ent.IsAccountValid() {
|
|
return nil, fmt.Errorf("shadow: account is expired")
|
|
}
|
|
|
|
if !ent.IsPasswordValid() {
|
|
return nil, fmt.Errorf("shadow: password is expired")
|
|
}
|
|
|
|
if err := ent.VerifyPassword(password); err != nil {
|
|
if err == ErrWrongPassword {
|
|
return nil, module.ErrUnknownCredentials
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
return []string{username}, nil
|
|
}
|
|
|
|
func init() {
|
|
module.Register("shadow", New)
|
|
}
|