-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtoken.go
223 lines (209 loc) · 5.54 KB
/
token.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
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
package ucenter
import (
"crypto/md5"
"fmt"
"github.com/garyburd/redigo/redis"
"strconv"
"time"
)
// UIDOffset 一毫秒内的偏移量
var UIDOffset = 0
// GetUID 得到全局唯一ID
// 时间毫秒(44) + 节点(5) + 自增长id(15位)
// 时间可以保证400年不重复
// 节点可以用于有多个机器同时产生id时,可以设置63个结点
// 每一毫秒能生成最多65535个id,但是由于机器的限制,
// 一毫秒内光调用GetUID也不能达到60000次(按3GHz算,相当于30个指令周期调用一次)
// 所以综合总论400年内不可能有重复
// 0 45 50 64
// +-----------+------+------------+
// |timestamp |node |increment |
// +-----------+------+------------+
func GetUID(node int) uint64 {
now := time.Now().UnixNano() / int64(1e6)
UIDOffset++
if UIDOffset > 1024 {
UIDOffset = 0
}
value := (now << 20) + int64(node<<15) + int64(UIDOffset)
return uint64(value)
}
// GetNewToken 产生新的token
func GetNewToken() string {
UID := GetUID(Config.NodeIdentfy)
token := md5.Sum([]byte(strconv.FormatInt(int64(UID), 10)))
tokenStr := fmt.Sprintf("%x", token)
return tokenStr
}
// TokenInfo token信息
type TokenInfo struct {
UserName string
RefreshToken string
RefreshTokenCreated string
AccessToken string
AccessTokenCreated string
PreAccessToken string
}
// TokenType type of token
type TokenType string
const (
refreshToken TokenType = "refresh_token"
accessToken TokenType = "access_token"
preAccessToken TokenType = "pre_access_token"
)
// SetRefreshToken set refresh token for database or redis
func SetRefreshToken(name string, token string) error {
if redisPool == nil {
u, err := GetTokenInfo(name)
if u == nil {
sql := "insert into " + Config.TokenTablename +
"(user_name, refresh_token, rtoken_created)" +
" values(?,?, now())"
_, err := db.Exec(sql, name, token)
if err != nil {
fmt.Println(err)
return ErrSetRefreshToken
}
return nil
}
sql := "update " + Config.TokenTablename +
" set refresh_token= ?, " +
" rtoken_created = now() where user_name=?"
_, err = db.Exec(sql, token, name)
if err != nil {
fmt.Println(err)
return ErrSetRefreshToken
}
return nil
}
// set redis cache, refresh_token 不设置过期时间
c := redisPool.Get()
defer c.Close()
_, err := c.Do("SET", "refresh_token@"+name, token)
if err != nil {
fmt.Println(err)
return ErrSetRefreshToken
}
return nil
}
// SetAccessToken set refresh_token for database or redis
func SetAccessToken(name string, token string) error {
if redisPool == nil {
u, err := GetTokenInfo(name)
if u == nil {
sql := "insert into " + Config.TokenTablename +
"(user_name, access_token, atoken_created)" +
" values(?,?, now())"
_, err := db.Exec(sql, name, token)
if err != nil {
fmt.Println(err)
return ErrSetAccessToken
}
return nil
}
sql := "update " + Config.TokenTablename +
" set access_token= ?, " +
" atoken_created = now() where user_name=?"
_, err = db.Exec(sql, token, name)
if err != nil {
fmt.Println(err)
return ErrSetAccessToken
}
return nil
}
// set redis cache, access_token
c := redisPool.Get()
defer c.Close()
_, err := c.Do("SET", "access_token@"+name, token,
"EX", strconv.Itoa(Config.TokenExpiresIn))
if err != nil {
fmt.Println(err)
return ErrSetAccessToken
}
return nil
}
// SetPreAccessToken set refresh_token for database or redis
func SetPreAccessToken(name string, token string) error {
if redisPool == nil {
u, err := GetTokenInfo(name)
if u == nil {
sql := "insert into " + Config.TokenTablename +
"(user_name, pre_access_token)" +
" values(?,?)"
_, err := db.Exec(sql, name, token)
if err != nil {
fmt.Println(err)
return ErrSetPreAccessToken
}
return nil
}
sql := "update " + Config.TokenTablename +
" set pre_access_token= ? " +
"where user_name=?"
_, err = db.Exec(sql, token, name)
if err != nil {
fmt.Println(err)
return ErrSetPreAccessToken
}
return nil
}
// set redis cache, pre_access_token
c := redisPool.Get()
defer c.Close()
_, err := c.Do("SET", "pre_access_token@"+name, token,
"EX", strconv.Itoa(Config.PreTokenExpireIn))
if err != nil {
fmt.Println(err)
return ErrSetPreAccessToken
}
return nil
}
// GetTokenInfo get token from database or redis
func GetTokenInfo(name string) (*TokenInfo, error) {
if redisPool == nil {
sql := "select user_name,refresh_token,rtoken_created," +
"access_token,atoken_created,pre_access_token from " +
Config.TokenTablename + " where user_name=?"
rows, err := db.Query(sql, name)
if err != nil {
return nil, err
}
defer rows.Close()
for rows.Next() {
var t TokenInfo
if err = rows.Scan(&t.UserName, &t.RefreshToken,
&t.RefreshTokenCreated, &t.AccessToken,
&t.AccessTokenCreated,
&t.PreAccessToken); err == nil {
return &t, nil
}
}
return nil, ErrTokenNotExist
}
c := redisPool.Get()
defer c.Close()
refreshToken, err := redis.String(c.Do("GET",
"refresh_token@"+name))
if err != nil {
fmt.Println("redis get failed:", err)
return nil, ErrGetRedis
}
accessToken, err := redis.String(c.Do("GET",
"access_token@"+name))
if err != nil {
fmt.Println("redis get failed:", err)
return nil, ErrGetRedis
}
pretoken, err := redis.String(c.Do("GET",
"pre_access_token@"+name))
if err != nil {
fmt.Println("redis get failed:", err)
return nil, ErrGetRedis
}
var t TokenInfo
t.UserName = name
t.AccessToken = accessToken
t.RefreshToken = refreshToken
t.PreAccessToken = pretoken
return &t, nil
}