maddy/internal/auth/pam/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

90 lines
2 KiB
Go

package pam
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
expectAddress bool
Log log.Logger
}
func New(modName, instName string, _, inlineArgs []string) (module.Module, error) {
if len(inlineArgs) != 0 {
return nil, errors.New("pam: inline arguments are not used")
}
return &Auth{
instName: instName,
Log: log.Logger{Name: modName},
}, nil
}
func (a *Auth) Name() string {
return "pam"
}
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)
cfg.Bool("expect_address", false, false, &a.expectAddress)
if _, err := cfg.Process(); err != nil {
return err
}
if !canCallDirectly && !a.useHelper {
return errors.New("pam: this build lacks support for direct libpam invocation, use helper binary")
}
if a.useHelper {
a.helperPath = filepath.Join(config.LibexecDirectory, "maddy-pam-helper")
if _, err := os.Stat(a.helperPath); err != nil {
return fmt.Errorf("pam: no helper binary (maddy-pam-helper) found in %s", config.LibexecDirectory)
}
}
return nil
}
func (a *Auth) AuthPlain(username, password string) ([]string, error) {
var accountName string
if a.expectAddress {
var err error
accountName, _, err = address.Split(username)
if err != nil {
return nil, err
}
} else {
accountName = username
}
if a.useHelper {
if err := external.AuthUsingHelper(a.helperPath, accountName, password); err != nil {
return nil, err
}
}
err := runPAMAuth(accountName, password)
if err != nil {
return nil, err
}
return []string{username}, nil
}
func init() {
module.Register("pam", New)
}