/** * json web token * @author yutent * @date 2020/09/16 17:23:52 */ import { base64encode, base64decode, hmac, sha1 } from 'crypto.js' const HS256_HEADER = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9' const DEFAULT_CONFIG = { ttl: 3600 * 24 * 7, level: 0, // 校验级别, 0: 不校验客户端, 2: ua, 4: ip, 6: ua + ip secret: 'it_is_secret_key' // jwt密钥, 使用时请修改 } function hmac_base64(str, secret) { let buf = hmac('sha256', str, secret, 'buffer') return base64encode(buf, true) } function JwtMiddleware(req, res, next) { let { secret, level } = this.get('jwt') let deviceID = '' let ssid // options请求不处理jwt if (req.method === 'OPTIONS') { return next() } // 校验UA if (level & 2) { deviceID += req.header('user-agent') } // 校验IP if (level & 4) { deviceID += req.ip() } if (deviceID) { deviceID = sha1(deviceID) } req.__mix_key__ = secret + deviceID next() } export function createJwt() { return { name: 'jwt', install(conf = {}) { if (!conf.secret) { console.warn( new Error( 'You must set a `secret` key for jwt, or it will use the default key.' ) ) } let jwt = Object.assign({}, DEFAULT_CONFIG, conf) this.set({ jwt }) this.use(JwtMiddleware) return { ttl: jwt.ttl, // 签名, 返回token // header: base64("{"typ":"JWT","alg":"HS256"}") // 这里固定使用sha256 sign(data, secret) { // 加入过期时间, let payload = { data, expires: Date.now() + this.ttl * 1000 } let token = '' payload = base64encode(JSON.stringify(payload), true) token = hmac_base64(`${HS256_HEADER}.${payload}`, secret) return `${HS256_HEADER}.${payload}.${token}` }, // 校验token verify(token = '', secret) { let jwt = token.split('.') let [_, payload, auth] = jwt if (jwt.length !== 3) { return false } payload = JSON.parse(base64decode(payload, true)) // 如果已经过期, 则不再校验hash if (payload.expires < Date.now()) { return false } if (hmac_base64(jwt.join('.'), secret) === auth) { return payload.data } return false } } } } }