SSL详解!
www.openssl.org
服務(wù)端:
客戶端:
?
Linux網(wǎng)絡(luò)編程:加密通訊協(xié)議SSL編程
服務(wù)器端代碼:
#include <stdio.h> #include <stdlib.h> #include <errno.h> #include <string.h> #include <sys/types.h> #include <netinet/in.h> #include <sys/socket.h> #include <sys/wait.h> #include <unistd.h> #include <arpa/inet.h> #include <openssl/ssl.h> #include <openssl/err.h>#define MAXBUF 1024 /************關(guān)于本文檔******************************************** *filename: ssl-server.c *purpose: 演示利用 OpenSSL 庫進(jìn)行基于 IP層的 SSL 加密通訊的方法,這是服務(wù)器端例子 *wrote by: zhoulifa(zhoulifa@163.com) 周立發(fā)(http://zhoulifa.bokee.com) Linux愛好者 Linux知識傳播者 SOHO族 開發(fā)者 最擅長C語言 *date time:2007-02-02 19:40 *Note: 任何人可以任意復(fù)制代碼并運用這些文檔,當(dāng)然包括你的商業(yè)用途 * 但請遵循GPL *Thanks to:Google *Hope:希望越來越多的人貢獻(xiàn)自己的力量,為科學(xué)技術(shù)發(fā)展出力 * 科技站在巨人的肩膀上進(jìn)步更快!感謝有開源前輩的貢獻(xiàn)! *********************************************************************/ int main(int argc, char **argv) {int sockfd, new_fd;socklen_t len;struct sockaddr_in my_addr, their_addr;unsigned int myport, lisnum;char buf[MAXBUF + 1];SSL_CTX *ctx;if (argv[1])myport = atoi(argv[1]);elsemyport = 7838;if (argv[2])lisnum = atoi(argv[2]);elselisnum = 2;/* SSL 庫初始化 */SSL_library_init();/* 載入所有 SSL 算法 */OpenSSL_add_all_algorithms();/* 載入所有 SSL 錯誤消息 */SSL_load_error_strings();/* 以 SSL V2 和 V3 標(biāo)準(zhǔn)兼容方式產(chǎn)生一個 SSL_CTX ,即 SSL Content Text */ctx = SSL_CTX_new(SSLv23_server_method());/* 也可以用 SSLv2_server_method() 或 SSLv3_server_method() 單獨表示 V2 或 V3標(biāo)準(zhǔn) */if (ctx == NULL) {ERR_print_errors_fp(stdout);exit(1);}/* 載入用戶的數(shù)字證書, 此證書用來發(fā)送給客戶端。 證書里包含有公鑰 */if (SSL_CTX_use_certificate_file(ctx, argv[4], SSL_FILETYPE_PEM) <= 0) {ERR_print_errors_fp(stdout);exit(1);}/* 載入用戶私鑰 */if (SSL_CTX_use_PrivateKey_file(ctx, argv[5], SSL_FILETYPE_PEM) <= 0) {ERR_print_errors_fp(stdout);exit(1);}/* 檢查用戶私鑰是否正確 */if (!SSL_CTX_check_private_key(ctx)) {ERR_print_errors_fp(stdout);exit(1);}/* 開啟一個 socket 監(jiān)聽 */if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {perror("socket");exit(1);} elseprintf("socket created\n");bzero(&my_addr, sizeof(my_addr));my_addr.sin_family = PF_INET;my_addr.sin_port = htons(myport);if (argv[3])my_addr.sin_addr.s_addr = inet_addr(argv[3]);elsemy_addr.sin_addr.s_addr = INADDR_ANY;if (bind(sockfd, (struct sockaddr *) &my_addr, sizeof(struct sockaddr))== -1) {perror("bind");exit(1);} elseprintf("binded\n");if (listen(sockfd, lisnum) == -1) {perror("listen");exit(1);} elseprintf("begin listen\n");while (1) {SSL *ssl;len = sizeof(struct sockaddr);/* 等待客戶端連上來 */if ((new_fd =accept(sockfd, (struct sockaddr *) &their_addr,&len)) == -1) {perror("accept");exit(errno);} elseprintf("server: got connection from %s, port %d, socket %d\n",inet_ntoa(their_addr.sin_addr),ntohs(their_addr.sin_port), new_fd);/* 基于 ctx 產(chǎn)生一個新的 SSL */ssl = SSL_new(ctx);/* 將連接用戶的 socket 加入到 SSL */SSL_set_fd(ssl, new_fd);/* 建立 SSL 連接 */if (SSL_accept(ssl) == -1) {perror("accept");close(new_fd);break;}/* 開始處理每個新連接上的數(shù)據(jù)收發(fā) */bzero(buf, MAXBUF + 1);strcpy(buf, "server->client");/* 發(fā)消息給客戶端 */len = SSL_write(ssl, buf, strlen(buf));if (len <= 0) {printf("消息'%s'發(fā)送失敗!錯誤代碼是%d,錯誤信息是'%s'\n",buf, errno, strerror(errno));goto finish;} elseprintf("消息'%s'發(fā)送成功,共發(fā)送了%d個字節(jié)!\n",buf, len);bzero(buf, MAXBUF + 1);/* 接收客戶端的消息 */len = SSL_read(ssl, buf, MAXBUF);if (len > 0)printf("接收消息成功:'%s',共%d個字節(jié)的數(shù)據(jù)\n",buf, len);elseprintf("消息接收失敗!錯誤代碼是%d,錯誤信息是'%s'\n",errno, strerror(errno));/* 處理每個新連接上的數(shù)據(jù)收發(fā)結(jié)束 */finish:/* 關(guān)閉 SSL 連接 */SSL_shutdown(ssl);/* 釋放 SSL */SSL_free(ssl);/* 關(guān)閉 socket */close(new_fd);}/* 關(guān)閉監(jiān)聽的 socket */close(sockfd);/* 釋放 CTX */SSL_CTX_free(ctx);return 0; }客戶端代碼:
#include <stdio.h> #include <string.h> #include <errno.h> #include <sys/socket.h> #include <resolv.h> #include <stdlib.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #include <openssl/ssl.h> #include <openssl/err.h>#define MAXBUF 1024void ShowCerts(SSL * ssl) {X509 *cert;char *line;cert = SSL_get_peer_certificate(ssl);if (cert != NULL) {printf("數(shù)字證書信息:\n");line = X509_NAME_oneline(X509_get_subject_name(cert), 0, 0);printf("證書: %s\n", line);free(line);line = X509_NAME_oneline(X509_get_issuer_name(cert), 0, 0);printf("頒發(fā)者: %s\n", line);free(line);X509_free(cert);} elseprintf("無證書信息!\n"); } /************關(guān)于本文檔******************************************** *filename: ssl-client.c *purpose: 演示利用 OpenSSL 庫進(jìn)行基于 IP層的 SSL 加密通訊的方法,這是客戶端例子 *wrote by: zhoulifa(zhoulifa@163.com) 周立發(fā)(http://zhoulifa.bokee.com) Linux愛好者 Linux知識傳播者 SOHO族 開發(fā)者 最擅長C語言 *date time:2007-02-02 20:10 *Note: 任何人可以任意復(fù)制代碼并運用這些文檔,當(dāng)然包括你的商業(yè)用途 * 但請遵循GPL *Thanks to:Google *Hope:希望越來越多的人貢獻(xiàn)自己的力量,為科學(xué)技術(shù)發(fā)展出力 * 科技站在巨人的肩膀上進(jìn)步更快!感謝有開源前輩的貢獻(xiàn)! *********************************************************************/ int main(int argc, char **argv) {int sockfd, len;struct sockaddr_in dest;char buffer[MAXBUF + 1];SSL_CTX *ctx;SSL *ssl;if (argc != 3) {printf("參數(shù)格式錯誤!正確用法如下:\n\t\t%s IP地址 端口\n\t比如:\t%s 127.0.0.1 80\n此程序用來從某個 IP 地址的服務(wù)器某個端口接收最多 MAXBUF 個字節(jié)的消息",argv[0], argv[0]);exit(0);}/* SSL 庫初始化,參看 ssl-server.c 代碼 */SSL_library_init();OpenSSL_add_all_algorithms();SSL_load_error_strings();ctx = SSL_CTX_new(SSLv23_client_method());if (ctx == NULL) {ERR_print_errors_fp(stdout);exit(1);}/* 創(chuàng)建一個 socket 用于 tcp 通信 */if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {perror("Socket");exit(errno);}printf("socket created\n");/* 初始化服務(wù)器端(對方)的地址和端口信息 */bzero(&dest, sizeof(dest));dest.sin_family = AF_INET;dest.sin_port = htons(atoi(argv[2]));if (inet_aton(argv[1], (struct in_addr *) &dest.sin_addr.s_addr) == 0) {perror(argv[1]);exit(errno);}printf("address created\n");/* 連接服務(wù)器 */if (connect(sockfd, (struct sockaddr *) &dest, sizeof(dest)) != 0) {perror("Connect ");exit(errno);}printf("server connected\n");/* 基于 ctx 產(chǎn)生一個新的 SSL */ssl = SSL_new(ctx);SSL_set_fd(ssl, sockfd);/* 建立 SSL 連接 */if (SSL_connect(ssl) == -1)ERR_print_errors_fp(stderr);else {printf("Connected with %s encryption\n", SSL_get_cipher(ssl));ShowCerts(ssl);}/* 接收對方發(fā)過來的消息,最多接收 MAXBUF 個字節(jié) */bzero(buffer, MAXBUF + 1);/* 接收服務(wù)器來的消息 */len = SSL_read(ssl, buffer, MAXBUF);if (len > 0)printf("接收消息成功:'%s',共%d個字節(jié)的數(shù)據(jù)\n",buffer, len);else {printf("消息接收失敗!錯誤代碼是%d,錯誤信息是'%s'\n",errno, strerror(errno));goto finish;}bzero(buffer, MAXBUF + 1);strcpy(buffer, "from client->server");/* 發(fā)消息給服務(wù)器 */len = SSL_write(ssl, buffer, strlen(buffer));if (len < 0)printf("消息'%s'發(fā)送失敗!錯誤代碼是%d,錯誤信息是'%s'\n",buffer, errno, strerror(errno));elseprintf("消息'%s'發(fā)送成功,共發(fā)送了%d個字節(jié)!\n",buffer, len);finish:/* 關(guān)閉連接 */SSL_shutdown(ssl);SSL_free(ssl);close(sockfd);SSL_CTX_free(ctx);return 0; }編譯程序用下列命令:
gcc?-Wall?ssl-client.c?-o?client
gcc?-Wall?ssl-server.c?-o?server
運行程序用如下命令:
./server?7838?1?127.0.0.1?cacert.pem?privkey.pem
./client?127.0.0.1?7838
用下面這兩個命令產(chǎn)生上述cacert.pem和privkey.pem文件:
openssl?genrsa?-out?privkey.pem?2048
openssl?req?-new?-x509?-key?privkey.pem?-out?cacert.pem?-days?1095
具體請參考?“OpenSSL體系下使用密鑰數(shù)字證書等”
如果想對SSL有更深入的了解,請學(xué)習(xí)計算機(jī)安全相關(guān)的內(nèi)容,尤其是非對稱加密技術(shù)。
如果想對SSL庫的源代碼有深入學(xué)習(xí),請去?www.openssl.org?下載源碼來閱讀。
?
總結(jié)
- 上一篇: 野指针和悬空指针的形成原因和如何避免!
- 下一篇: calloc、malloc、reallo