gf框架gf-jwt使用

说明

插件地址 https://github.com/gogf/gf-jwt
gf-jwt是gf框架的jwt 中间件, 再次强调gf-jwt是一个中间件
gf-jwt插件是鹏哥 根据 gin-jwt的设计灵感来开发的
gf-jwt插件是基于go-jwt开发的

使用说明

由于github中的说明文档对新手不太友好,新手一般无从下手。所以本篇文章就以最简单的方式向大家介绍这个插件的使用方法

你稍微记一下

刘二狗:怎么生成token ?
答: gf-jwt已经封装好了,不用你自己动手来生成

s.BindHandler("POST:/login", auth.GfJWTMiddleware.LoginHandler)

请求返回:

{
"code": 200,
"expire": "2020-01-08T11:16:41+08:00",
"token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJleHAiOjE1Nzg0NTM0MDEsImlkIjoiYWRtaW4iLCJvcmlnX2lhdCI6MTU3ODQ1MzEwMSwidXNlcm5hbWUiOiJhZG1pbiJ9.YwigtrdlUDHYYRpd9aOu-FlheQM2IaYkNLLG-qA4L4o"
}

张木棍:那我想自己定义返回的视图函数咋搞?
答:生成token、刷新token、验证token......gf-jwt中都已经封装好了,想自定义的话也很简单, 只需自己定义指定函数,配置一下即可

// 定义自定义中间件
var CustomGfJWTMiddleware *jwt.GfJWTMiddleware

// 重写该方法
func init () {
    // 重新定义GfJWTMiddleware
    authMiddleware, err := jwt.New(&jwt.GfJWTMiddleware{
        Realm:           "test zone",           // 用于展示中间件的名称
        Key:             []byte("secret key"),  // 密钥
        Timeout:         time.Minute * 5,       // token过期时间
        MaxRefresh:      time.Minute * 5,
        IdentityKey:     "id",                  // 身份验证的key值
        TokenLookup:     "header: Authorization, query: token, cookie: jwt",    // token检索模式,用于提取token-> Authorization
        TokenHeadName:   "Bearer",              // token在请求头时的名称,默认值为Bearer
                                                // 客户端在header中传入Authorization 对一个值是Bearer + 空格 + token
        TimeFunc:        time.Now,              // 测试或服务器在其他时区可设置该属性
        Authenticator:   Authenticator,         // 根据登录信息对用户进行身份验证的回调函数
        LoginResponse:   LoginResponse,         // 完成登录后返回的信息,用户可自定义返回数据,默认返回
        RefreshResponse: auth.RefreshResponse,  // 刷新token后返回的信息,用户可自定义返回数据,默认返回
        Unauthorized:    auth.Unauthorized,     // 处理不进行授权的逻辑
        IdentityHandler: auth.IdentityHandler,  // 解析并设置用户身份信息
        PayloadFunc:     auth.PayloadFunc,      // 登录期间的回调的函数
    })
    if err != nil {
        glog.Fatal("JWT Error:" + err.Error())
    }
    CustomGfJWTMiddleware = authMiddleware
}

王大宝: 配置好了咋个使用嘞?
答: 首先定义中间件,中间件中使用自己重写的CustomGfJWTMiddleware

// 自定义中间件
func MiddlewareAuth(r *ghttp.Request) {
    // 使用 gf-jwt中间件
    CustomGfJWTMiddleware.MiddlewareFunc()(r)
    r.Middleware.Next()
}

然后在视图函数中使用就好了

// 使用中间件
s.Group("/api", func(group *ghttp.RouterGroup) {
    // 使用中间件
    group.Middleware(MiddlewareAuth)

    // 需要验证token的视图函数
    group.GET("/get_info", func(r *ghttp.Request) {
        r.Response.Write("api get_info...")
    })

    // 刷新token
    group.POST("/refresh_token", CustomGfJWTMiddleware.RefreshHandler)
})

啸天犬:完整代码拿来看看
答:好

package main

import (
    "gfjwttest/auth"
    jwt "github.com/gogf/gf-jwt"
    "github.com/gogf/gf/frame/g"
    "github.com/gogf/gf/net/ghttp"
    "github.com/gogf/gf/os/glog"
    "github.com/gogf/gf/util/gvalid"
    "net/http"
    "time"
)

// 定义自定义中间件
var CustomGfJWTMiddleware *jwt.GfJWTMiddleware

// 重写该方法
func init () {
    // 重新定义GfJWTMiddleware
    authMiddleware, err := jwt.New(&jwt.GfJWTMiddleware{
        Realm:           "test zone",           // 用于展示中间件的名称
        Key:             []byte("secret key"),  // 密钥
        Timeout:         time.Minute * 5,       // token过期时间
        MaxRefresh:      time.Minute * 5,
        IdentityKey:     "id",                  // 身份验证的key值
        TokenLookup:     "header: Authorization, query: token, cookie: jwt",    // token检索模式,用于提取token-> Authorization
        TokenHeadName:   "Bearer",              // token在请求头时的名称,默认值为Bearer
                                                // 客户端在header中传入Authorization 对一个值是Bearer + 空格 + token
        TimeFunc:        time.Now,              // 测试或服务器在其他时区可设置该属性
        Authenticator:   Authenticator,         // 根据登录信息对用户进行身份验证的回调函数
        LoginResponse:   LoginResponse,         // 完成登录后返回的信息,用户可自定义返回数据,默认返回
        RefreshResponse: auth.RefreshResponse,  // 刷新token后返回的信息,用户可自定义返回数据,默认返回
        Unauthorized:    auth.Unauthorized,     // 处理不进行授权的逻辑
        IdentityHandler: auth.IdentityHandler,  // 解析并设置用户身份信息
        PayloadFunc:     auth.PayloadFunc,      // 登录期间的回调的函数
    })
    if err != nil {
        glog.Fatal("JWT Error:" + err.Error())
    }
    CustomGfJWTMiddleware = authMiddleware
}

// 自定义中间件
func MiddlewareAuth(r *ghttp.Request) {
    // 使用 gf-jwt中间件
    CustomGfJWTMiddleware.MiddlewareFunc()(r)
    r.Middleware.Next()
}

func main() {
    s := g.Server()

    // 登录,返回token
    s.BindHandler("POST:/login", CustomGfJWTMiddleware.LoginHandler)

    // 使用中间件
    s.Group("/api", func(group *ghttp.RouterGroup) {
        // 使用中间件
        group.Middleware(MiddlewareAuth)

        // 需要验证token的视图函数
        group.GET("/get_info", func(r *ghttp.Request) {
            r.Response.Write("api get_info...")
        })

        // 刷新token
        group.POST("/refresh_token", CustomGfJWTMiddleware.RefreshHandler)
    })

    s.Run()
}

//身份验证器用于验证登录参数。
//它必须返回用户数据作为用户标识符,它将存储在声明数组中。
//检查错误(e)以确定适当的错误消息。
func Authenticator(r *ghttp.Request) (interface{}, error) {
    data := r.GetMap()
    if e := gvalid.CheckMap(data, auth.ValidationRules); e != nil {
        return "", jwt.ErrFailedAuthentication
    }
    if data["username"] == "admin" && data["password"] == "111111" {
        return g.Map {
            "username": data["username"],
            "id":       1,
        }, nil
    }
    return nil, jwt.ErrFailedAuthentication
}


// LoginResponse用于定义自定义的登录成功回调函数。
func LoginResponse(r *ghttp.Request, code int, token string, expire time.Time) {
    _ = r.Response.WriteJson(g.Map{
        "id":   1,
        "code":   http.StatusOK,
        "token":  token,
        "expire": expire.Format(time.RFC3339),
    })
    r.ExitAll()
}

胖虎:客户端请求一个我看看
答:好

拿到token

李富贵:请求一下数据接口我看看(登录后的)
答:行吧, 记得把刚才请求的token加载请求头中

请求结果

Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJleHAiOjE1Nzg0NTQ1MDAsImlkIjoxLCJvcmlnX2lhdCI6MTU3ODQ1NDIwMCwidXNlcm5hbWUiOiJhZG1pbiJ9.EOmHjtSS2NVNJzMyyAXfQ0CtlX0zg0laQJGkxI8wN3M

刘碧萝:能给我小心心吗?
答:不能,滚!

刘碧萝:好嘞

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351