Linux网络编程一步一步学-HTTPS客户端程序示例

本文提供了一个HTTPS客户端编程实例,展示了如何使用SSL/TLS进行安全连接,包括解析URL获取主机名和端口号、创建socket连接、初始化SSL上下文、发送HTTPS请求及接收响应。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

源代码如下:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <sys/types.h>
  5. #include <sys/socket.h>
  6. #include <errno.h>
  7. #include <unistd.h>
  8. #include <netinet/in.h>
  9. #include <limits.h>
  10. #include <netdb.h>
  11. #include <arpa/inet.h>
  12. #include <ctype.h>
  13. #include <openssl/crypto.h>
  14. #include <openssl/ssl.h>
  15. #include <openssl/err.h>
  16. #include <openssl/rand.h>
  17. #define DEBUG 1
  18. /********************************************
  19. 功能:搜索字符串右边起的第一个匹配字符
  20. ********************************************/
  21. char *Rstrchr(char *s, char x)
  22. {
  23.     int i = strlen(s);
  24.     if (!(*s))
  25.         return 0;
  26.     while (s[i - 1])
  27.         if (strchr(s + (i - 1), x))
  28.             return (s + (i - 1));
  29.         else
  30.             i--;
  31.     return 0;
  32. }
  33. /**************************************************************
  34. 功能:从字符串src中分析出网站地址和端口,并得到用户要下载的文件
  35. ***************************************************************/
  36. void GetHost(char *src, char *web, char *file, int *port)
  37. {
  38.     char *pA;
  39.     char *pB;
  40.     memset(web, 0, sizeof(web));
  41.     memset(file, 0, sizeof(file));
  42.     *port = 0;
  43.     if (!(*src))
  44.         return;
  45.     pA = src;
  46.     if (!strncmp(pA, "http://", strlen("http://")))
  47.         pA = src + strlen("http://");
  48.     else if (!strncmp(pA, "https://", strlen("https://")))
  49.         pA = src + strlen("https://");
  50.     pB = strchr(pA, '/');
  51.     if (pB) {
  52.         memcpy(web, pA, strlen(pA) - strlen(pB));
  53.         if (pB + 1) {
  54.             memcpy(file, pB + 1, strlen(pB) - 1);
  55.             file[strlen(pB) - 1] = 0;
  56.         }
  57.     } else
  58.         memcpy(web, pA, strlen(pA));
  59.     if (pB)
  60.         web[strlen(pA) - strlen(pB)] = 0;
  61.     else
  62.         web[strlen(pA)] = 0;
  63.     pA = strchr(web, ':');
  64.     if (pA)
  65.         *port = atoi(pA + 1);
  66.     else
  67.         *port = 443;
  68. }
  69. /************关于本文档********************************************
  70. *filename: https-client.c
  71. *purpose: 演示HTTPS客户端编程方法
  72. *wrote by: zhoulifa(zhoulifa@163.com) 周立发(http://zhoulifa.bokee.com)
  73. Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言
  74. *date time:2007-01-30 20:06
  75. *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
  76. * 但请遵循GPL
  77. *Thanks to:Google
  78. *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
  79. * 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
  80. *********************************************************************/
  81. int main(int argc, char *argv[])
  82. {
  83.     int sockfd, ret;
  84.     char buffer[1024];
  85.     struct sockaddr_in server_addr;
  86.     struct hostent *host;
  87.     int portnumber, nbytes;
  88.     char host_addr[256];
  89.     char host_file[1024];
  90.     char local_file[256];
  91.     FILE *fp;
  92.     char request[1024];
  93.     int send, totalsend;
  94.     int i;
  95.     char *pt;
  96.     SSL *ssl;
  97.     SSL_CTX *ctx;
  98.     if (argc != 2) {
  99.         if (DEBUG)
  100.             fprintf(stderr, "Usage:%s webpage-address/a/n", argv[0]);
  101.         exit(1);
  102.     }
  103.     if (DEBUG)
  104.         printf("parameter.1 is: %s/n", argv[1]);
  105.     GetHost(argv[1], host_addr, host_file, &portnumber);        /*分析网址、端口、文件名等 */
  106.     if (DEBUG)
  107.         printf("webhost:%s/n", host_addr);
  108.     if (DEBUG)
  109.         printf("hostfile:%s/n", host_file);
  110.     if (DEBUG)
  111.         printf("portnumber:%d/n/n", portnumber);
  112.     if ((host = gethostbyname(host_addr)) == NULL) {        /*取得主机IP地址 */
  113.         if (DEBUG)
  114.             fprintf(stderr, "Gethostname error, %s/n", strerror(errno));
  115.         exit(1);
  116.     }
  117.     /* 客户程序开始建立 sockfd描述符 */
  118.     if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {        /*建立SOCKET连接 */
  119.         if (DEBUG)
  120.             fprintf(stderr, "Socket Error:%s/a/n", strerror(errno));
  121.         exit(1);
  122.     }
  123.     /* 客户程序填充服务端的资料 */
  124.     bzero(&server_addr, sizeof(server_addr));
  125.     server_addr.sin_family = AF_INET;
  126.     server_addr.sin_port = htons(portnumber);
  127.     server_addr.sin_addr = *((struct in_addr *) host->h_addr);
  128.     /* 客户程序发起连接请求 */
  129.     if (connect(sockfd, (struct sockaddr *) (&server_addr), sizeof(struct sockaddr)) == -1) {        /*连接网站 */
  130.         if (DEBUG)
  131.             fprintf(stderr, "Connect Error:%s/a/n", strerror(errno));
  132.         exit(1);
  133.     }
  134.     /* SSL初始化 */
  135.     SSL_library_init();
  136.     SSL_load_error_strings();
  137.     ctx = SSL_CTX_new(SSLv23_client_method());
  138.     if (ctx == NULL) {
  139.         ERR_print_errors_fp(stderr);
  140.         exit(1);
  141.     }
  142.     ssl = SSL_new(ctx);
  143.     if (ssl == NULL) {
  144.         ERR_print_errors_fp(stderr);
  145.         exit(1);
  146.     }
  147.     /* 把socket和SSL关联 */
  148.     ret = SSL_set_fd(ssl, sockfd);
  149.     if (ret == 0) {
  150.         ERR_print_errors_fp(stderr);
  151.         exit(1);
  152.     }
  153.     RAND_poll();
  154.     while (RAND_status() == 0) {
  155.         unsigned short rand_ret = rand() % 65536;
  156.         RAND_seed(&rand_ret, sizeof(rand_ret));
  157.     }
  158.     ret = SSL_connect(ssl);
  159.     if (ret != 1) {
  160.         ERR_print_errors_fp(stderr);
  161.         exit(1);
  162.     }
  163.     sprintf(request, "GET /%s HTTP/1.1/r/nAccept: */*/r/nAccept-Language: zh-cn/r/n/
  164. User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)/r/n/
  165. Host: %s:%d/r/nConnection: Close/r/n/r/n", host_file, host_addr,
  166.             portnumber);
  167.     if (DEBUG)
  168.         printf("%s", request);        /*准备request,将要发送给主机 */
  169.     /*取得真实的文件名 */
  170.     if (host_file && *host_file)
  171.         pt = Rstrchr(host_file, '/');
  172.     else
  173.         pt = 0;
  174.     memset(local_file, 0, sizeof(local_file));
  175.     if (pt && *pt) {
  176.         if ((pt + 1) && *(pt + 1))
  177.             strcpy(local_file, pt + 1);
  178.         else
  179.             memcpy(local_file, host_file, strlen(host_file) - 1);
  180.     } else if (host_file && *host_file)
  181.         strcpy(local_file, host_file);
  182.     else
  183.         strcpy(local_file, "index.html");
  184.     if (DEBUG)
  185.         printf("local filename to write:%s/n/n", local_file);
  186.     /*发送https请求request */
  187.     send = 0;
  188.     totalsend = 0;
  189.     nbytes = strlen(request);
  190.     while (totalsend < nbytes) {
  191.         send = SSL_write(ssl, request + totalsend, nbytes - totalsend);
  192.         if (send == -1) {
  193.             if (DEBUG)
  194.                 ERR_print_errors_fp(stderr);
  195.             exit(0);
  196.         }
  197.         totalsend += send;
  198.         if (DEBUG)
  199.             printf("%d bytes send OK!/n", totalsend);
  200.     }
  201.     fp = fopen(local_file, "a");
  202.     if (!fp) {
  203.         if (DEBUG)
  204.             printf("create file error! %s/n", strerror(errno));
  205.         return 0;
  206.     }
  207.     if (DEBUG)
  208.         printf("/nThe following is the response header:/n");
  209.     i = 0;
  210.     /* 连接成功了,接收https响应,response */
  211.     while ((nbytes = SSL_read(ssl, buffer, 1)) == 1) {
  212.         if (i < 4) {
  213.             if (buffer[0] == '/r' || buffer[0] == '/n')
  214.                 i++;
  215.             else
  216.                 i = 0;
  217.             if (DEBUG)
  218.                 printf("%c", buffer[0]);        /*把https头信息打印在屏幕上 */
  219.         } else {
  220.             fwrite(buffer, 1, 1, fp);        /*将https主体信息写入文件 */
  221.             i++;
  222.             if (i % 1024 == 0)
  223.                 fflush(fp);        /*每1K时存盘一次 */
  224.         }
  225.     }
  226.     fclose(fp);
  227.     /* 结束通讯 */
  228.     ret = SSL_shutdown(ssl);
  229.     if (ret != 1) {
  230.         ERR_print_errors_fp(stderr);
  231.         exit(1);
  232.     }
  233.     close(sockfd);
  234.     SSL_free(ssl);
  235.     SSL_CTX_free(ctx);
  236.     ERR_free_strings();
  237.     exit(0);
  238. }
编译此程序用下列命令:
gcc -Wall https-client.c -lssl -o httpsclient

运行此程序来取得HTTPS服务器上的页面,比如:
./httpsclient https://127.0.0.1/test.html

关键之处在于建立socket之后的SSL相关初始化以及中间的recv/send用SSL_read和SSL_write代替,最后记得释放SSL资源即可。
可以对比之前的文章来发现异同:

HTTP协议的C语言编程实现实例

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值