Go HTTP Middleware學(xué)習(xí)2

使用第三方中間件

很多第三方庫提供了不同類型的可重用的中間件組件,可以用在很多共同的功能:比如授權(quán),登錄,壓縮響應(yīng)頭等.當(dāng)使用Go開發(fā)一個(gè)web應(yīng)用,可以使用這些第三方庫應(yīng)用到自己的項(xiàng)目中.

  • 使用Gorilla Handlers

這個(gè)開發(fā)組件提供了很多net/http包下的handlers.
一個(gè)使用Gorilla的LoggingHandler和CompressHandler的例子

// Gorilla Handlers
package main

import (
    "fmt"
    "log"
    "net/http"
    "os"

    "github.com/gorilla/handlers"
)

func index(w http.ResponseWriter, r *http.Request) {
    log.Println("Execute index Handler")
    fmt.Fprintf(w, "Welcome!")

}

func about(w http.ResponseWriter, r *http.Request) {
    log.Println("Execute message Handler")
    fmt.Fprintf(w, "Message Go!")
}

func main() {
    indexHandler := http.HandlerFunc(index)
    aboutHandler := http.HandlerFunc(about)

    logFile, err := os.OpenFile("server.log", os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666)
    if err != nil {
        panic(err)
    }

    http.Handle("/", handlers.LoggingHandler(logFile, handlers.CompressHandler(indexHandler)))
    http.Handle("about", handlers.LoggingHandler(logFile, handlers.CompressHandler(aboutHandler)))
    server := &http.Server{
        Addr: ":9090",
    }
    log.Println("Listening...")
    server.ListenAndServe()

}

//運(yùn)行時(shí)訪問日志信息server.log
::1 - - [29/Jul/2016:18:48:22 +0800] "GET / HTTP/1.1" 200 32
::1 - - [29/Jul/2016:18:48:25 +0800] "GET / HTTP/1.1" 200 32
::1 - - [29/Jul/2016:18:48:38 +0800] "GET /about HTTP/1.1" 200 32
::1 - - [02/Aug/2016:08:24:51 +0800] "GET / HTTP/1.1" 200 32
::1 - - [02/Aug/2016:08:24:51 +0800] "GET /favicon.ico HTTP/1.1" 200 32
::1 - - [02/Aug/2016:08:24:54 +0800] "GET / HTTP/1.1" 200 32
::1 - - [02/Aug/2016:08:24:54 +0800] "GET /favicon.ico HTTP/1.1" 200 32
::1 - - [02/Aug/2016:08:25:35 +0800] "GET /about HTTP/1.1" 200 32
::1 - - [02/Aug/2016:08:25:35 +0800] "GET /favicon.ico HTTP/1.1" 200 32

此外,還有很多的第三方庫,可以到Github查找對應(yīng)的文檔,查閱. Alice;Negroni.

  • 在中間件分享值

有時(shí)候,我們需要提供值給下一個(gè)handler或者在應(yīng)用的handler和中間件的handler共享值.比如,當(dāng)通過一個(gè)中間件的handler授權(quán)應(yīng)用的權(quán)限,就需要在請求的handler環(huán)中提供用戶的信息給下一個(gè)handler.

使用Gorilla上下文.

很多第三庫允許在請求的生命周期里保存值,用來共享,Gorilla的context包是一個(gè)非常好的選擇,用來在請求的生命周期里共享數(shù)據(jù).

使用context的例子

package main

import (
    "fmt"
    "log"
    "net/http"

    "github.com/codegangsta/negroni"
    "github.com/gorilla/context"
)

func Authorize(w http.ResponseWriter, r *http.Request, next http.HandlerFunc) {
    token := r.Header.Get("X-AppToken")
    if token == "1l9o9v0e" {
        log.Printf("Authorized to the system!")
        context.Set(r, "user", "xiaoming")
        next(w, r)
    } else {
        http.Error(w, "Not Authorized", 401)
    }
}

func index(w http.ResponseWriter, r *http.Request) {
    user := context.Get(r, "user")
    fmt.Fprintf(w, "Welcome %s!", user)
}

func main() {
    mux := http.NewServeMux()
    mux.HandleFunc("/", index)
    n := negroni.Classic()
    n.Use(negroni.HandlerFunc(Authorize))
    n.UseHandler(mux)
    n.Run(":9090")
}

一個(gè)中間件的handler被創(chuàng)建,用來驗(yàn)證HTTP請求,HTTP頭 的 "X-AppToken"從請求對象讀取的驗(yàn)證token.這也是RESTful APIs驗(yàn)證的方式之一:客戶端必須在請求頭發(fā)送一個(gè)驗(yàn)證的token.還有的方式是通過session驗(yàn)證.
在這個(gè)程序,我們想傳遞用戶名給下一個(gè)handler:應(yīng)用的handler.因此在context對象設(shè)置了user的值,這個(gè)值在請求的生命周期里面都可以訪問到.比如這里的index handler就訪問這個(gè)user的值.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末粉私,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,525評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件滨巴,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)夭咬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,203評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來铆隘,“玉大人卓舵,你說我怎么就攤上這事“蚰疲” “怎么了掏湾?”我有些...
    開封第一講書人閱讀 164,862評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長肿嘲。 經(jīng)常有香客問我融击,道長,這世上最難降的妖魔是什么雳窟? 我笑而不...
    開封第一講書人閱讀 58,728評論 1 294
  • 正文 為了忘掉前任尊浪,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘拇涤。我一直安慰自己捣作,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,743評論 6 392
  • 文/花漫 我一把揭開白布工育。 她就那樣靜靜地躺著虾宇,像睡著了一般。 火紅的嫁衣襯著肌膚如雪如绸。 梳的紋絲不亂的頭發(fā)上嘱朽,一...
    開封第一講書人閱讀 51,590評論 1 305
  • 那天,我揣著相機(jī)與錄音怔接,去河邊找鬼搪泳。 笑死,一個(gè)胖子當(dāng)著我的面吹牛扼脐,可吹牛的內(nèi)容都是我干的岸军。 我是一名探鬼主播,決...
    沈念sama閱讀 40,330評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼瓦侮,長吁一口氣:“原來是場噩夢啊……” “哼艰赞!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起肚吏,我...
    開封第一講書人閱讀 39,244評論 0 276
  • 序言:老撾萬榮一對情侶失蹤方妖,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后罚攀,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體党觅,經(jīng)...
    沈念sama閱讀 45,693評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,885評論 3 336
  • 正文 我和宋清朗相戀三年斋泄,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了杯瞻。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,001評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡炫掐,死狀恐怖魁莉,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情募胃,我是刑警寧澤沛厨,帶...
    沈念sama閱讀 35,723評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站摔认,受9級特大地震影響逆皮,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜参袱,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,343評論 3 330
  • 文/蒙蒙 一电谣、第九天 我趴在偏房一處隱蔽的房頂上張望秽梅。 院中可真熱鬧,春花似錦剿牺、人聲如沸企垦。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,919評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽钞诡。三九已至,卻和暖如春湃崩,著一層夾襖步出監(jiān)牢的瞬間荧降,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,042評論 1 270
  • 我被黑心中介騙來泰國打工攒读, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留朵诫,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,191評論 3 370
  • 正文 我出身青樓薄扁,卻偏偏與公主長得像剪返,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子邓梅,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,955評論 2 355

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,144評論 25 707
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理脱盲,服務(wù)發(fā)現(xiàn),斷路器日缨,智...
    卡卡羅2017閱讀 134,657評論 18 139
  • HTTP 中間件 HTTP中間件,在現(xiàn)實(shí)項(xiàng)目開發(fā)中簡化了很多工作,Go開發(fā)社區(qū)還沒有很大的興趣吸收成熟的web開發(fā)...
    小Q逛逛閱讀 3,859評論 0 5
  • 這篇文章是 Golang 開源庫 Negroni 的 README.md 中推薦一篇的文章钱反,講的是 Golang ...
    王諳然閱讀 7,242評論 0 6
  • 紅綠燈火,閃爍 引著漁船殿遂,回航 橙紅彎月诈铛,倒映 在墨一般的海上 伴著你的影 波光瀲滟乙各,浮動(dòng) 撩撥了誰的心海 風(fēng)輕拂...
    布小夭閱讀 244評論 0 0