summaryrefslogtreecommitdiff
path: root/src/keytrans/openpgp2ssh.c
blob: 92bdc1955c81e0cada4eb80a5177017f5b226575 (plain)
  1. #include "gnutls-helpers.h"
  2. #include <gnutls/openpgp.h>
  3. #include <gnutls/x509.h>
  4. /* for waitpid() */
  5. #include <sys/types.h>
  6. #include <sys/wait.h>
  7. /*
  8. Author: Daniel Kahn Gillmor <dkg@fifthhorseman.net>
  9. Date: 2008-06-12 13:47:41-0400
  10. License: GPL v3 or later
  11. monkeysphere key translator: execute this with an OpenPGP key on
  12. stdin, (please indicate the specific keyid that you want as the
  13. first argument if there are subkeys). At the moment, only public
  14. keys and passphraseless secret keys work.
  15. For secret keys, it will spit out a PEM-encoded version of the key
  16. on stdout, which can be fed into ssh-add like this:
  17. gpg --export-secret-keys $KEYID | openpgp2ssh $KEYID | ssh-add -c /dev/stdin
  18. For public keys, it will spit out a single line of text that can
  19. (with some massaging) be used in an openssh known_hosts or
  20. authorized_keys file. For example:
  21. echo server.example.org $(gpg --export $KEYID | openpgp2ssh $KEYID) >> ~/.ssh/known_hosts
  22. Requirements: I've only built this so far with GnuTLS v2.3.x.
  23. GnuTLS 2.2.x does not contain the appropriate functionality.
  24. */
  25. /* FIXME: keyid should be const as well */
  26. int convert_private_pgp_to_x509(gnutls_x509_privkey_t* output, const gnutls_openpgp_privkey_t* pgp_privkey, gnutls_openpgp_keyid_t* keyid) {
  27. gnutls_datum_t m, e, d, p, q, u, g, y, x;
  28. gnutls_pk_algorithm_t pgp_algo;
  29. unsigned int pgp_bits;
  30. int ret;
  31. gnutls_openpgp_keyid_t curkeyid;
  32. int subkeyidx;
  33. int subkeycount;
  34. int found = 0;
  35. init_datum(&m);
  36. init_datum(&e);
  37. init_datum(&d);
  38. init_datum(&p);
  39. init_datum(&q);
  40. init_datum(&u);
  41. init_datum(&g);
  42. init_datum(&y);
  43. init_datum(&x);
  44. subkeycount = gnutls_openpgp_privkey_get_subkey_count(*pgp_privkey);
  45. if (subkeycount < 0) {
  46. err(0,"Could not determine subkey count (got value %d)\n", subkeycount);
  47. return 1;
  48. }
  49. if ((keyid == NULL) &&
  50. (subkeycount > 0)) {
  51. err(0,"No keyid passed in, but there were %d keys to choose from\n", subkeycount + 1);
  52. return 1;
  53. }
  54. if (keyid != NULL) {
  55. ret = gnutls_openpgp_privkey_get_key_id(*pgp_privkey, curkeyid);
  56. if (ret) {
  57. err(0,"Could not get keyid (error: %d)\n", ret);
  58. return 1;
  59. }
  60. }
  61. if ((keyid == NULL) || (memcmp(*keyid, curkeyid, sizeof(gnutls_openpgp_keyid_t)) == 0)) {
  62. /* we want to export the primary key: */
  63. err(0,"exporting primary key\n");
  64. /* FIXME: this is almost identical to the block below for subkeys.
  65. This clumsiness seems inherent in the gnutls OpenPGP API,
  66. though. ugh. */
  67. pgp_algo = gnutls_openpgp_privkey_get_pk_algorithm(*pgp_privkey, &pgp_bits);
  68. if (pgp_algo < 0) {
  69. err(0, "failed to get OpenPGP key algorithm (error: %d)\n", pgp_algo);
  70. return 1;
  71. }
  72. if (pgp_algo == GNUTLS_PK_RSA) {
  73. err(0,"OpenPGP RSA Key, with %d bits\n", pgp_bits);
  74. ret = gnutls_openpgp_privkey_export_rsa_raw(*pgp_privkey, &m, &e, &d, &p, &q, &u);
  75. if (GNUTLS_E_SUCCESS != ret) {
  76. err(0, "failed to export RSA key parameters (error: %d)\n", ret);
  77. return 1;
  78. }
  79. } else if (pgp_algo == GNUTLS_PK_DSA) {
  80. err(0,"OpenPGP DSA Key, with %d bits\n", pgp_bits);
  81. ret = gnutls_openpgp_privkey_export_dsa_raw(*pgp_privkey, &p, &q, &g, &y, &x);
  82. if (GNUTLS_E_SUCCESS != ret) {
  83. err(0,"failed to export DSA key parameters (error: %d)\n", ret);
  84. return 1;
  85. }
  86. }
  87. found = 1;
  88. } else {
  89. /* lets trawl through the subkeys until we find the one we want: */
  90. for (subkeyidx = 0; (subkeyidx < subkeycount) && !found; subkeyidx++) {
  91. ret = gnutls_openpgp_privkey_get_subkey_id(*pgp_privkey, subkeyidx, curkeyid);
  92. if (ret) {
  93. err(0,"Could not get keyid of subkey with index %d (error: %d)\n", subkeyidx, ret);
  94. return 1;
  95. }
  96. if (memcmp(*keyid, curkeyid, sizeof(gnutls_openpgp_keyid_t)) == 0) {
  97. err(0,"exporting subkey index %d\n", subkeyidx);
  98. /* FIXME: this is almost identical to the block above for the
  99. primary key. */
  100. pgp_algo = gnutls_openpgp_privkey_get_subkey_pk_algorithm(*pgp_privkey, subkeyidx, &pgp_bits);
  101. if (pgp_algo < 0) {
  102. err(0,"failed to get the algorithm of the OpenPGP public key (error: %d)\n", pgp_algo);
  103. return pgp_algo;
  104. } else if (pgp_algo == GNUTLS_PK_RSA) {
  105. err(0,"OpenPGP RSA key, with %d bits\n", pgp_bits);
  106. ret = gnutls_openpgp_privkey_export_subkey_rsa_raw(*pgp_privkey, subkeyidx, &m, &e, &d, &p, &q, &u);
  107. if (GNUTLS_E_SUCCESS != ret) {
  108. err(0,"failed to export RSA key parameters (error: %d)\n", ret);
  109. return 1;
  110. }
  111. } else if (pgp_algo == GNUTLS_PK_DSA) {
  112. err(0,"OpenPGP DSA Key, with %d bits\n", pgp_bits);
  113. ret = gnutls_openpgp_privkey_export_subkey_dsa_raw(*pgp_privkey, subkeyidx, &p, &q, &g, &y, &x);
  114. if (GNUTLS_E_SUCCESS != ret) {
  115. err(0,"failed to export DSA key parameters (error: %d)\n", ret);
  116. return 1;
  117. }
  118. }
  119. found = 1;
  120. }
  121. }
  122. }
  123. if (!found) {
  124. err(0,"Could not find key in input\n");
  125. return 1;
  126. }
  127. if (pgp_algo == GNUTLS_PK_RSA) {
  128. ret = gnutls_x509_privkey_import_rsa_raw (*output, &m, &e, &d, &p, &q, &u);
  129. if (GNUTLS_E_SUCCESS != ret) {
  130. err(0, "failed to import RSA key parameters (error: %d)\n", ret);
  131. return 1;
  132. }
  133. } else if (pgp_algo == GNUTLS_PK_DSA) {
  134. ret = gnutls_x509_privkey_import_dsa_raw (*output, &p, &q, &g, &y, &x);
  135. if (GNUTLS_E_SUCCESS != ret) {
  136. err(0,"failed to import DSA key parameters (error: %d)\n", ret);
  137. return 1;
  138. }
  139. } else {
  140. err(0,"OpenPGP Key was not RSA or DSA -- can't deal! (actual algorithm was: %d)\n", pgp_algo);
  141. return 1;
  142. }
  143. ret = gnutls_x509_privkey_fix(*output);
  144. if (ret != 0) {
  145. err(0,"failed to fix up the private key in X.509 format (error: %d)\n", ret);
  146. return 1;
  147. }
  148. return 0;
  149. }
  150. /* FIXME: keyid should be const also */
  151. int emit_public_openssh_from_pgp(const gnutls_openpgp_crt_t* pgp_crt, gnutls_openpgp_keyid_t* keyid) {
  152. gnutls_openpgp_keyid_t curkeyid;
  153. int ret;
  154. int subkeyidx;
  155. int subkeycount;
  156. int found = 0;
  157. gnutls_datum_t m, e, p, q, g, y, algolabel;
  158. unsigned int bits;
  159. gnutls_pk_algorithm_t algo;
  160. const gnutls_datum_t* all[5];
  161. const char* algoname;
  162. int mpicount;
  163. /* output_data must be at least 2 chars longer than the maximum possible
  164. algorithm name: */
  165. char output_data[20];
  166. /* variables for the output conversion: */
  167. int pipestatus;
  168. int pipefd, child_pid;
  169. char* const b64args[] = {"/usr/bin/base64", "--wrap=0", NULL};
  170. init_datum(&m);
  171. init_datum(&e);
  172. init_datum(&p);
  173. init_datum(&q);
  174. init_datum(&g);
  175. init_datum(&algolabel);
  176. /* figure out if we've got the right thing: */
  177. subkeycount = gnutls_openpgp_crt_get_subkey_count(*pgp_crt);
  178. if (subkeycount < 0) {
  179. err(0,"Could not determine subkey count (got value %d)\n", subkeycount);
  180. return 1;
  181. }
  182. if ((keyid == NULL) &&
  183. (subkeycount > 0)) {
  184. err(0,"No keyid passed in, but there were %d keys to choose from\n", subkeycount + 1);
  185. return 1;
  186. }
  187. if (keyid != NULL) {
  188. ret = gnutls_openpgp_crt_get_key_id(*pgp_crt, curkeyid);
  189. if (ret) {
  190. err(0,"Could not get keyid (error: %d)\n", ret);
  191. return 1;
  192. }
  193. }
  194. if ((keyid == NULL) || (memcmp(*keyid, curkeyid, sizeof(gnutls_openpgp_keyid_t)) == 0)) {
  195. /* we want to export the primary key: */
  196. err(0,"exporting primary key\n");
  197. /* FIXME: this is almost identical to the block below for subkeys.
  198. This clumsiness seems inherent in the gnutls OpenPGP API,
  199. though. ugh. */
  200. algo = gnutls_openpgp_crt_get_pk_algorithm(*pgp_crt, &bits);
  201. if (algo < 0) {
  202. err(0,"failed to get the algorithm of the OpenPGP public key (error: %d)\n", algo);
  203. return algo;
  204. } else if (algo == GNUTLS_PK_RSA) {
  205. err(0,"OpenPGP RSA certificate, with %d bits\n", bits);
  206. ret = gnutls_openpgp_crt_get_pk_rsa_raw(*pgp_crt, &m, &e);
  207. if (GNUTLS_E_SUCCESS != ret) {
  208. err(0,"failed to export RSA certificate parameters (error: %d)\n", ret);
  209. return 1;
  210. }
  211. } else if (algo == GNUTLS_PK_DSA) {
  212. err(0,"OpenPGP DSA certificate, with %d bits\n", bits);
  213. ret = gnutls_openpgp_crt_get_pk_dsa_raw(*pgp_crt, &p, &q, &g, &y);
  214. if (GNUTLS_E_SUCCESS != ret) {
  215. err(0,"failed to export DSA certificate parameters (error: %d)\n", ret);
  216. return 1;
  217. }
  218. }
  219. found = 1;
  220. } else {
  221. /* lets trawl through the subkeys until we find the one we want: */
  222. for (subkeyidx = 0; (subkeyidx < subkeycount) && !found; subkeyidx++) {
  223. ret = gnutls_openpgp_crt_get_subkey_id(*pgp_crt, subkeyidx, curkeyid);
  224. if (ret) {
  225. err(0,"Could not get keyid of subkey with index %d (error: %d)\n", subkeyidx, ret);
  226. return 1;
  227. }
  228. if (memcmp(*keyid, curkeyid, sizeof(gnutls_openpgp_keyid_t)) == 0) {
  229. err(0,"exporting subkey index %d\n", subkeyidx);
  230. /* FIXME: this is almost identical to the block above for the
  231. primary key. */
  232. algo = gnutls_openpgp_crt_get_subkey_pk_algorithm(*pgp_crt, subkeyidx, &bits);
  233. if (algo < 0) {
  234. err(0,"failed to get the algorithm of the OpenPGP public key (error: %d)\n", algo);
  235. return algo;
  236. } else if (algo == GNUTLS_PK_RSA) {
  237. err(0,"OpenPGP RSA certificate, with %d bits\n", bits);
  238. ret = gnutls_openpgp_crt_get_subkey_pk_rsa_raw(*pgp_crt, subkeyidx, &m, &e);
  239. if (GNUTLS_E_SUCCESS != ret) {
  240. err(0,"failed to export RSA certificate parameters (error: %d)\n", ret);
  241. return 1;
  242. }
  243. } else if (algo == GNUTLS_PK_DSA) {
  244. err(0,"OpenPGP DSA certificate, with %d bits\n", bits);
  245. ret = gnutls_openpgp_crt_get_subkey_pk_dsa_raw(*pgp_crt, subkeyidx, &p, &q, &g, &y);
  246. if (GNUTLS_E_SUCCESS != ret) {
  247. err(0,"failed to export DSA certificate parameters (error: %d)\n", ret);
  248. return 1;
  249. }
  250. }
  251. found = 1;
  252. }
  253. }
  254. }
  255. if (!found) {
  256. err(0,"Could not find key in input\n");
  257. return 1;
  258. }
  259. /* if we made it this far, we've got MPIs, and we've got the
  260. algorithm, so we just need to emit the info */
  261. if (algo == GNUTLS_PK_RSA) {
  262. algoname = "ssh-rsa";
  263. mpicount = 3;
  264. all[0] = &algolabel;
  265. all[1] = &e;
  266. all[2] = &m;
  267. } else if (algo == GNUTLS_PK_DSA) {
  268. algoname = "ssh-dss";
  269. mpicount = 5;
  270. all[0] = &algolabel;
  271. all[1] = &p;
  272. all[2] = &q;
  273. all[3] = &g;
  274. all[4] = &y;
  275. } else {
  276. err(0,"Key algorithm was neither DSA nor RSA (it was %d). Can't deal. Sorry!\n", algo);
  277. return 1;
  278. }
  279. if (ret = datum_from_string(&algolabel, algoname), ret) {
  280. err(0,"couldn't label string (error: %d)\n", ret);
  281. return ret;
  282. }
  283. snprintf(output_data, sizeof(output_data), "%s ", algoname);
  284. pipefd = create_writing_pipe(&child_pid, b64args[0], b64args);
  285. if (pipefd < 0) {
  286. err(0,"failed to create a writing pipe (returned %d)\n", pipefd);
  287. return pipefd;
  288. }
  289. write(1, output_data, strlen(output_data));
  290. if (0 != write_data_fd_with_length(pipefd, all, mpicount)) {
  291. err(0,"was not able to write out RSA key data\n");
  292. return 1;
  293. }
  294. close(pipefd);
  295. if (child_pid != waitpid(child_pid, &pipestatus, 0)) {
  296. err(0,"could not wait for child process to return for some reason.\n");
  297. return 1;
  298. }
  299. if (pipestatus != 0) {
  300. err(0,"base64 pipe died with return code %d\n", pipestatus);
  301. return pipestatus;
  302. }
  303. write(1, "\n", 1);
  304. return 0;
  305. }
  306. int main(int argc, char* argv[]) {
  307. gnutls_datum_t data;
  308. int ret;
  309. gnutls_x509_privkey_t x509_privkey;
  310. gnutls_openpgp_privkey_t pgp_privkey;
  311. gnutls_openpgp_crt_t pgp_crt;
  312. char output_data[10240];
  313. size_t ods = sizeof(output_data);
  314. gnutls_openpgp_keyid_t keyid;
  315. gnutls_openpgp_keyid_t* use_keyid;
  316. init_gnutls();
  317. /* figure out what keyid we should be looking for: */
  318. use_keyid = NULL;
  319. if (argv[1] != NULL) {
  320. ret = convert_string_to_keyid(keyid, argv[1]);
  321. if (ret != 0)
  322. return ret;
  323. use_keyid = &keyid;
  324. }
  325. init_datum(&data);
  326. /* slurp in the key from stdin */
  327. if (ret = set_datum_fd(&data, 0), ret) {
  328. err(0,"didn't read file descriptor 0\n");
  329. return 1;
  330. }
  331. if (ret = gnutls_openpgp_privkey_init(&pgp_privkey), ret) {
  332. err(0,"Failed to initialized OpenPGP private key (error: %d)\n", ret);
  333. return 1;
  334. }
  335. /* check whether it's a private key or a public key, by trying them: */
  336. if ((gnutls_openpgp_privkey_import(pgp_privkey, &data, GNUTLS_OPENPGP_FMT_RAW, NULL, 0) == 0) ||
  337. (gnutls_openpgp_privkey_import(pgp_privkey, &data, GNUTLS_OPENPGP_FMT_BASE64, NULL, 0) == 0)) {
  338. /* we're dealing with a private key */
  339. err(0,"Translating private key\n");
  340. if (ret = gnutls_x509_privkey_init(&x509_privkey), ret) {
  341. err(0,"Failed to initialize X.509 private key for output (error: %d)\n", ret);
  342. return 1;
  343. }
  344. ret = convert_private_pgp_to_x509(&x509_privkey, &pgp_privkey, use_keyid);
  345. gnutls_openpgp_privkey_deinit(pgp_privkey);
  346. if (ret)
  347. return ret;
  348. ret = gnutls_x509_privkey_export (x509_privkey,
  349. GNUTLS_X509_FMT_PEM,
  350. output_data,
  351. &ods);
  352. if (ret == 0) {
  353. write(1, output_data, ods);
  354. }
  355. gnutls_x509_privkey_deinit(x509_privkey);
  356. } else {
  357. if (ret = gnutls_openpgp_crt_init(&pgp_crt), ret) {
  358. err(0,"Failed to initialized OpenPGP certificate (error: %d)\n", ret);
  359. return 1;
  360. }
  361. if ((gnutls_openpgp_crt_import(pgp_crt, &data, GNUTLS_OPENPGP_FMT_RAW) == 0) ||
  362. (gnutls_openpgp_crt_import(pgp_crt, &data, GNUTLS_OPENPGP_FMT_BASE64) == 0)) {
  363. /* we're dealing with a public key */
  364. err(0,"Translating public key\n");
  365. ret = emit_public_openssh_from_pgp(&pgp_crt, use_keyid);
  366. } else {
  367. /* we have no idea what kind of key this is at all anyway! */
  368. err(0,"Input does not contain any form of OpenPGP key I recognize.\n");
  369. return 1;
  370. }
  371. }
  372. gnutls_global_deinit();
  373. return 0;
  374. }