seaweedfs/weed/security/guard.go

138 lines
3.4 KiB
Go
Raw Normal View History

2015-01-06 06:20:04 +08:00
package security
import (
"errors"
"fmt"
"github.com/seaweedfs/seaweedfs/weed/glog"
2015-01-06 06:20:04 +08:00
"net"
"net/http"
"strings"
2015-01-06 06:20:04 +08:00
)
var (
ErrUnauthorized = errors.New("unauthorized token")
)
/*
Guard is to ensure data access security.
There are 2 ways to check access:
2022-09-15 14:06:44 +08:00
1. white list. It's checking request ip address.
2. JSON Web Token(JWT) generated from secretKey.
The jwt can come from:
1. url parameter jwt=...
2. request header "Authorization"
3. cookie with the name "jwt"
2015-01-06 06:20:04 +08:00
The white list is checked first because it is easy.
Then the JWT is checked.
The Guard will also check these claims if provided:
1. "exp" Expiration Time
2. "nbf" Not Before
Generating JWT:
2022-09-15 14:06:44 +08:00
1. use HS256 to sign
2. optionally set "exp", "nbf" fields, in Unix time,
the number of seconds elapsed since January 1, 1970 UTC.
2015-01-06 06:20:04 +08:00
Referenced:
https://github.com/pkieltyka/jwtauth/blob/master/jwtauth.go
*/
type Guard struct {
whiteListIp map[string]struct{}
whiteListCIDR map[string]*net.IPNet
2019-06-06 15:29:02 +08:00
SigningKey SigningKey
ExpiresAfterSec int
ReadSigningKey SigningKey
ReadExpiresAfterSec int
2015-01-06 06:20:04 +08:00
isWriteActive bool
isEmptyWhiteList bool
2015-01-06 06:20:04 +08:00
}
2019-06-06 15:29:02 +08:00
func NewGuard(whiteList []string, signingKey string, expiresAfterSec int, readSigningKey string, readExpiresAfterSec int) *Guard {
g := &Guard{
SigningKey: SigningKey(signingKey),
ExpiresAfterSec: expiresAfterSec,
ReadSigningKey: SigningKey(readSigningKey),
ReadExpiresAfterSec: readExpiresAfterSec,
}
g.UpdateWhiteList(whiteList)
2015-01-06 06:20:04 +08:00
return g
}
func (g *Guard) WhiteList(f http.HandlerFunc) http.HandlerFunc {
2019-06-06 15:29:02 +08:00
if !g.isWriteActive {
//if no security needed, just skip all checking
2015-01-06 06:20:04 +08:00
return f
}
return func(w http.ResponseWriter, r *http.Request) {
2015-02-08 07:35:28 +08:00
if err := g.checkWhiteList(w, r); err != nil {
2015-01-06 06:20:04 +08:00
w.WriteHeader(http.StatusUnauthorized)
return
}
f(w, r)
}
}
func GetActualRemoteHost(r *http.Request) (host string, err error) {
host = r.Header.Get("HTTP_X_FORWARDED_FOR")
if host == "" {
host = r.Header.Get("X-FORWARDED-FOR")
}
if strings.Contains(host, ",") {
host = host[0:strings.Index(host, ",")]
}
if host == "" {
host, _, err = net.SplitHostPort(r.RemoteAddr)
}
return
}
2015-02-08 07:35:28 +08:00
func (g *Guard) checkWhiteList(w http.ResponseWriter, r *http.Request) error {
if g.isEmptyWhiteList {
2015-02-08 07:35:28 +08:00
return nil
}
2015-01-06 06:20:04 +08:00
host, err := GetActualRemoteHost(r)
if err != nil {
return fmt.Errorf("get actual remote host %s in checkWhiteList failed: %v", r.RemoteAddr, err)
}
2015-10-20 13:03:18 +08:00
if _, ok := g.whiteListIp[host]; ok {
return nil
}
for _, cidrnet := range g.whiteListCIDR {
// If the whitelist entry contains a "/" it
// is a CIDR range, and we should check the
remote := net.ParseIP(host)
if cidrnet.Contains(remote) {
return nil
2015-01-06 06:20:04 +08:00
}
}
2016-06-27 03:49:08 +08:00
glog.V(0).Infof("Not in whitelist: %s", r.RemoteAddr)
2021-12-29 19:38:14 +08:00
return fmt.Errorf("Not in whitelist: %s", r.RemoteAddr)
2015-02-08 07:35:28 +08:00
}
func (g *Guard) UpdateWhiteList(whiteList []string) {
whiteListIp := make(map[string]struct{})
whiteListCIDR := make(map[string]*net.IPNet)
for _, ip := range whiteList {
if strings.Contains(ip, "/") {
_, cidrnet, err := net.ParseCIDR(ip)
if err != nil {
glog.Errorf("Parse CIDR %s in whitelist failed: %v", ip, err)
}
whiteListCIDR[ip] = cidrnet
} else {
whiteListIp[ip] = struct{}{}
}
}
g.isEmptyWhiteList = len(whiteListIp) == 0 && len(whiteListCIDR) == 0
g.isWriteActive = !g.isEmptyWhiteList || len(g.SigningKey) != 0
g.whiteListIp = whiteListIp
g.whiteListCIDR = whiteListCIDR
}