libevent和libcurl實現http和https伺服器 cJSON使用
前言
libevent和libcurl都是功能強大的開源庫;libevent主要實現伺服器,包含了select、epoll等高併發的實現;libcurl實現了curl命令的API封裝,主要作為客戶端。這兩個開源庫的安裝可以參考我的這篇部落格:https://www.cnblogs.com/liudw-0215/p/9917422.html,並且我的程式碼都提交在了我的github上了,可以點左上角圖示,跳轉到github,倉庫是libcurl。
一、curl的兩種使用方法
1、命令列模式
所謂命令列模式,就是直接linux的命令列直接可以執行的curl命令,curl可以做很多事情,我主要介紹作為客戶端傳送xml和json資料,因為命令列模式非常要注意格式問題!
(1)傳送xml格式資料
格式如下:
echo '<?xml version="1.0" encoding="UTF-8"?> <soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:itsm="http://itsm.soa.csg.cn/"> <soapenv:Header xmlns:auth="http://itsm.soa.csg.cn/"> <auth:user>local_admin</auth:user> <auth:password>local_admin</auth:password> </soapenv:Header> <soapenv:Body> <itsm:accountOper> <operType>1</operType> <operItems> <operItem> <deviceName>測試虛擬機器181106</deviceName> <deviceIP>11.11.22.23</deviceIP> <protocol>裝置帳戶</protocol> <accountName>administrator</accountName> </operItem> </operItems> </itsm:accountOper> </soapenv:Body> </soapenv:Envelope> '|curl -X POST -H 'Content-type:text/xml' -d @- http://10.94.1.167:80/ITSMWebServer/itsm
說明:
-
- echo後面跟的是xml格式資料,格式一般都是跟第三方平臺約定好的,不能發這種格式,接收又是另一種格式,那沒法解析了,都要提前約定好的!
- 中間是“|”管道符,將echo的輸出作為curl的輸入
- POST 說明是post請求
- -H 攜帶的訊息頭
- 最後的url,是要傳送的地址
(2)傳送json格式資料
格式如下:
curl -H "Content-Type:application/json" -H "appName:spvas" -H "password:123123" -H "pswdHashType:SHA1" -X POST -k -g -d '{"param":[{"objectID":112,"type":1,"operate":1,"operatorID":100,"result":0,"time":1539941168,"policytype":0}]}' http://172.16.1.21:9999/rest/spvas/objChange.do
說明:
- -H 依然是訊息頭
- -d 後面是json格式的資料了
2、libcurl庫使用
1、安裝
想要使用libcurl庫,首先需要先安裝,安裝參考我的這篇部落格寫的很詳細:https://www.cnblogs.com/liudw-0215/p/9917422.html
2、使用libcurl的API
主要就是呼叫libcurl庫的API介面,下面介紹的http的POST請求,libcurl很多介面,不能一一介紹,需要時可以再去查詢。
(1)初始化curl控制代碼
CURL* curl = NULL;
curl = curl_easy_init();
(2)設定curl的url
curl_easy_setopt(curl, CURLOPT_URL, "http://172.16.1.96:7777/login");
(3)開啟post請求開關
curl_easy_setopt(curl, CURLOPT_POST, true);
(4)新增post資料
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, post_str);
(5)設定一個處理伺服器響應的回撥函式
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, deal_response);
(6)給回撥函式傳遞一個形參
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &responseData);
(7)向伺服器傳送請求,等待伺服器的響應
res = curl_easy_perform(curl);
3、總體程式碼
客戶端總體程式碼如下:
// // Created by ldw on 2018/11/8. // #include "cJSON.h" #include <curl/curl.h> #include<string.h> #define RESPONSE_DATA_LEN 4096 //用來接收伺服器一個buffer typedef struct login_response_data { login_response_data() { memset(data, 0, RESPONSE_DATA_LEN); data_len = 0; } char data[RESPONSE_DATA_LEN]; int data_len; }response_data_t; //處理從伺服器返回的資料,將資料拷貝到arg中 size_t deal_response(void *ptr, size_t n, size_t m, void *arg) { int count = m*n; response_data_t *response_data = (response_data_t*)arg; memcpy(response_data->data, ptr, count); response_data->data_len = count; return response_data->data_len; } #define POSTDATA "{\"username\":\"gailun\",\"password\":\"123123\",\"driver\":\"yes\"}" int main() { char *post_str = NULL; CURL* curl = NULL; CURLcode res; response_data_t responseData;//專門用來存放從伺服器返回的資料 //初始化curl控制代碼 curl = curl_easy_init(); if(curl == NULL) { return 1; } //封裝一個數據協議 /* ====給服務端的協議==== http://ip:port/login [json_data] { username: "gailun", password: "123123", driver: "yes" } * * * */ //(1)封裝一個json字串 cJSON *root = cJSON_CreateObject(); cJSON_AddStringToObject(root, "username", "ldw"); cJSON_AddStringToObject(root, "password", "123123"); cJSON_AddStringToObject(root, "driver", "yes"); post_str = cJSON_Print(root); cJSON_Delete(root); root = NULL; //(2) 向web伺服器 傳送http請求 其中post資料 json字串 //1 設定curl url curl_easy_setopt(curl, CURLOPT_URL, "http://172.16.1.96:7777/login"); //客戶端忽略CA證書認證 用於https跳過證書認證 curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, false); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, false); //2 開啟post請求開關 curl_easy_setopt(curl, CURLOPT_POST, true); //3 新增post資料 curl_easy_setopt(curl, CURLOPT_POSTFIELDS, post_str); //4 設定一個處理伺服器響應的回撥函式 curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, deal_response); //5 給回撥函式傳遞一個形參 curl_easy_setopt(curl, CURLOPT_WRITEDATA, &responseData); //6 向伺服器傳送請求,等待伺服器的響應 res = curl_easy_perform(curl); if (res != CURLE_OK) { return 1; } curl_easy_cleanup(curl); //(3) 處理伺服器響應的資料 此刻的responseData就是從伺服器獲取的資料 /* //成功 { result: "ok", } //失敗 { result: "error", reason: "why...." } * * */ //(4) 解析伺服器返回的json字串 //cJSON *root; root = cJSON_Parse(responseData.data); cJSON *result = cJSON_GetObjectItem(root, "result"); if(result && strcmp(result->valuestring, "ok") == 0) { printf("data:%s\n",responseData.data); //登陸成功 return 0; } else { //登陸失敗 cJSON* reason = cJSON_GetObjectItem(root, "reason"); if (reason) { //已知錯誤 return 1; } else { //未知的錯誤 return 1; } return 1; } return 0; }View Code
這是客戶端的總體程式碼,但是還無法測試,因為沒有服務端,下面會介紹用libevent庫來搭建http的服務端;因為資料格式是json,所以用到了cJSON,可以到我的github上進行下載,編譯命令:g++ login.cpp cJSON.cpp -o login -lcurl
二、libevent庫
1、安裝
libevent依然是開源庫,使用之前依然需要安裝,安裝參考我的這篇部落格寫的很詳細:https://www.cnblogs.com/liudw-0215/p/9917422.html
2、搭建http伺服器
安裝之後,就可以使用了,主要都是呼叫libcurl庫的API函式,main函式如下:
int main(int argc, char *argv[]) { //自定義訊號處理函式 signal(SIGHUP, signal_handler); signal(SIGTERM, signal_handler); signal(SIGINT, signal_handler); signal(SIGQUIT, signal_handler); //預設引數 char *httpd_option_listen = "0.0.0.0"; int httpd_option_port = 7777; int httpd_option_daemon = 0; int httpd_option_timeout = 120; //in seconds //獲取引數 int c; while ((c = getopt(argc, argv, "l:p:dt:h")) != -1) { switch (c) { case 'l' : httpd_option_listen = optarg; break; case 'p' : httpd_option_port = atoi(optarg); break; case 'd' : httpd_option_daemon = 1; break; case 't' : httpd_option_timeout = atoi(optarg); break; case 'h' : default : show_help(); exit(EXIT_SUCCESS); } } //判斷是否設定了-d,以daemon執行 if (httpd_option_daemon) { pid_t pid; pid = fork(); if (pid < 0) { perror("fork failed"); exit(EXIT_FAILURE); } if (pid > 0) { //生成子程序成功,退出父程序 exit(EXIT_SUCCESS); } } /* 使用libevent建立HTTP Server */ //初始化event API event_init(); //建立一個http server struct evhttp *httpd; httpd = evhttp_start(httpd_option_listen, httpd_option_port); evhttp_set_timeout(httpd, httpd_option_timeout); //也可以為特定的URI指定callback evhttp_set_cb(httpd, "/", httpd_handler, NULL); evhttp_set_cb(httpd, "/login", login_handler, NULL); //迴圈處理events event_dispatch(); evhttp_free(httpd); return 0; }
3、測試http服務
- 啟動服務端
從我的github上下載之後,http服務在libcurl/http_server/這個目錄,寫Makefile,然後直接make就可以了,如下:
make之後生成了server,執行:./server,啟動服務
- 啟動客戶端
在libcurl/login/這個目錄,執行:g++ login.cpp cJSON.cpp -o login -lcurl,進行編譯,生成login,啟動客戶端:./login,客戶端執行結果,如下:
服務端響應結果,如下:
至此,完成了演示,用libcurl和libevent搭建的http伺服器與客戶端,沒有問題。是不是覺得到此就結束了,才沒有呢?下面,將要介紹https伺服器,那為什麼要用https伺服器呢?跟隨我找到謎底吧!
4、搭建https伺服器
(1)https介紹
http傳輸過程都是明文傳輸,很不安全;就產生https,進行加密傳輸,但加密過程並沒有那麼簡單,如下圖所示:
說明:
主要經歷了兩個階段:
- 非對稱加密過程
通過公鑰、私鑰和CA證書,進行驗證,最終獲得會話金鑰
- 對稱加密過程
可能會想?直接都用非對稱加密得了,為啥用對稱加密?因為非對稱效率很低,所以要用對稱加密!
用非對稱過程得到的金鑰,對資料進行加密然後傳輸。
(2)https伺服器實現
libevent庫應該從2.1版本之後才支援https的,所以在2.1之前的版本還要單獨安裝openssl!
mian函式如下:
int main (int argc, char **argv) { /*OpenSSL 初始化 */ common_setup (); if (argc > 1) { char *end_ptr; long lp = strtol(argv[1], &end_ptr, 0); if (*end_ptr) { fprintf(stderr, "Invalid integer\n"); return -1; } if (lp <= 0) { fprintf(stderr, "Port must be positive\n"); return -1; } if (lp >= USHRT_MAX) { fprintf(stderr, "Port must fit 16-bit range\n"); return -1; } serverPort = (unsigned short)lp; } /* now run http server (never returns) */ return serve_some_http (); }
(3)測試https伺服器
- 啟動服務端
從我的github上下載之後,http服務在libcurl/https_server/這個目錄,寫Makefile,然後直接make就可以了;
- 啟動客戶端
修改http的客戶端就可以了,如下:
curl_easy_setopt(curl, CURLOPT_URL, "https://172.16.1.96:8080/login"); //客戶端忽略CA證書認證 用於https跳過證書認證 curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, false); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, false);
說明:
在http後面加上“s”;再加上跳過證書認證,就可以了