1. 程式人生 > >c語言實現簡單web伺服器

c語言實現簡單web伺服器

浪費了“黃金五年”的Java程式設計師,還有救嗎? >>>   

c語言實現簡單web伺服器

1http簡單介紹

http超文字傳輸協議:host主機地址:port埠/url

host會被DNS伺服器 解析成IP地址,所以有時候可以直接用域名,

http預設訪問80埠,https預設訪問443埠

 

大致流程就是:瀏覽器輸入地址後,首先和web伺服器建立tcp連線,

然後瀏覽器傳送http請求報文, web伺服器響應處理這個報文,

然後給他回覆一個響應,然後伺服器主動斷開連線。

 

 

 

 

2http請求報文格式

 

首先第一個就是請求的方法,方法有一下這些:
GET,POST,HEAD,PUT,DELETE,OPTIONS,TRACE,CONNECT;

 

1GET

在瀏覽器輸入的網址,瀏覽器就會發送GET的http報文請求。

如果不寫url 預設就是 "/"  伺服器 可根據這個響應對應的頁面.

頭部資訊就包含一些重要的請求資訊,如主機地址.

瀏覽器版本 , 手機的頁面就是根據這個去做的.

GET 攜帶引數是在url裡面的, POST是攜帶在包體裡面的.

包體成為body,請求頭部叫做head。

GET傳遞引數,格式  /url?username=xxx&passwd=bbb  通過問號解析引數部分

url的傳遞引數是有限制的,每個瀏覽器限制都不一樣。url不允許有回車換行

 

 

2POST 

POST也是一個請求操作,他的資料引數攜帶在http請求的body裡面。

所有的引數都不允許有回車換行的存在, 很多時候如果必須要攜帶

回車換行的話,必須先把資料轉換成base64編碼,因為它沒有回車換成.他是解決網路傳輸的常用方法。

 

 

 

 

3http響應報文格式

 

1狀態碼:請求是否成功,狀態碼描述:成功或失敗的原因

有時候訪問一網頁 會出現404,這個404就是這個狀態碼.

 

 

 

4http資料傳輸模式

1傳輸中兩個重要的引數: 寫在頭裡面

transfer-encoding:identity,chunked表示當前這個body是什麼協議發過來的

content-length:length:資料包的長度

 

 

2identity 直接傳送模式,length在後面表示資料長度

 

3chunked 模式,後面跟的是每一個chunk包 [包頭,包資料]

包頭:第一個位元組表示一個ASSIC資料,第二個位元組也是ASSIC資料

兩個位元組加起來,組成一個16進位制資料。

後面兩個位元組,固定的0d0a(回車換行符)兩個位元組這4個位元組就是一個chunk的包頭,

後面的資料包 根據前面的兩個位元組來決定。資料包的結束標誌是

30  0d  0a  30ascll碼代表的是0

也就是說chunk包的結束:是遇到一個為等於0的chunk結束。

然後把這個包整合一下,形成完整的資料。

 

 

 

 

5http狀態碼和表

 

/*

{

[100] = "Continue",

[101] = "Switching Protocols",

[200] = "OK",

[201] = "Created",

[202] = "Accepted",

[203] = "Non-Authoritative Information",

[204] = "No Content",

[205] = "Reset Content",

[206] = "Partial Content",

[300] = "Multiple Choices",

[301] = "Moved Permanently",

[302] = "Found",

[303] = "See Other",

[304] = "Not Modified",

[305] = "Use Proxy",

[307] = "Temporary Redirect",

[400] = "Bad Request",

[401] = "Unauthorized",

[402] = "Payment Required",

[403] = "Forbidden",

[404] = "Not Found",

[405] = "Method Not Allowed",

[406] = "Not Acceptable",

[407] = "Proxy Authentication Required",

[408] = "Request Time-out",

[409] = "Conflict",

[410] = "Gone",

[411] = "Length Required",

[412] = "Precondition Failed",

[413] = "Request Entity Too Large",

[414] = "Request-URI Too Large",

[415] = "Unsupported Media Type",

[416] = "Requested range not satisfiable",

[417] = "Expectation Failed",

[500] = "Internal Server Error",

[501] = "Not Implemented",

[502] = "Bad Gateway",

[503] = "Service Unavailable",

[504] = "Gateway Time-out",

[505] = "HTTP Version not supported",

}

*/

 

 

 

 

6使用http_parmer解析 URL讀取 然後返回給客戶端

 

//解析我們的http報文
http_parser p;
http_parser_init(&p,HTTP_REQUEST);
http_parser_settings s;
http_parser_settings_init(&s);
//解析到了url 回撥
s.on_url = ws_on_url;
//重置解析資訊
init_ws_params();
// 解析器執行。返回解析的位元組數
http_parser_execute(&p,&s,http_req,len);
//設定回撥函式
switch (p.method)  //報文響應的方式
{
case HTTP_GET:
{
int len_URL = filter_url(WS_HTTP.url);
//訪問的網頁  
if (strncmp("/", WS_HTTP.url, len_URL) == 0){ //訪問test這個html
strncpy(WS_HTTP.url, "www_root/index.html", strlen("www_root/index.html"));
}
else if (strncmp("/test", WS_HTTP.url, len_URL) == 0){ //訪問預設的url 根目錄
strncpy(WS_HTTP.url, "www_root/test.html", strlen("www_root/test.html"));
}
char* file_data = open_files(WS_HTTP.url);
//傳送報文  也就是響應客戶端
//釋放記憶體
free(file_data);
}
break;
case HTTP_POST:
break;
}
//end
printf("\n");


讀取檔案
static char* 
open_files(char* filename)
{
	//讀取這個檔案
	FILE* f = fopen(filename,"rb");
	//檔案大小
	int file_size = 0;
	fseek(f,0,SEEK_END);
	file_size = ftell(f);
	//指標又到檔案頭
	fseek(f, 0, 0);

	char* file_data = malloc(file_size + 1);
	fread(file_data, file_size,1, f);
	file_data[file_size] = 0;


	fclose(f);
	return file_data;
	
}


 

 

7響應請求報文

 

 

//使用identity來發送響應報文 
static void
write_ok_identity(int sock, char* body)
{
int len = strlen(body);
//使用直接模式 transfer-encoding:identity
char* send_line = malloc(len + 8096);
memset(send_line, 0, sizeof(send_line));
//迴應http的頭
sprintf(send_line, "HTTP/1.1  %d %s\r\n", 200, "OK");
//設定頭部的一些資訊  比如傳送模式 body的長度
char* walk = send_line;
//跳過這個頭部
walk = walk + strlen(walk);
sprintf(walk,"transfer-encoding:%s\r\n","identity");
//body的長度
walk = walk + strlen(walk);  //頭結束
sprintf(walk, "content-length: %d\r\n\r\n0", len);
//寫入資料部分
walk = walk + strlen(walk);
sprintf(walk, "%s", body);
//傳送報文 響應  給客戶端
send(sock, send_line, strlen(send_line), 0);
free(send_line);
walk = NULL;
}

 

 

 

 

本文原作者超級極客,原文連結http://blog.51cto.com/12158490/2059931

標籤: c++ HTTP

宣告:本文內容由網際網路使用者自發貢獻自行上傳,本網站不擁有所有權,未作人工編輯處理,也不承擔相關法律責任。如果您發現有涉嫌版權的內容,歡迎傳送郵件至:[email protected] 進行舉報,並提供相關證據,工作人員會在5個工作日內聯絡你,一經查實,本站將立刻刪除