|
|
|
@ -7,6 +7,8 @@ package models
|
|
|
|
|
import ( |
|
|
|
|
"encoding/json" |
|
|
|
|
"errors" |
|
|
|
|
"fmt" |
|
|
|
|
"net/smtp" |
|
|
|
|
"strings" |
|
|
|
|
"time" |
|
|
|
|
|
|
|
|
@ -50,6 +52,22 @@ func (cfg *LDAPConfig) ToDB() ([]byte, error) {
|
|
|
|
|
return json.Marshal(cfg.Ldapsource) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
type SMTPConfig struct { |
|
|
|
|
Auth string |
|
|
|
|
Host string |
|
|
|
|
Post string |
|
|
|
|
TLS bool |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// implement
|
|
|
|
|
func (cfg *SMTPConfig) FromDB(bs []byte) error { |
|
|
|
|
return json.Unmarshal(bs, cfg) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (cfg *SMTPConfig) ToDB() ([]byte, error) { |
|
|
|
|
return json.Marshal(cfg) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
type LoginSource struct { |
|
|
|
|
Id int64 |
|
|
|
|
Type int |
|
|
|
@ -69,6 +87,10 @@ func (source *LoginSource) LDAP() *LDAPConfig {
|
|
|
|
|
return source.Cfg.(*LDAPConfig) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (source *LoginSource) SMTP() *SMTPConfig { |
|
|
|
|
return source.Cfg.(*SMTPConfig) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// for xorm callback
|
|
|
|
|
func (source *LoginSource) BeforeSet(colName string, val xorm.Cell) { |
|
|
|
|
if colName == "type" { |
|
|
|
@ -76,6 +98,8 @@ func (source *LoginSource) BeforeSet(colName string, val xorm.Cell) {
|
|
|
|
|
switch ty { |
|
|
|
|
case LT_LDAP: |
|
|
|
|
source.Cfg = new(LDAPConfig) |
|
|
|
|
case LT_SMTP: |
|
|
|
|
source.Cfg = new(SMTPConfig) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
@ -172,10 +196,19 @@ func LoginUser(uname, passwd string) (*User, error) {
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
for _, source := range sources { |
|
|
|
|
u, err := LoginUserLdapSource(nil, u.LoginName, passwd, |
|
|
|
|
source.Id, source.Cfg.(*LDAPConfig), true) |
|
|
|
|
if err == nil { |
|
|
|
|
return u, err |
|
|
|
|
if source.Type == LT_LDAP { |
|
|
|
|
u, err := LoginUserLdapSource(nil, u.LoginName, passwd, |
|
|
|
|
source.Id, source.Cfg.(*LDAPConfig), true) |
|
|
|
|
if err == nil { |
|
|
|
|
return u, err |
|
|
|
|
} |
|
|
|
|
} else if source.Type == LT_SMTP { |
|
|
|
|
u, err := LoginUserSMTPSource(nil, u.LoginName, passwd, |
|
|
|
|
source.Id, source.Cfg.(*SMTPConfig), true) |
|
|
|
|
|
|
|
|
|
if err == nil { |
|
|
|
|
return u, err |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -200,6 +233,8 @@ func LoginUser(uname, passwd string) (*User, error) {
|
|
|
|
|
return LoginUserLdapSource(u, u.LoginName, passwd, |
|
|
|
|
source.Id, source.Cfg.(*LDAPConfig), false) |
|
|
|
|
case LT_SMTP: |
|
|
|
|
return LoginUserSMTPSource(u, u.LoginName, passwd, |
|
|
|
|
source.Id, source.Cfg.(*SMTPConfig), false) |
|
|
|
|
} |
|
|
|
|
return nil, ErrUnsupportedLoginType |
|
|
|
|
} |
|
|
|
@ -232,3 +267,89 @@ func LoginUserLdapSource(user *User, name, passwd string, sourceId int64, cfg *L
|
|
|
|
|
|
|
|
|
|
return RegisterUser(user) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
type loginAuth struct { |
|
|
|
|
username, password string |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func LoginAuth(username, password string) smtp.Auth { |
|
|
|
|
return &loginAuth{username, password} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (a *loginAuth) Start(server *smtp.ServerInfo) (string, []byte, error) { |
|
|
|
|
return "LOGIN", []byte(a.username), nil |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (a *loginAuth) Next(fromServer []byte, more bool) ([]byte, error) { |
|
|
|
|
if more { |
|
|
|
|
switch string(fromServer) { |
|
|
|
|
case "Username:": |
|
|
|
|
return []byte(a.username), nil |
|
|
|
|
case "Password:": |
|
|
|
|
return []byte(a.password), nil |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
return nil, nil |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
var ( |
|
|
|
|
smtpAuths = []string{"plain", "login", ""} |
|
|
|
|
) |
|
|
|
|
|
|
|
|
|
func SmtpAuth(addr string, a smtp.Auth) error { |
|
|
|
|
c, err := smtp.Dial(addr) |
|
|
|
|
if err != nil { |
|
|
|
|
return err |
|
|
|
|
} |
|
|
|
|
defer c.Close() |
|
|
|
|
|
|
|
|
|
if ok, _ := c.Extension("STARTTLS"); ok { |
|
|
|
|
if err = c.StartTLS(nil); err != nil { |
|
|
|
|
return err |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if ok, _ := c.Extension("AUTH"); ok { |
|
|
|
|
if err = c.Auth(a); err != nil { |
|
|
|
|
return err |
|
|
|
|
} |
|
|
|
|
return nil |
|
|
|
|
} else { |
|
|
|
|
return ErrUnsupportedLoginType |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Query if name/passwd can login against the LDAP direcotry pool
|
|
|
|
|
// Create a local user if success
|
|
|
|
|
// Return the same LoginUserPlain semantic
|
|
|
|
|
func LoginUserSMTPSource(user *User, name, passwd string, sourceId int64, cfg *SMTPConfig, autoRegister bool) (*User, error) { |
|
|
|
|
var auth smtp.Auth |
|
|
|
|
if cfg.Auth == "plain" { |
|
|
|
|
auth = smtp.PlainAuth("", name, passwd, cfg.Host) |
|
|
|
|
} else if cfg.Auth == "login" { |
|
|
|
|
auth = LoginAuth(name, passwd) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
err := SmtpAuth(fmt.Sprintf("%s:%d", cfg.Host, cfg.Post), auth) |
|
|
|
|
if err != nil { |
|
|
|
|
return nil, err |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if !autoRegister { |
|
|
|
|
return user, nil |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// fake a local user creation
|
|
|
|
|
user = &User{ |
|
|
|
|
LowerName: strings.ToLower(name), |
|
|
|
|
Name: strings.ToLower(name), |
|
|
|
|
LoginType: LT_SMTP, |
|
|
|
|
LoginSource: sourceId, |
|
|
|
|
LoginName: name, |
|
|
|
|
IsActive: true, |
|
|
|
|
Passwd: passwd, |
|
|
|
|
Email: name, |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return RegisterUser(user) |
|
|
|
|
} |
|
|
|
|