mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-04 12:27:37 +03:00
https://datatracker.ietf.org/doc/html/rfc2617 server will send both Basic and Digest header to client client can use either Basic or Digest for authentication Change-Id: Iaa6629c143551770c836af3ead823bd148b244c6
99 lines
2 KiB
Go
99 lines
2 KiB
Go
package auth
|
|
|
|
import (
|
|
"crypto/md5"
|
|
"encoding/hex"
|
|
"fmt"
|
|
|
|
"github.com/sagernet/sing/common"
|
|
"github.com/sagernet/sing/common/param"
|
|
)
|
|
|
|
const Realm = "sing-box"
|
|
|
|
type Challenge struct {
|
|
Username string
|
|
Nonce string
|
|
CNonce string
|
|
Nc string
|
|
Response string
|
|
}
|
|
|
|
type User struct {
|
|
Username string
|
|
Password string
|
|
}
|
|
|
|
type Authenticator struct {
|
|
userMap map[string][]string
|
|
}
|
|
|
|
func NewAuthenticator(users []User) *Authenticator {
|
|
if len(users) == 0 {
|
|
return nil
|
|
}
|
|
au := &Authenticator{
|
|
userMap: make(map[string][]string),
|
|
}
|
|
for _, user := range users {
|
|
au.userMap[user.Username] = append(au.userMap[user.Username], user.Password)
|
|
}
|
|
return au
|
|
}
|
|
|
|
func (au *Authenticator) Verify(username string, password string) bool {
|
|
passwordList, ok := au.userMap[username]
|
|
return ok && common.Contains(passwordList, password)
|
|
}
|
|
|
|
func (au *Authenticator) VerifyDigest(method string, uri string, s string) (string, bool) {
|
|
c, err := ParseChallenge(s)
|
|
if err != nil {
|
|
return "", false
|
|
}
|
|
if c.Username == "" || c.Nonce == "" || c.Nc == "" || c.CNonce == "" || c.Response == "" {
|
|
return "", false
|
|
}
|
|
passwordList, ok := au.userMap[c.Username]
|
|
if ok {
|
|
for _, password := range passwordList {
|
|
ha1 := md5str(c.Username + ":" + Realm + ":" + password)
|
|
ha2 := md5str(method + ":" + uri)
|
|
resp := md5str(ha1 + ":" + c.Nonce + ":" + c.Nc + ":" + c.CNonce + ":auth:" + ha2)
|
|
if resp == c.Response {
|
|
return c.Username, true
|
|
}
|
|
}
|
|
}
|
|
return "", false
|
|
}
|
|
|
|
func ParseChallenge(s string) (*Challenge, error) {
|
|
pp, err := param.Parse(s)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("digest: invalid challenge: %w", err)
|
|
}
|
|
var c Challenge
|
|
|
|
for _, p := range pp {
|
|
switch p.Key {
|
|
case "username":
|
|
c.Username = p.Value
|
|
case "nonce":
|
|
c.Nonce = p.Value
|
|
case "cnonce":
|
|
c.CNonce = p.Value
|
|
case "nc":
|
|
c.Nc = p.Value
|
|
case "response":
|
|
c.Response = p.Value
|
|
}
|
|
}
|
|
return &c, nil
|
|
}
|
|
|
|
func md5str(str string) string {
|
|
h := md5.New()
|
|
h.Write([]byte(str))
|
|
return hex.EncodeToString(h.Sum(nil))
|
|
}
|