Я некоторое время пытался сгенерировать пару ключей RSA, используя расширение openssl в PHP, и сохранить результат как пару ключей, совместимую с 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 (), но, что бы я ни пытался, результат никогда не будет эквивалентен тому, что я получаю от использования команды ssh-keygen -y -f
для того же закрытого ключа RSA, сгенерированного openssl.
Вот упрощенная версия моего кода:
<?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?