2020-07-26 17:09:05 +08:00
|
|
|
package auth
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
|
2020-08-08 16:40:10 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/collection"
|
|
|
|
"github.com/tal-tech/go-zero/core/stores/redis"
|
2020-07-26 17:09:05 +08:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/metadata"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
)
|
|
|
|
|
|
|
|
const defaultExpiration = 5 * time.Minute
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// An Authenticator is used to authenticate the rpc requests.
|
2020-07-26 17:09:05 +08:00
|
|
|
type Authenticator struct {
|
|
|
|
store *redis.Redis
|
|
|
|
key string
|
|
|
|
cache *collection.Cache
|
|
|
|
strict bool
|
|
|
|
}
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// NewAuthenticator returns an Authenticator.
|
2020-07-26 17:09:05 +08:00
|
|
|
func NewAuthenticator(store *redis.Redis, key string, strict bool) (*Authenticator, error) {
|
|
|
|
cache, err := collection.NewCache(defaultExpiration)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Authenticator{
|
|
|
|
store: store,
|
|
|
|
key: key,
|
|
|
|
cache: cache,
|
|
|
|
strict: strict,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// Authenticate authenticates the given ctx.
|
2020-07-26 17:09:05 +08:00
|
|
|
func (a *Authenticator) Authenticate(ctx context.Context) error {
|
|
|
|
md, ok := metadata.FromIncomingContext(ctx)
|
|
|
|
if !ok {
|
|
|
|
return status.Error(codes.Unauthenticated, missingMetadata)
|
|
|
|
}
|
|
|
|
|
|
|
|
apps, tokens := md[appKey], md[tokenKey]
|
|
|
|
if len(apps) == 0 || len(tokens) == 0 {
|
|
|
|
return status.Error(codes.Unauthenticated, missingMetadata)
|
|
|
|
}
|
|
|
|
|
|
|
|
app, token := apps[0], tokens[0]
|
|
|
|
if len(app) == 0 || len(token) == 0 {
|
|
|
|
return status.Error(codes.Unauthenticated, missingMetadata)
|
|
|
|
}
|
|
|
|
|
|
|
|
return a.validate(app, token)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Authenticator) validate(app, token string) error {
|
|
|
|
expect, err := a.cache.Take(app, func() (interface{}, error) {
|
|
|
|
return a.store.Hget(a.key, app)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
if a.strict {
|
|
|
|
return status.Error(codes.Internal, err.Error())
|
|
|
|
}
|
2021-02-09 10:58:11 +08:00
|
|
|
|
|
|
|
return nil
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if token != expect {
|
|
|
|
return status.Error(codes.Unauthenticated, accessDenied)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|