sing/common/auth/auth.go
Tommy Wu 0a3c811e42 add Digest authentication for http proxy server
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
2025-02-12 15:00:40 +08:00

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