這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
到了新單位總算有點閑了,接著倒騰HTTP。之前的文章可以參考:《基於TCP通訊端,通過Golang類比HTTP請求》和《基於TCP通訊端,通過Golang類比HTTP請求(續)》。
之前都是研究的用戶端,現在來研究一下服務端。《HTTP權威指南》上面有一個非常簡單的用perl開發的一個伺服器,我就用大Golang照著寫一個。HTTP協議是基於傳輸層的TCP協議,監聽80連接埠。簡單來寫(複雜的我也不會),就是最簡單的TCP監聽,接收訊息,處理並返回。我是參考的《Go socket編程實踐: TCP伺服器和用戶端實現》。主邏輯就是這樣:
ln, err := net.Listen("tcp", fmt.Sprintf("%s:%d", "", *portFlag))defer ln.Close()if err != nil {panic(err)}log.Printf("<<<Server Accepting on Port %d>>>\n\n", *portFlag)for {conn, err := ln.Accept()if err != nil {log.Panicln(err)}go handleConnection(conn)}
我做東西喜歡從最簡單開始,一個是因為我會的不多,一個是簡單的來自己能理明白,以後維護其實更省事。
迴歸正題,這塊其實從念書的時候就老寫,分分鐘搞定。然後就是HTTP協議的東西。當然,還是按最簡單的來。HTTP分為三部分,起始行(start line)、首部(header)和主體(body)。
起始行只有一行,就是協議版本和狀態代碼這些,以CRLF結尾,也就是\r\n;接著就是頭,頭裡面的內容就是索引值對,每組索引值對以CRLF結尾;頭和主體中間,還需要多一個CRLF,我猜是為瞭解析內容方便。主體之後不需要CRLF。
就是詳細的請求格式。其中SP是空格。
一些必要的頭(因為我發現如果沒有,請求會失敗),就是Content-Type和Content-length。Content-length就是Body的長度。
func handleConnection(conn net.Conn) {defer conn.Close()log.Printf("[%s]<<<Request From %s>>>\n", time.Now().String(), conn.RemoteAddr())serve_time := time.Now()buffers := bytes.Buffer{}buffers.WriteString("HTTP/1.1 200 OK\r\n")buffers.WriteString("Server: Cyeam\r\n")buffers.WriteString("Date: " + serve_time.Format(time.RFC1123) + "\r\n")buffers.WriteString("Content-Type: text/html; charset=utf-8\r\n")buffers.WriteString("Content-length:" + fmt.Sprintf("%d", len(DEFAULT_HTML)) + "\r\n")buffers.WriteString("\r\n")buffers.WriteString(DEFAULT_HTML)conn.Write(buffers.Bytes())}
最後補充一點,串連記得關閉
參考文獻
- 【1】《HTTP權威指南》
- 【2】James F.Kurose, Keith W.Ross.COMPUTER NETWORKING. A Top-Down Approach Featuring the Internet(Third Edition).
- 【3】NETWORKING INFO BLOG. HTTP Message Format.
原文連結:最簡單的HTTP SERVER,轉載請註明來源!