通过 php curl 请求网页并不能获取到证书信息,此时需要使用 ssl socket 获取证书内容。下面来一起看看看详细的介绍:
示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
// 创建 stream context $context = stream_context_create([ 'ssl' => [ 'capture_peer_cert' => true, 'capture_peer_cert_chain' => true, ], ]); $resource = stream_socket_client( "ssl://$domain:$port" , $errno , $errstr , 30, STREAM_CLIENT_CONNECT, $context ); $cert = stream_context_get_params( $resource ); $ssl = $cert [ 'options' ][ 'ssl' ]; $resource = $ssl [ 'peer_certificate' ]; // 网站证书中只有公钥,通过 openssl_pkey_get_details 导出公钥 $ret = [ 'crt' => '' , 'pub' => '' , ]; $pkey = openssl_pkey_get_public( $resource ); $ret [ 'pub' ] = openssl_pkey_get_details( $pkey )[ 'key' ]; openssl_x509_export( $resource , $pem ); $ret [ 'crt' ] = $pem ; foreach ( $ssl [ 'peer_certificate_chain' ] as $resource ) { openssl_x509_export( $resource , $pem ); $ret [ 'crt' ] .= "\n" . $pem ; } // 保存 $ret['crt'] 为 domain.crt // 保存 $ret['pub'] 为 domain.pub return $ret ; |
验证证书中的公钥A是否正确,通过私钥导出公钥B,比较两者发现一致。
1
2
3
4
5
6
7
8
9
10
11
12
13
|
$domain = 'blog.zhengxianjun.com' ; $port = '443' ; // ... $pub_a = $ret [ 'pub' ]; $private_key_path = '/conf/ssl/blog.zhengxianjun.com.key' ; // 证书没有设置密码,$passphrase 为空字符串 $pkey = openssl_pkey_get_private(file_get_content( $private_key_path ), $passphrase = '' ); $pub_b = openssl_pkey_get_details( $pkey )[ 'key' ]; // 两者一致 var_dump( $pub_a === $pub_b ); |
函数 stream_socket_client 还有一个用途是当知道服务器 IP 时,能获取到服务器可能可以使用的域名。
1
2
3
4
5
6
7
8
|
$resource = stream_socket_client( "ssl://$ip:$port" , $errno , $errstr , 30, STREAM_CLIENT_CONNECT, $context ); $cert = stream_context_get_params( $resource ); // 解析 X.509 格式证书 $info = openssl_x509_parse( $cert [ 'options' ][ 'ssl' ][ 'peer_certificate' ]); // 获取证书中的可信域名列表 $domain = str_replace ( 'DNS:' , '' , $info [ 'extensions' ][ 'subjectAltName' ]); |
以上可以看到获取网站证书并不能获得私钥。
在一些使用 CDN 的站点,如果使用了 HTTPS 同时又希望使用自有域名,是否需要将自己的私钥提供给 CDN 厂商呢?实际上证书路径与使用者名称(支持 https 的域名)并不需要一致。
也就是使用自有域名并进行 CDN 加速时不需要使用自有的 ssl 证书,只需将自己的 CDN 域名加到厂商证书的域名列表即可。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:https://blog.zhengxianjun.com/2017/02/php-curl-ssl/