參考資訊:
https://www.geeksforgeeks.org/socket-programming-cc/
https://github.com/openssl/openssl/wiki/Simple_TLS_Server
https://github.com/zapstar/two-way-ssl-c/blob/master/certs_gen.sh
https://stackoverflow.com/questions/16255323/make-an-https-request-using-sockets-on-linux
server.c
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <openssl/ssl.h> int main(int argc, char const* argv[]) { int opt = 1; int fd = -1; int client = 0; ssize_t r = 0; char buf[255] = { 0 }; struct sockaddr_in addr = { 0 }; const char *hello = "hello from server !"; if (argc != 3) { printf("Usage: %s IP Port\n", argv[0]); return 0; } SSL_library_init(); SSL_CTX *ctx = SSL_CTX_new(SSLv23_server_method()); SSL_CTX_use_certificate_file(ctx, "server_cert.pem", SSL_FILETYPE_PEM); SSL_CTX_use_PrivateKey_file(ctx, "server_key.pem", SSL_FILETYPE_PEM); fd = socket(AF_INET, SOCK_STREAM, 0); if (fd < 0) { printf("failed to init socket\n"); return -1; } if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt)) != 0) { printf("failed to set sockopt\n"); return -1; } addr.sin_family = AF_INET; addr.sin_addr.s_addr = inet_addr(argv[1]); addr.sin_port = htons(atoi(argv[2])); if (bind(fd, (struct sockaddr *)&addr, sizeof(addr)) != 0) { printf("failed to bind address\n"); return -1; } if (listen(fd, 3) != 0) { printf("failed to listen\n"); return -1; } socklen_t addrlen = sizeof(addr); client = accept(fd, (struct sockaddr *)&addr, &addrlen); if (client < 0) { printf("failed to accept from client\n"); return -1; } SSL *ssl = SSL_new(ctx); SSL_set_fd(ssl, client); if (SSL_accept(ssl) != 1) { printf("failed to accept with ssl\n"); return -1; } r = SSL_read(ssl, buf, sizeof(buf)); printf("%s (r=%d)\n", buf, r); SSL_write(ssl, hello, strlen(hello)); close(client); SSL_shutdown(ssl); SSL_free(ssl); close(fd); SSL_CTX_free(ctx); return 0; }
client.c
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <unistd.h> #include <arpa/inet.h> #include <openssl/ssl.h> int main(int argc, char const* argv[]) { int r = 0; int fd = -1; char buf[255] = { 0 }; struct sockaddr_in addr = { 0 }; const char *hello = "hello from client !"; if (argc != 3) { printf("Usage: %s IP Port\n", argv[0]); return 0; } SSL_library_init(); SSL_CTX *ctx = SSL_CTX_new(SSLv23_client_method()); SSL_CTX_use_certificate_file(ctx, "client_cert.pem", SSL_FILETYPE_PEM); SSL_CTX_use_PrivateKey_file(ctx, "client_key.pem", SSL_FILETYPE_PEM); fd = socket(AF_INET, SOCK_STREAM, 0); if (fd < 0) { printf("failed to init socket\n"); return -1; } addr.sin_family = AF_INET; addr.sin_port = htons(atoi(argv[2])); if (inet_pton(AF_INET, argv[1], &addr.sin_addr) != 1) { printf("failed to translate address\n"); return -1; } SSL *ssl = SSL_new(ctx); SSL_set_fd(ssl, fd); if (connect(fd, (struct sockaddr *)&addr, sizeof(addr)) != 0) { printf("failed to connect to server\n"); return -1; } if (SSL_connect(ssl) != 1) { printf("failed to connect with ssl\n"); return -1; } SSL_write(ssl, hello, strlen(hello)); usleep(100000); r = SSL_read(ssl, buf, sizeof(buf)); printf("%s (r=%d)\n", buf, r); SSL_shutdown(ssl); SSL_free(ssl); close(fd); SSL_CTX_free(ctx); return 0; }
編譯、執行
$ openssl req -x509 -nodes -days 3650 -newkey rsa:4096 -keyout ca_key.pem -out ca_cert.pem -subj "/C=US/ST=Acme State/L=Acme City/O=Acme Inc./CN=example.com" $ openssl genrsa -out server_key.pem 4096 $ openssl req -new -key server_key.pem -out server_cert.csr -subj "/C=US/ST=Acme State/L=Acme City/O=Acme Inc./CN=server.example.com" $ openssl x509 -req -days 1460 -in server_cert.csr -CA ca_cert.pem -CAkey ca_key.pem -CAcreateserial -out server_cert.pem $ openssl genrsa -out client_key.pem 4096 $ openssl req -new -key client_key.pem -out client_cert.csr -subj "/C=US/ST=Acme State/L=Acme City/O=Acme Inc./CN=client.example.com" $ openssl x509 -req -days 1460 -in client_cert.csr -CA ca_cert.pem -CAkey ca_key.pem -CAcreateserial -out client_cert.pem $ gcc server.c -o server -lssl $ gcc client.c -o client -lssl $ ./server 127.0.0.1 9999 & $ ./client 127.0.0.1 9999 hello from client ! (r=19) hello from server ! (r=19)