• openssl实现双向认证教程(服务端代码+客户端代码+证书生成)


    参考链接

    注意事项

    •  openssl版本差异很可能导致程序编译与运行出现问题
    • 本程序在OpenSSL 1.1.1  11 Sep 2018 版本下执行编译没有问题

    创建目录

    • 目录结构如下所示
    • 创建目录使用命令mkdir
    • 创建文件使用命令touch

    路径说明

    • client路径:/home/chy-cpabe/ssl_server_client/client/pem
    • server路径:/home/chy-cpabe/ssl_server_client/server/pem
    • ca路径:/home/chy-cpabe/ssl_server_client/ca 

    生成证书

    生成ca证书

    1. # CA证书及密钥生成方法一----直接生成CA密钥及其自签名证书
    2. openssl req -newkey rsa:2048 -passout pass:123456 -keyout ca_rsa_private.pem -x509 -days 365 -out ca.crt -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=CA/emailAddress=ca_email@qq.com"

    Server 

     生成server证书

    1. # 服务器证书及密钥生成方法一----直接生成服务器密钥及待签名证书
    2. openssl req -newkey rsa:2048 -passout pass:server -keyout server_rsa_private.pem -out server.csr -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=SERVER/emailAddress=server_email@qq.com"

    对server证书进行签名

    1. # 使用CA证书及密钥对服务器证书进行签名:
    2. openssl x509 -req -days 365 -in server.csr -CA /home/chy-cpabe/ssl_server_client/ca/ca.crt -CAkey /home/chy-cpabe/ssl_server_client/ca/ca_rsa_private.pem -passin pass:123456 -CAcreateserial -out server.crt

     生成未加密的server的密钥

    1. # 将加密的RSA密钥转成未加密的RSA密钥,避免每次读取都要求输入解密密码
    2. # 密码就是生成私钥文件时设置的passout、读取私钥文件时要输入的passin,比如这里要输入“server”
    3. openssl rsa -in server_rsa_private.pem -out server_rsa_private.pem.unsecure

     Client

    生成client证书

    1. # 客户端证书及密钥生成方法一----直接生成客户端密钥及待签名证书
    2. openssl req -newkey rsa:2048 -passout pass:client -keyout client_rsa_private.pem -out client.csr -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=CLIENT/emailAddress=client_email@qq.com"

    对client证书进行签名 

    1. # 使用CA证书及密钥对客户端证书进行签名:
    2. openssl x509 -req -days 365 -in client.csr -CA /home/chy-cpabe/ssl_server_client/ca/ca.crt -CAkey /home/chy-cpabe/ssl_server_client/ca/ca_rsa_private.pem -passin pass:123456 -CAcreateserial -out client.crt

      生成未加密的client的密钥

    1. # 将加密的RSA密钥转成未加密的RSA密钥,避免每次读取都要求输入解密密码
    2. # 密码就是生成私钥文件时设置的passout、读取私钥文件时要输入的passin,比如这里要输入“client”
    3. openssl rsa -in client_rsa_private.pem -out client_rsa_private.pem.unsecure

    程序

    • 注意事项:ca的证书需要程序内部指定,server和client的证书通过形参进行传递

    server端程序

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. #include
    10. #include
    11. #include
    12. #include
    13. #define MAXBUF 1024
    14. void ShowCerts(SSL * ssl)
    15. {
    16. X509 *cert;
    17. char *line;
    18. cert = SSL_get_peer_certificate(ssl);
    19. // SSL_get_verify_result()是重点,SSL_CTX_set_verify()只是配置启不启用并没有执行认证,调用该函数才会真证进行证书认证
    20. // 如果验证不通过,那么程序抛出异常中止连接
    21. if(SSL_get_verify_result(ssl) == X509_V_OK){
    22. printf("证书验证通过\n");
    23. }
    24. if (cert != NULL) {
    25. printf("数字证书信息:\n");
    26. line = X509_NAME_oneline(X509_get_subject_name(cert), 0, 0);
    27. printf("证书: %s\n", line);
    28. free(line);
    29. line = X509_NAME_oneline(X509_get_issuer_name(cert), 0, 0);
    30. printf("颁发者: %s\n", line);
    31. free(line);
    32. X509_free(cert);
    33. } else
    34. printf("无证书信息!\n");
    35. }
    36. int main(int argc, char **argv) {
    37. int sockfd, new_fd;
    38. socklen_t len;
    39. struct sockaddr_in my_addr, their_addr;
    40. unsigned int myport, lisnum;
    41. char buf[MAXBUF + 1];
    42. SSL_CTX *ctx;
    43. if (argv[1])
    44. myport = atoi(argv[1]);
    45. else
    46. myport = 7838;
    47. if (argv[2])
    48. lisnum = atoi(argv[2]);
    49. else
    50. lisnum = 2;
    51. /* SSL 库初始化 */
    52. SSL_library_init();
    53. /* 载入所有 SSL 算法 */
    54. OpenSSL_add_all_algorithms();
    55. /* 载入所有 SSL 错误消息 */
    56. SSL_load_error_strings();
    57. /* 以 SSL V2 和 V3 标准兼容方式产生一个 SSL_CTX ,即 SSL Content Text */
    58. ctx = SSL_CTX_new(SSLv23_server_method());
    59. /* 也可以用 SSLv2_server_method() 或 SSLv3_server_method() 单独表示 V2 或 V3标准 */
    60. if (ctx == NULL) {
    61. ERR_print_errors_fp(stdout);
    62. exit(1);
    63. }
    64. // 双向验证
    65. // SSL_VERIFY_PEER---要求对证书进行认证,没有证书也会放行
    66. // SSL_VERIFY_FAIL_IF_NO_PEER_CERT---要求客户端需要提供证书,但验证发现单独使用没有证书也会放行
    67. SSL_CTX_set_verify(ctx, SSL_VERIFY_PEER|SSL_VERIFY_FAIL_IF_NO_PEER_CERT, NULL);
    68. // 设置信任根证书
    69. if (SSL_CTX_load_verify_locations(ctx, "/home/chy-cpabe/ssl_server_client/ca/ca.crt",NULL)<=0){
    70. ERR_print_errors_fp(stdout);
    71. exit(1);
    72. }
    73. /* 载入用户的数字证书, 此证书用来发送给客户端。 证书里包含有公钥 */
    74. if (SSL_CTX_use_certificate_file(ctx, argv[3], SSL_FILETYPE_PEM) <= 0) {
    75. ERR_print_errors_fp(stdout);
    76. exit(1);
    77. }
    78. /* 载入用户私钥 */
    79. if (SSL_CTX_use_PrivateKey_file(ctx, argv[4], SSL_FILETYPE_PEM) <= 0) {
    80. ERR_print_errors_fp(stdout);
    81. exit(1);
    82. }
    83. /* 检查用户私钥是否正确 */
    84. if (!SSL_CTX_check_private_key(ctx)) {
    85. ERR_print_errors_fp(stdout);
    86. exit(1);
    87. }
    88. /* 开启一个 socket 监听 */
    89. if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
    90. perror("socket");
    91. exit(1);
    92. } else
    93. printf("socket created\n");
    94. bzero(&my_addr, sizeof(my_addr));
    95. my_addr.sin_family = PF_INET;
    96. my_addr.sin_port = htons(myport);
    97. my_addr.sin_addr.s_addr = INADDR_ANY;
    98. if (bind(sockfd, (struct sockaddr *) &my_addr, sizeof(struct sockaddr))
    99. == -1) {
    100. perror("bind");
    101. exit(1);
    102. } else
    103. printf("binded\n");
    104. if (listen(sockfd, lisnum) == -1) {
    105. perror("listen");
    106. exit(1);
    107. } else
    108. printf("begin listen\n");
    109. while (1) {
    110. SSL *ssl;
    111. len = sizeof(struct sockaddr);
    112. /* 等待客户端连上来 */
    113. if ((new_fd = accept(sockfd, (struct sockaddr *) &their_addr, &len))
    114. == -1) {
    115. perror("accept");
    116. exit(errno);
    117. } else
    118. printf("server: got connection from %s, port %d, socket %d\n",
    119. inet_ntoa(their_addr.sin_addr), ntohs(their_addr.sin_port),
    120. new_fd);
    121. /* 基于 ctx 产生一个新的 SSL */
    122. ssl = SSL_new(ctx);
    123. /* 将连接用户的 socket 加入到 SSL */
    124. SSL_set_fd(ssl, new_fd);
    125. /* 建立 SSL 连接 */
    126. if (SSL_accept(ssl) == -1) {
    127. perror("accept");
    128. close(new_fd);
    129. break;
    130. }
    131. ShowCerts(ssl);
    132. /* 开始处理每个新连接上的数据收发 */
    133. bzero(buf, MAXBUF + 1);
    134. strcpy(buf, "server->client");
    135. /* 发消息给客户端 */
    136. len = SSL_write(ssl, buf, strlen(buf));
    137. if (len <= 0) {
    138. printf("消息'%s'发送失败!错误代码是%d,错误信息是'%s'\n", buf, errno,
    139. strerror(errno));
    140. goto finish;
    141. } else
    142. printf("消息'%s'发送成功,共发送了%d个字节!\n", buf, len);
    143. bzero(buf, MAXBUF + 1);
    144. /* 接收客户端的消息 */
    145. len = SSL_read(ssl, buf, MAXBUF);
    146. if (len > 0)
    147. printf("接收消息成功:'%s',共%d个字节的数据\n", buf, len);
    148. else
    149. printf("消息接收失败!错误代码是%d,错误信息是'%s'\n",
    150. errno, strerror(errno));
    151. /* 处理每个新连接上的数据收发结束 */
    152. finish:
    153. /* 关闭 SSL 连接 */
    154. SSL_shutdown(ssl);
    155. /* 释放 SSL */
    156. SSL_free(ssl);
    157. /* 关闭 socket */
    158. close(new_fd);
    159. }
    160. /* 关闭监听的 socket */
    161. close(sockfd);
    162. /* 释放 CTX */
    163. SSL_CTX_free(ctx);
    164. return 0;
    165. }

    client程序

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. #include
    10. #include
    11. #include
    12. #define MAXBUF 1024
    13. void ShowCerts(SSL * ssl)
    14. {
    15. X509 *cert;
    16. char *line;
    17. cert = SSL_get_peer_certificate(ssl);
    18. // SSL_get_verify_result()是重点,SSL_CTX_set_verify()只是配置启不启用并没有执行认证,调用该函数才会真证进行证书认证
    19. // 如果验证不通过,那么程序抛出异常中止连接
    20. if(SSL_get_verify_result(ssl) == X509_V_OK){
    21. printf("证书验证通过\n");
    22. }
    23. if (cert != NULL) {
    24. printf("数字证书信息:\n");
    25. line = X509_NAME_oneline(X509_get_subject_name(cert), 0, 0);
    26. printf("证书: %s\n", line);
    27. free(line);
    28. line = X509_NAME_oneline(X509_get_issuer_name(cert), 0, 0);
    29. printf("颁发者: %s\n", line);
    30. free(line);
    31. X509_free(cert);
    32. } else
    33. printf("无证书信息!\n");
    34. }
    35. int main(int argc, char **argv)
    36. {
    37. int sockfd, len;
    38. struct sockaddr_in dest;
    39. char buffer[MAXBUF + 1];
    40. SSL_CTX *ctx;
    41. SSL *ssl;
    42. if (argc != 5) {
    43. printf("参数格式错误!正确用法如下:\n\t\t%s IP地址 端口\n\t比如:\t%s 127.0.0.1 80\n此程序用来从某个"
    44. "IP 地址的服务器某个端口接收最多 MAXBUF 个字节的消息",
    45. argv[0], argv[0]);
    46. exit(0);
    47. }
    48. /* SSL 库初始化,参看 ssl-server.c 代码 */
    49. SSL_library_init();
    50. OpenSSL_add_all_algorithms();
    51. SSL_load_error_strings();
    52. ctx = SSL_CTX_new(SSLv23_client_method());
    53. if (ctx == NULL) {
    54. ERR_print_errors_fp(stdout);
    55. exit(1);
    56. }
    57. // 双向验证
    58. // SSL_VERIFY_PEER---要求对证书进行认证,没有证书也会放行
    59. // SSL_VERIFY_FAIL_IF_NO_PEER_CERT---要求客户端需要提供证书,但验证发现单独使用没有证书也会放行
    60. SSL_CTX_set_verify(ctx, SSL_VERIFY_PEER|SSL_VERIFY_FAIL_IF_NO_PEER_CERT, NULL);
    61. // 设置信任根证书
    62. if (SSL_CTX_load_verify_locations(ctx, "/home/chy-cpabe/ssl_server_client/ca/ca.crt",NULL)<=0){
    63. ERR_print_errors_fp(stdout);
    64. exit(1);
    65. }
    66. /* 载入用户的数字证书, 此证书用来发送给客户端。 证书里包含有公钥 */
    67. if (SSL_CTX_use_certificate_file(ctx, argv[3], SSL_FILETYPE_PEM) <= 0) {
    68. ERR_print_errors_fp(stdout);
    69. exit(1);
    70. }
    71. /* 载入用户私钥 */
    72. if (SSL_CTX_use_PrivateKey_file(ctx, argv[4], SSL_FILETYPE_PEM) <= 0) {
    73. ERR_print_errors_fp(stdout);
    74. exit(1);
    75. }
    76. /* 检查用户私钥是否正确 */
    77. if (!SSL_CTX_check_private_key(ctx)) {
    78. ERR_print_errors_fp(stdout);
    79. exit(1);
    80. }
    81. /* 创建一个 socket 用于 tcp 通信 */
    82. if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
    83. perror("Socket");
    84. exit(errno);
    85. }
    86. printf("socket created\n");
    87. /* 初始化服务器端(对方)的地址和端口信息 */
    88. bzero(&dest, sizeof(dest));
    89. dest.sin_family = AF_INET;
    90. dest.sin_port = htons(atoi(argv[2]));
    91. if (inet_aton(argv[1], (struct in_addr *) &dest.sin_addr.s_addr) == 0) {
    92. perror(argv[1]);
    93. exit(errno);
    94. }
    95. printf("address created\n");
    96. /* 连接服务器 */
    97. if (connect(sockfd, (struct sockaddr *) &dest, sizeof(dest)) != 0) {
    98. perror("Connect ");
    99. exit(errno);
    100. }
    101. printf("server connected\n");
    102. /* 基于 ctx 产生一个新的 SSL */
    103. ssl = SSL_new(ctx);
    104. SSL_set_fd(ssl, sockfd);
    105. /* 建立 SSL 连接 */
    106. if (SSL_connect(ssl) == -1)
    107. ERR_print_errors_fp(stderr);
    108. else {
    109. printf("Connected with %s encryption\n", SSL_get_cipher(ssl));
    110. ShowCerts(ssl);
    111. }
    112. /* 接收对方发过来的消息,最多接收 MAXBUF 个字节 */
    113. bzero(buffer, MAXBUF + 1);
    114. /* 接收服务器来的消息 */
    115. len = SSL_read(ssl, buffer, MAXBUF);
    116. if (len > 0)
    117. printf("接收消息成功:'%s',共%d个字节的数据\n",
    118. buffer, len);
    119. else {
    120. printf
    121. ("消息接收失败!错误代码是%d,错误信息是'%s'\n",
    122. errno, strerror(errno));
    123. goto finish;
    124. }
    125. bzero(buffer, MAXBUF + 1);
    126. strcpy(buffer, "from client->server");
    127. /* 发消息给服务器 */
    128. len = SSL_write(ssl, buffer, strlen(buffer));
    129. if (len < 0)
    130. printf
    131. ("消息'%s'发送失败!错误代码是%d,错误信息是'%s'\n",
    132. buffer, errno, strerror(errno));
    133. else
    134. printf("消息'%s'发送成功,共发送了%d个字节!\n",
    135. buffer, len);
    136. finish:
    137. /* 关闭连接 */
    138. SSL_shutdown(ssl);
    139. SSL_free(ssl);
    140. close(sockfd);
    141. SSL_CTX_free(ctx);
    142. return 0;
    143. }

     编译程序

    • server端
    • sudo gcc ssl_server.c -o server -lssl -lcrypto -ldl
    • client端
    • sudo gcc ssl_client.c -o client -lssl -lcrypto -ldl

    运行程序

    •  server端
    • sudo ./server 7838 1 /home/chy-cpabe/ssl_server_client/server/pem/server.crt /home/chy-cpabe/ssl_server_client/server/pem/server_rsa_private.pem.unsecure
    • client端
    • sudo ./client 127.0.0.1 7838 /home/chy-cpabe/ssl_server_client/client/pem/client.crt /home/chy-cpabe/ssl_server_client/client/pem/client_rsa_private.pem.unsecure

    运行截图

     server端

    client端

    补充知识

    1. # CA证书及密钥生成方法一----直接生成CA密钥及其自签名证书
    2. # 如果想以后读取私钥文件ca_rsa_private.pem时不需要输入密码,亦即不对私钥进行加密存储,那么将-passout pass:123456替换成-nodes
    3. openssl req -newkey rsa:2048 -passout pass:123456 -keyout ca_rsa_private.pem -x509 -days 365 -out ca.crt -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=CA/emailAddress=youremail@qq.com"
    4. # CA证书及密钥生成方法二----分步生成CA密钥及其自签名证书:
    5. # openssl genrsa -aes256 -passout pass:123456 -out ca_rsa_private.pem 2048
    6. # openssl req -new -x509 -days 365 -key ca_rsa_private.pem -passin pass:123456 -out ca.crt -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=CA/emailAddress=youremail@qq.com"
    7. # 服务器证书及密钥生成方法一----直接生成服务器密钥及待签名证书
    8. # 如果想以后读取私钥文件server_rsa_private.pem时不需要输入密码,亦即不对私钥进行加密存储,那么将-passout pass:server替换成-nodes
    9. openssl req -newkey rsa:2048 -passout pass:server -keyout server_rsa_private.pem -out server.csr -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=SERVER/emailAddress=youremail@qq.com"
    10. # 服务器证书及密钥生成方法二----分步生成服务器密钥及待签名证书
    11. # openssl genrsa -aes256 -passout pass:server -out server_rsa_private.pem 2048
    12. # openssl req -new -key server_rsa_private.pem -passin pass:server -out server.csr -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=SERVER/emailAddress=youremail@qq.com"
    13. # 使用CA证书及密钥对服务器证书进行签名:
    14. openssl x509 -req -days 365 -in server.csr -CA ca.crt -CAkey ca_rsa_private.pem -passin pass:123456 -CAcreateserial -out server.crt
    15. # 将加密的RSA密钥转成未加密的RSA密钥,避免每次读取都要求输入解密密码
    16. # 密码就是生成私钥文件时设置的passout、读取私钥文件时要输入的passin,比如这里要输入“server”
    17. openssl rsa -in server_rsa_private.pem -out server_rsa_private.pem.unsecure
    18. # 客户端证书及密钥生成方法一----直接生成客户端密钥及待签名证书
    19. # 如果想以后读取私钥文件client_rsa_private.pem时不需要输入密码,亦即不对私钥进行加密存储,那么将-passout pass:client替换成-nodes
    20. openssl req -newkey rsa:2048 -passout pass:client -keyout client_rsa_private.pem -out client.csr -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=CLIENT/emailAddress=youremail@qq.com"
    21. # 客户端证书及密钥生成方法二----分步生成客户端密钥及待签名证书:
    22. # openssl genrsa -aes256 -passout pass:client -out client_rsa_private.pem 2048
    23. # openssl req -new -key client_rsa_private.pem -passin pass:client -out client.csr -subj "/C=CN/ST=GD/L=SZ/O=COM/OU=NSP/CN=CLIENT/emailAddress=youremail@qq.com"
    24. # 使用CA证书及密钥对客户端证书进行签名:
    25. openssl x509 -req -days 365 -in client.csr -CA ca.crt -CAkey ca_rsa_private.pem -passin pass:123456 -CAcreateserial -out client.crt
    26. # 将加密的RSA密钥转成未加密的RSA密钥,避免每次读取都要求输入解密密码
    27. # 密码就是生成私钥文件时设置的passout、读取私钥文件时要输入的passin,比如这里要输入“client”
    28. openssl rsa -in client_rsa_private.pem -out client_rsa_private.pem.unsecure

     

  • 相关阅读:
    Unity Rain Ai 插件的使用入门
    MySQL启动不了,无法启动MySQL服务解决技巧
    C# 给List编个序号
    Bellman-Ford算法
    IP初学习
    错误:找不到或无法加载主类
    Layui 表单设计器
    Springboot整合Websocket(推送消息通知)
    【K8S】K8S服务搭建
    【Python基础系列】Part2. 列表
  • 原文地址:https://blog.csdn.net/CHYabc123456hh/article/details/125880756