2016-11-16 107 views
2

我對SSL/TLS一無所知,並嘗試使用gRPC中的SSL/TLS通道,方法是按照在線說明進行操作。 這裏是服務器代碼:在gRPC客戶端服務器通信中使用SSL

std::string server_address("0.0.0.0:50051"); 
    GreeterServiceImpl service; 

    grpc::SslServerCredentialsOptions::PemKeyCertPair pkcp ={"a","b"}; 
    grpc::SslServerCredentialsOptions ssl_opts; 
    ssl_opts.pem_root_certs=""; 
    ssl_opts.pem_key_cert_pairs.push_back(pkcp); 

    std::shared_ptr<grpc::ServerCredentials> creds; 
    creds = grpc::SslServerCredentials(ssl_opts); 

    ServerBuilder builder; 
    builder.AddListeningPort(server_address, creds); 
    builder.RegisterService(&service); 
    std::unique_ptr<Server> server(builder.BuildAndStart()); 

服務器將無法啓動,並與下面的錯誤終止。

E1115 13:00:55.657846941 17129 ssl_transport_security.c:636] Invalid cert chain file. 
E1115 13:00:55.657936436 17129 security_connector.c:830] Handshaker factory creation failed with TSI_INVALID_ARGUMENT. 
E1115 13:00:55.657954952 17129 server_secure_chttp2.c:344] {"created":"@1479243655.657946821","description":"Unable to create secure server with credentials of type Ssl.","file":"src/core/ext/transport/chttp2/server/secure/server_secure_chttp2.c","file_line":242,"security_status":1} 
Server listening on 0.0.0.0:50051 
Segmentation fault (core dumped) 

我想要做的只是使用SSL進行服務器和客戶端通信。看起來我缺少服務器中的正確證書,並且認爲客戶端需要類似的東西。由於我沒有任何SSL背景知識,如果有人能夠指點我如何創建這些證書並正確使用gRPC通信的例子,那將是非常好的。

回答

1

這看起來與證書的創建方式有關。我忘記了StackOverflow中顯示如何創建證書的文章。使用該方法並插入密鑰和證書以使其工作。

下面是我如何創建密鑰(server.key)和證書(server.crt)。

mypass="pass123" 

echo Generate server key: 
openssl genrsa -passout pass:$mypass -des3 -out server.key 4096 

echo Generate server signing request: 
openssl req -passin pass:$mypass -new -key server.key -out server.csr -subj "/C=US/ST=CA/L=SanFrancisco/O=Google/OU=youtube/CN=localhost" 

echo Self-sign server certificate: 
openssl x509 -req -passin pass:$mypass -days 365 -in server.csr -signkey server.key -set_serial 01 -out server.crt 

echo Remove passphrase from server key: 
openssl rsa -passin pass:$mypass -in server.key -out server.key 

rm server.csr 

請注意在這裏使用localhost。在創建客戶端通道時應該使用相同的值。 我修改了grpc代碼庫的例子來使用它們。看起來像下面的服務器。

std::string server_address("0.0.0.0:50051"); 
GreeterServiceImpl service; 

std::string servercert = read_keycert("server.crt"); 
std::string serverkey = read_keycert("server.key"); 

grpc::SslServerCredentialsOptions::PemKeyCertPair pkcp; 
pkcp.private_key = serverkey; 
pkcp.cert_chain = servercert; 

grpc::SslServerCredentialsOptions ssl_opts; 
ssl_opts.pem_root_certs=""; 
ssl_opts.pem_key_cert_pairs.push_back(pkcp); 

std::shared_ptr<grpc::ServerCredentials> creds; 
creds = grpc::SslServerCredentials(ssl_opts); 

ServerBuilder builder; 
builder.AddListeningPort(server_address, creds); 
builder.RegisterService(&service); 
std::unique_ptr<Server> server(builder.BuildAndStart()); 
server->Wait(); 

在客戶端只需要證書。以下是我如何使用它。

std::string cacert = read_keycert("server.crt"); 
grpc::SslCredentialsOptions ssl_opts; 
ssl_opts.pem_root_certs=cacert; 

auto ssl_creds = grpc::SslCredentials(ssl_opts); 
GreeterClient greeter(grpc::CreateChannel("localhost:50051", ssl_creds)); 

密鑰和證書的生成和使用方式在每個用例中都不相同。我想如果兩臺服務器通信,那麼這些服務器可能會有所不同不確定。希望這有助於一些新手從gRPC C++中的安全SSL/TLS開始。