2020-07-29 18:00:04 +08:00
|
|
|
package handler
|
2020-07-26 17:09:05 +08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-07-30 16:22:18 +08:00
|
|
|
"errors"
|
2020-07-26 17:09:05 +08:00
|
|
|
"net/http"
|
|
|
|
"net/http/httputil"
|
|
|
|
|
2022-01-14 11:01:02 +08:00
|
|
|
"github.com/golang-jwt/jwt/v4"
|
2022-01-04 15:51:32 +08:00
|
|
|
"github.com/zeromicro/go-zero/core/logx"
|
2022-02-19 20:50:33 +08:00
|
|
|
"github.com/zeromicro/go-zero/rest/internal/response"
|
2022-01-04 15:51:32 +08:00
|
|
|
"github.com/zeromicro/go-zero/rest/token"
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-07-30 16:22:18 +08:00
|
|
|
jwtAudience = "aud"
|
|
|
|
jwtExpire = "exp"
|
|
|
|
jwtId = "jti"
|
|
|
|
jwtIssueAt = "iat"
|
|
|
|
jwtIssuer = "iss"
|
|
|
|
jwtNotBefore = "nbf"
|
|
|
|
jwtSubject = "sub"
|
|
|
|
noDetailReason = "no detail reason"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
errInvalidToken = errors.New("invalid auth token")
|
|
|
|
errNoClaims = errors.New("no auth params")
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
2022-11-04 21:55:17 +08:00
|
|
|
// An AuthorizeOptions is authorize options.
|
2020-07-26 17:09:05 +08:00
|
|
|
AuthorizeOptions struct {
|
|
|
|
PrevSecret string
|
|
|
|
Callback UnauthorizedCallback
|
|
|
|
}
|
|
|
|
|
2021-03-01 19:15:35 +08:00
|
|
|
// UnauthorizedCallback defines the method of unauthorized callback.
|
2020-07-26 17:09:05 +08:00
|
|
|
UnauthorizedCallback func(w http.ResponseWriter, r *http.Request, err error)
|
2021-03-01 19:15:35 +08:00
|
|
|
// AuthorizeOption defines the method to customize an AuthorizeOptions.
|
|
|
|
AuthorizeOption func(opts *AuthorizeOptions)
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
2022-04-27 21:34:54 +08:00
|
|
|
// Authorize returns an authorization middleware.
|
2020-07-26 17:09:05 +08:00
|
|
|
func Authorize(secret string, opts ...AuthorizeOption) func(http.Handler) http.Handler {
|
|
|
|
var authOpts AuthorizeOptions
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(&authOpts)
|
|
|
|
}
|
|
|
|
|
2020-08-12 12:12:31 +08:00
|
|
|
parser := token.NewTokenParser()
|
2020-07-26 17:09:05 +08:00
|
|
|
return func(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2020-12-29 10:25:55 +08:00
|
|
|
tok, err := parser.ParseToken(r, secret, authOpts.PrevSecret)
|
2020-07-26 17:09:05 +08:00
|
|
|
if err != nil {
|
|
|
|
unauthorized(w, r, err, authOpts.Callback)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-12-29 10:25:55 +08:00
|
|
|
if !tok.Valid {
|
2020-07-30 16:22:18 +08:00
|
|
|
unauthorized(w, r, errInvalidToken, authOpts.Callback)
|
2020-07-26 17:09:05 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-12-29 10:25:55 +08:00
|
|
|
claims, ok := tok.Claims.(jwt.MapClaims)
|
2020-07-26 17:09:05 +08:00
|
|
|
if !ok {
|
2020-07-30 16:22:18 +08:00
|
|
|
unauthorized(w, r, errNoClaims, authOpts.Callback)
|
2020-07-26 17:09:05 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := r.Context()
|
|
|
|
for k, v := range claims {
|
|
|
|
switch k {
|
|
|
|
case jwtAudience, jwtExpire, jwtId, jwtIssueAt, jwtIssuer, jwtNotBefore, jwtSubject:
|
|
|
|
// ignore the standard claims
|
|
|
|
default:
|
|
|
|
ctx = context.WithValue(ctx, k, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
next.ServeHTTP(w, r.WithContext(ctx))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-01 19:15:35 +08:00
|
|
|
// WithPrevSecret returns an AuthorizeOption with setting previous secret.
|
2020-07-26 17:09:05 +08:00
|
|
|
func WithPrevSecret(secret string) AuthorizeOption {
|
|
|
|
return func(opts *AuthorizeOptions) {
|
|
|
|
opts.PrevSecret = secret
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-01 19:15:35 +08:00
|
|
|
// WithUnauthorizedCallback returns an AuthorizeOption with setting unauthorized callback.
|
2020-07-26 17:09:05 +08:00
|
|
|
func WithUnauthorizedCallback(callback UnauthorizedCallback) AuthorizeOption {
|
|
|
|
return func(opts *AuthorizeOptions) {
|
|
|
|
opts.Callback = callback
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func detailAuthLog(r *http.Request, reason string) {
|
|
|
|
// discard dump error, only for debug purpose
|
|
|
|
details, _ := httputil.DumpRequest(r, true)
|
|
|
|
logx.Errorf("authorize failed: %s\n=> %+v", reason, string(details))
|
|
|
|
}
|
|
|
|
|
|
|
|
func unauthorized(w http.ResponseWriter, r *http.Request, err error, callback UnauthorizedCallback) {
|
2022-02-19 20:50:33 +08:00
|
|
|
writer := response.NewHeaderOnceResponseWriter(w)
|
2020-07-26 17:09:05 +08:00
|
|
|
|
2020-07-30 16:22:18 +08:00
|
|
|
if err != nil {
|
|
|
|
detailAuthLog(r, err.Error())
|
|
|
|
} else {
|
|
|
|
detailAuthLog(r, noDetailReason)
|
|
|
|
}
|
2021-08-22 09:27:20 +08:00
|
|
|
|
2022-01-20 21:09:45 +08:00
|
|
|
// let callback go first, to make sure we respond with user-defined HTTP header
|
2020-07-26 17:09:05 +08:00
|
|
|
if callback != nil {
|
|
|
|
callback(writer, r, err)
|
|
|
|
}
|
2022-01-20 21:09:45 +08:00
|
|
|
|
|
|
|
// if user not setting HTTP header, we set header with 401
|
|
|
|
writer.WriteHeader(http.StatusUnauthorized)
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|