PHP openssl生成数字证书密钥
实际可用的CA签名程序:ca_sign.php.txt
实例演示:
关于PHP生成证书密钥的资料真是好少啊,查了半天,最终还是在官方文档找到了相关资料,又根据自己的理解,整理成了以下代码,分成两部分:生成证书密钥、加密解密数据。直接复制下来做成两个文件运行就好啦。已经写了详细的注释,相信PHP程序员都能看得懂。
generate.php
$dn = array(
"countryName" => 'CN', //所在国家名称
"stateOrProvinceName" => 'GD', //所在省份名称
"localityName" => 'GZ', //所在城市名称
"organizationName" => 'DGQ', //注册人姓名
"organizationalUnitName" => 'GDEC', //组织名称
"commonName" => 'www.dgq.com', //公共名称
"emailAddress" => 'user@domain.com' //邮箱
);
$privkeypass = '111111'; //私钥密码
$numberofdays = 3650; //有效时长
$cerpath = "./test.cer"; //生成证书路径
$pfxpath = "./test.pfx"; //密钥文件路径
//生成证书
$cacert = file_get_contents("./cacert.cer"); //CA证书文件
$cakey = array(file_get_contents("./private/cakey.pem"),NULL); //CA私钥文件
$privkey = openssl_pkey_new( array(" private _key_bits" => 2048, "private_key_type" => OPENSSL_KEYTYPE_ RSA ) );
$csr = openssl_csr_new($dn, $privkey, array('digest_alg' => 'sha256'));
//$sscert = openssl_csr_sign($csr, null, $privkey, $numberofdays); //自签证书
$sscert = openssl_csr_sign($csr, $cacert, $cakey, $numberofdays,array('digest_alg' => 'sha256')); //CA签发证书
openssl_x509_export_to_file($sscert, $cerpath); //导出证书到文件
openssl_pkcs12_export_to_file($sscert, $pfxpath, $privkey, $privkeypass); //生成密钥文件
crypt.php
//私钥加密
$cer_key = file_get_contents($pfxpath); //获取密钥内容
openssl_pkcs12_read($cer_key, $certs, $privkeypass);
openssl_sign($data, $signMsg, $certs['pkey'],OPENSSL_ALGO_SHA1); //注册生成加密信息
$signMsg = base64_encode($signMsg); //base64转码加密信息
echo $signMsg;
//公钥解密
$cer_key = file_get_contents($cerpath); //获取证书内容
$unsignMsg=base64_decode($signMsg);//base64解码加密信息
$cer = openssl_x509_read($cer_key); //读取公钥
$res = openssl_verify($data, $unsignMsg, $cer); //验证
echo $res; //输出验证结果,1:验证成功,0:验证失败
一个简单的加密解密类:
class RsaCrypt {
const PRIVATE_KEY_FILE_PATH = './rsa_private_key.pem';
const PUBLIC_KEY_FILE_PATH = './rsa_public_key.pem';
public static function encode($orignData) {
//密钥文件的路径
$privateKeyFilePath = self::PRIVATE_KEY_FILE_PATH;
extension_loaded('openssl') or die('php需要openssl扩展支持');
(file_exists($privateKeyFilePath)) or die('密钥的文件路径不正确');
//生成Resource类型的密钥,如果密钥文件内容被破坏,openssl_pkey_get_private函数返回false
$privateKey = openssl_pkey_get_private(file_get_contents($privateKeyFilePath));
($privateKey) or die('密钥不可用');
//加密以后的数据,用于在网路上传输
$encryptData = '';
///////////////////////////////用私钥加密////////////////////////
if (openssl_private_encrypt($orignData, $encryptData, $privateKey)) {
return $encryptData;
} else {
die('加密失败');
}
}
public static function decode($encryptData) {
//公钥文件的路径
$publicKeyFilePath = self::PUBLIC_KEY_FILE_PATH;
extension_loaded('openssl') or die('php需要openssl扩展支持');
(file_exists($publicKeyFilePath)) or die('公钥的文件路径不正确');
//生成Resource类型的公钥,如果公钥文件内容被破坏,openssl_pkey_get_public函数返回false
$publicKey = openssl_pkey_get_public(file_get_contents($publicKeyFilePath));
($publicKey) or die('公钥不可用');
//解密以后的数据
$decryptData = '';
///////////////////////////////用公钥解密////////////////////////
if (openssl_public_decrypt($encryptData, $decryptData, $publicKey)) {
return $decryptData;
} else {
die('解密失败');
}
}
}
特别说明:
x509,公钥证书,只有公钥。
p7,签名或加密。可以往里面塞x509,同时没有签名或加密内容。
p12,含有私钥,同时可以有公钥,有口令保护。
p7的作用就是电子信封。
X509是基本规范
P7和P12是两个实现规范,P7是数字信封,P12是带有私钥的证书规范。
x509是数字证书的规范,P7和P12是两种封装形式。比如说同样的电影,有的是avi格式,有的是mpg,大概就这个意思。
P7一般是把证书分成两个文件,一个公钥一个私钥,有PEM和DER两种编码方式。PEM比较多见,就是纯文本的,P7一般是分发公钥用,看到的就是一串可见字符串,扩展名经常是.crt,.cer,.key等。DER是二进制编码。
P12是把证书压成一个文件,.pfx 。主要是考虑分发证书,私钥是要绝对保密的,不能随便以文本方式散播。所以P7格式不适合分发。.pfx中可以加密码保护,所以相对安全些。
在实践中要中,用户证书都是放在USB Key中分发,服务器证书经常还是以文件方式分发。服务器证书和用户证书,都是X509证书,就是里面的属性有区别。
X509 是证书规范
PKCS#7 是消息语法 (常用于数字签名与加密)
PKCS#12 个人消息交换与打包语法 (如.PFX .P12)打包成带公钥与私钥
评论已关闭