1
0
mirror of https://github.com/go-micro/go-micro.git synced 2025-01-23 17:53:05 +02:00

158 lines
2.9 KiB
Go
Raw Normal View History

2020-04-29 09:21:17 +01:00
package jwt
import (
2020-04-29 13:21:51 +01:00
"sync"
2020-05-14 13:56:51 +01:00
"time"
2020-04-29 09:21:17 +01:00
2021-01-20 21:01:10 +00:00
"github.com/asim/go-micro/v3/auth"
"github.com/asim/go-micro/v3/cmd"
jwtToken "github.com/asim/go-micro/plugins/auth/jwt/v3/token"
2020-04-29 09:21:17 +01:00
)
2020-12-29 15:49:26 +00:00
func init() {
cmd.DefaultAuths["jwt"] = NewAuth
}
2020-04-29 09:21:17 +01:00
// NewAuth returns a new instance of the Auth service
func NewAuth(opts ...auth.Option) auth.Auth {
j := new(jwt)
j.Init(opts...)
return j
}
2020-12-12 20:08:39 +00:00
func NewRules() auth.Rules {
return new(jwtRules)
}
2020-04-29 09:21:17 +01:00
type jwt struct {
2020-12-12 20:08:39 +00:00
sync.Mutex
2020-04-29 09:21:17 +01:00
options auth.Options
2020-12-29 15:49:26 +00:00
jwt jwtToken.Provider
2020-12-12 20:08:39 +00:00
}
2020-04-29 13:21:51 +01:00
2020-12-12 20:08:39 +00:00
type jwtRules struct {
2020-04-29 13:21:51 +01:00
sync.Mutex
2020-12-12 20:08:39 +00:00
rules []*auth.Rule
2020-04-29 09:21:17 +01:00
}
func (j *jwt) String() string {
return "jwt"
}
func (j *jwt) Init(opts ...auth.Option) {
2020-04-29 13:21:51 +01:00
j.Lock()
defer j.Unlock()
2020-04-29 09:21:17 +01:00
for _, o := range opts {
o(&j.options)
}
2020-12-29 15:49:26 +00:00
j.jwt = jwtToken.New(
jwtToken.WithPrivateKey(j.options.PrivateKey),
jwtToken.WithPublicKey(j.options.PublicKey),
2020-04-29 09:21:17 +01:00
)
}
func (j *jwt) Options() auth.Options {
2020-04-29 13:21:51 +01:00
j.Lock()
defer j.Unlock()
2020-04-29 09:21:17 +01:00
return j.options
}
func (j *jwt) Generate(id string, opts ...auth.GenerateOption) (*auth.Account, error) {
2020-04-29 13:21:51 +01:00
options := auth.NewGenerateOptions(opts...)
account := &auth.Account{
ID: id,
Type: options.Type,
Scopes: options.Scopes,
Metadata: options.Metadata,
2020-05-21 16:41:55 +01:00
Issuer: j.Options().Namespace,
2020-04-29 13:21:51 +01:00
}
// generate a JWT secret which can be provided to the Token() method
// and exchanged for an access token
secret, err := j.jwt.Generate(account)
if err != nil {
return nil, err
}
account.Secret = secret.Token
// return the account
return account, nil
2020-04-29 09:21:17 +01:00
}
2020-12-12 20:08:39 +00:00
func (j *jwtRules) Grant(rule *auth.Rule) error {
2020-04-29 13:21:51 +01:00
j.Lock()
defer j.Unlock()
2020-05-20 11:59:01 +01:00
j.rules = append(j.rules, rule)
2020-04-29 13:21:51 +01:00
return nil
2020-04-29 09:21:17 +01:00
}
2020-12-12 20:08:39 +00:00
func (j *jwtRules) Revoke(rule *auth.Rule) error {
2020-04-29 13:21:51 +01:00
j.Lock()
defer j.Unlock()
2020-05-20 11:59:01 +01:00
rules := []*auth.Rule{}
2020-06-02 10:26:33 +08:00
for _, r := range j.rules {
2020-05-20 11:59:01 +01:00
if r.ID != rule.ID {
2020-04-29 13:21:51 +01:00
rules = append(rules, r)
}
}
j.rules = rules
return nil
2020-04-29 09:21:17 +01:00
}
2020-12-12 20:08:39 +00:00
func (j *jwtRules) Verify(acc *auth.Account, res *auth.Resource, opts ...auth.VerifyOption) error {
j.Lock()
2020-05-20 11:59:01 +01:00
defer j.Unlock()
2020-05-20 16:49:52 +01:00
2020-05-21 16:41:55 +01:00
var options auth.VerifyOptions
2020-05-20 16:49:52 +01:00
for _, o := range opts {
o(&options)
}
2020-12-12 19:08:36 +00:00
return auth.Verify(j.rules, acc, res)
2020-05-20 11:59:01 +01:00
}
2020-04-29 13:21:51 +01:00
2020-12-12 20:08:39 +00:00
func (j *jwtRules) List(opts ...auth.ListOption) ([]*auth.Rule, error) {
2020-05-20 11:59:01 +01:00
j.Lock()
defer j.Unlock()
return j.rules, nil
2020-04-29 09:21:17 +01:00
}
func (j *jwt) Inspect(token string) (*auth.Account, error) {
return j.jwt.Inspect(token)
}
func (j *jwt) Token(opts ...auth.TokenOption) (*auth.Token, error) {
options := auth.NewTokenOptions(opts...)
2020-04-29 13:21:51 +01:00
secret := options.RefreshToken
if len(options.Secret) > 0 {
secret = options.Secret
}
account, err := j.jwt.Inspect(secret)
if err != nil {
return nil, err
2020-04-29 09:21:17 +01:00
}
2020-12-29 15:49:26 +00:00
access, err := j.jwt.Generate(account, jwtToken.WithExpiry(options.Expiry))
2020-05-14 13:56:51 +01:00
if err != nil {
return nil, err
}
2020-12-29 15:49:26 +00:00
refresh, err := j.jwt.Generate(account, jwtToken.WithExpiry(options.Expiry+time.Hour))
2020-04-29 09:21:17 +01:00
if err != nil {
return nil, err
}
return &auth.Token{
2020-05-14 13:56:51 +01:00
Created: access.Created,
Expiry: access.Expiry,
AccessToken: access.Token,
RefreshToken: refresh.Token,
2020-04-29 09:21:17 +01:00
}, nil
}