【问题标题】:Converting an OpenSSL generated RSA public key to OpenSSH format (PHP)将 OpenSSL 生成的 RSA 公钥转换为 OpenSSH 格式 (PHP)
【发布时间】:2011-07-28 07:17:05
【问题描述】:

我一直在尝试使用 PHP 的 openssl 扩展生成一个 RSA 密钥对,并将结果保存为与 OpenSSH 兼容的密钥对——这意味着私钥是 PEM 编码的(这很容易),而公钥存储在一个OpenSSH 具体格式如下:

ssh-rsa AAAAB3NzaC1yc2EAAAADAQABA...more base64 encoded stuff...

据我所知,这种格式包括:

  • 明文形式的密钥类型,后跟空格(即“openssh-rsa”)
  • 代表以下数据的 base64 编码字符串:
    • 算法名称的长度,以字节为单位(在本例中为 7),编码为 32 位无符号长大端
    • 算法名称,在本例中为“ssh-rsa”
    • 以字节为单位的 RSA 'e' 数字的长度,编码为 32 位无符号长大端序
    • RSA 'e' 编号
    • RSA 'n' 数字的长度,以字节为单位,编码为 32 位无符号长大端序
    • RSA 'n' 号码

我已经尝试使用 PHP 的 pack() 函数来实现这一点,但无论我尝试什么,结果永远不会等同于我在由 openssl 生成的同一个 RSA 私钥上使用 ssh-keygen -y -f 命令所得到的结果。

这是我的代码的简化版本:

<?php

// generate private key
$privKey = openssl_pkey_new(array(
    'private_key_bits' => 1024,
    'private_key_type' => OPENSSL_KEYTYPE_RSA
));

// convert public key to OpenSSH format
$keyInfo = openssl_pkey_get_details($privKey);
$data = pack("Na*", 7, 'ssh-rsa');
$data .= pack("Na*", strlen($keyInfo['rsa']['e']), $keyInfo['rsa']['e']);
$data .= pack("Na*", strlen($keyInfo['rsa']['n']), $keyInfo['rsa']['n']);

$pubKey = "ssh-rsa " . base64_encode($data);

echo "PHP generated RSA public key:\n$pubKey\n\n";

// For comparison, generate public key using ssh-keygen
openssl_pkey_export($privKey, $pem);
$umask = umask(0066); // this is needed for ssh-keygen to work properly
file_put_contents('/tmp/ssh-keygen-test', $pem);
umask($umask); 

exec('ssh-keygen -y -f /tmp/ssh-keygen-test', $out, $ret);
$otherPubKey = $out[0];
echo "ssh-keygen generated RSA public key:\n$otherPubKey\n\n";

echo ($pubKey == $otherPubKey ? "yes! they are the same\n" : "FAIL! they are different\n");

?>

关于如何在不依赖 ssh-keygen 的情况下做到这一点的任何提示?

【问题讨论】:

    标签: php openssl rsa openssh


    【解决方案1】:

    好的,我刚刚通过仔细查看来自Convert pem key to ssh-rsa format 的 C 实现引用解决了我自己的问题(我以前做过,但显然我错过了一些重要的东西)。我需要将 N 和 e 的第一个字符与 0x80 相加,如果匹配,则在数字的开头添加另一个 NULL 字符并将大小分别增加 1。

    我不确定为什么要这样做(我在我所做的在线搜索中没有找到对此的任何参考)但它确实有效。

    我只对此进行了基本测试,但它似乎运行良好,这里是我的代码:

    <?php
    
    $privKey = openssl_pkey_get_private($rsaKey);
    $pubKey = sshEncodePublicKey($privKey);
    
    echo "PHP generated RSA public key:\n$pubKey\n\n";
    
    function sshEncodePublicKey($privKey)
    {
        $keyInfo = openssl_pkey_get_details($privKey);
    
        $buffer  = pack("N", 7) . "ssh-rsa" . 
                   sshEncodeBuffer($keyInfo['rsa']['e']) . 
                   sshEncodeBuffer($keyInfo['rsa']['n']);
    
        return "ssh-rsa " . base64_encode($buffer); 
    }
    
    function sshEncodeBuffer($buffer)
    {
        $len = strlen($buffer);
        if (ord($buffer[0]) & 0x80) {
            $len++;
            $buffer = "\x00" . $buffer;
        }
    
        return pack("Na*", $len, $buffer);
    }
    ?>
    

    【讨论】:

    • 请注意sshEncodeBuffer 需要字节长度。如果您将mbstring.func_overload 设置为重载字符串函数,请将strlen($buffer) 替换为mb_strlen($buffer, '8bit');
    • @shevron 它非常适合 RSA 密钥。但它不适用于 DSA 密钥。如何转换 DSA 公钥?
    【解决方案2】:

    更简单的方法,使用phpseclib, a pure-PHP RSA implementation:

    <?php
    include('Crypt/RSA.php');
    
    $rsa = new Crypt_RSA();
    $rsa->loadKey('-----BEGIN PUBLIC KEY-----
    MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCqGKukO1De7zhZj6+H0qtjTkVxwTCpvKe4eCZ0
    FPqri0cb2JZfXJ/DgYSF6vUpwmJG8wVQZKjeGcjDOL5UlsuusFncCzWBQ7RKNUSesmQRMSGkVb1/
    3j+skZ6UtW+5u09lHNsj6tQ51s1SPrCBkedbNf0Tp0GbMJDyR4e9T04ZZwIDAQAB
    -----END PUBLIC KEY-----');
    $rsa->setPublicKey();
    
    $publickey = $rsa->getPublicKey(CRYPT_RSA_PUBLIC_FORMAT_OPENSSH);
    ?>
    

    输出:

    ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQCqGKukO1De7zhZj6+H0qtjTkVxwTCpvKe4eCZ0FPqri0cb2JZfXJ/DgYSF6vUpwmJG8wVQZKjeGcjDOL5UlsuusFncCzWBQ7RKNUSesmQRMSGkVb1/3j+skZ6UtW+5u09lHNsj6tQ51s1SPrCBkedbNf0Tp0GbMJDyR4e9T04ZZw== phpseclib-generated-key
    

    来源:

    http://phpseclib.sourceforge.net/rsa/examples.html#public,openssh

    【讨论】:

      猜你喜欢
      • 2019-05-22
      • 2016-10-03
      • 1970-01-01
      • 2013-10-22
      • 1970-01-01
      • 1970-01-01
      • 2018-08-15
      • 1970-01-01
      相关资源
      最近更新 更多