maddy/internal/auth/shadow/module.go
fox.cpp a45c7090c4
Improve auth. provider interface
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.
2020-02-27 01:22:47 +03:00

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)
}