-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathsign.go
100 lines (88 loc) · 3.53 KB
/
sign.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
package jwt
// Sign signs and generates a new token based on the algorithm and a secret key.
// The claims is the payload, the actual body of the token, should
// contain information about a specific authorized client.
// Note that the payload part is not encrypted,
// therefore it should NOT contain any private information
// (unless Encrypt/Decrypt functions are set, see GCM function too).
// See the `Verify` function to decode and verify the result token.
//
// Example Code to pass only standard Claims:
//
// token, err := jwt.Sign(jwt.HS256, []byte("secret"), jwt.Claims{...})
//
// Example Code to pass custom and expiration Claims manually:
//
// now := time.Now()
// token, err := jwt.Sign(jwt.HS256, []byte("secret"), map[string]interface{}{
// "iat": now.Unix(),
// "exp": now.Add(15 * time.Minute).Unix(),
// "foo": "bar",
// })
//
// Example Code for custom and standard claims using a SignOption:
//
// token, err := jwt.Sign(jwt.HS256, []byte("secret"), jwt.Map{"foo":"bar"}, jwt.MaxAge(15 * time.Minute))
// OR
// token, err := jwt.Sign(jwt.HS256, []byte("secret"), jwt.Map{"foo":"bar"}, jwt.Claims {Expiry: ...})
//
// Example Code for custom type as Claims + standard Claims:
//
// type User struct { Username string `json:"username"` }
// token, err := jwt.Sign(jwt.HS256, []byte("secret"), User{Username: "kataras"}, jwt.MaxAge(15 * time.Minute))
func Sign(alg Alg, key PrivateKey, claims interface{}, opts ...SignOption) ([]byte, error) {
return signToken(alg, key, nil, claims, nil, opts...)
}
// SignEncrypted same as `Sign` but it encrypts the payload part with the given "encrypt" function.
// The "encrypt" function is called AFTER Marshal.
// Look the `GCM` function for details.
func SignEncrypted(alg Alg, key PrivateKey, encrypt InjectFunc, claims interface{}, opts ...SignOption) ([]byte, error) {
return signToken(alg, key, encrypt, claims, nil, opts...)
}
// SignWithHeader same as `Sign` but accepts a custom json header structure too.
func SignWithHeader(alg Alg, key PrivateKey, claims interface{}, customHeader interface{}, opts ...SignOption) ([]byte, error) {
return signToken(alg, key, nil, claims, customHeader, opts...)
}
// SignEncryptedWithHeader same as `SignEncrypted` but accepts a custom json header structure too.
func SignEncryptedWithHeader(alg Alg, key PrivateKey, encrypt InjectFunc, claims interface{}, customHeader interface{}, opts ...SignOption) ([]byte, error) {
return signToken(alg, key, encrypt, claims, customHeader, opts...)
}
func signToken(alg Alg, key PrivateKey, encrypt InjectFunc, claims interface{}, customHeader interface{}, opts ...SignOption) ([]byte, error) {
if len(opts) > 0 {
var standardClaims Claims
for _, opt := range opts {
if opt == nil {
continue
}
opt.ApplyClaims(&standardClaims)
}
claims = Merge(claims, standardClaims)
}
payload, err := Marshal(claims)
if err != nil {
return nil, err
}
if encrypt != nil {
payload, err = encrypt(payload)
if err != nil {
return nil, err
}
}
return encodeToken(alg, key, payload, customHeader)
}
// SignOption is just a helper which sets the standard claims at the `Sign` function.
//
// Available SignOptions:
// - MaxAge(time.Duration)
// - Claims{}
type SignOption interface {
// ApplyClaims should apply standard claims.
// Accepts the destination claims.
ApplyClaims(*Claims)
}
// SignOptionFunc completes the `SignOption`. It's a helper to pass a `SignOption` as a function.
type SignOptionFunc func(*Claims)
// ApplyClaims completes the `SignOption` interface.
func (f SignOptionFunc) ApplyClaims(c *Claims) {
f(c)
}