blob: d1b2a583a5f02ef3b8d25ec81681428655876bcf [file] [log] [blame]
Damien Millerd4a8b7e1999-10-27 13:42:43 +10001/*
Damien Miller95def091999-11-25 00:26:21 +11002 * Author: Tatu Ylonen <ylo@cs.hut.fi>
3 * Copyright (c) 1994 Tatu Ylonen <ylo@cs.hut.fi>, Espoo, Finland
4 * All rights reserved
Damien Miller95def091999-11-25 00:26:21 +11005 * Identity and host key generation and maintenance.
Damien Millere4340be2000-09-16 13:29:08 +11006 *
7 * As far as I am concerned, the code I have written for this software
8 * can be used freely for any purpose. Any derived versions of this
9 * software must be clearly marked as such, and if the derived work is
10 * incompatible with the protocol description in the RFC file, it must be
11 * called by a name other than "ssh" or "Secure Shell".
Damien Miller95def091999-11-25 00:26:21 +110012 */
Damien Millerd4a8b7e1999-10-27 13:42:43 +100013
14#include "includes.h"
Ben Lindstromd78ae762001-06-05 20:35:09 +000015RCSID("$OpenBSD: ssh-keygen.c,v 1.61 2001/05/25 14:37:32 markus Exp $");
Damien Millerd4a8b7e1999-10-27 13:42:43 +100016
Damien Millereba71ba2000-04-29 23:57:08 +100017#include <openssl/evp.h>
18#include <openssl/pem.h>
Damien Millereba71ba2000-04-29 23:57:08 +100019
Damien Millerd4a8b7e1999-10-27 13:42:43 +100020#include "xmalloc.h"
Damien Millereba71ba2000-04-29 23:57:08 +100021#include "key.h"
Ben Lindstromd09fcf52001-03-29 00:29:54 +000022#include "rsa.h"
Damien Millereba71ba2000-04-29 23:57:08 +100023#include "authfile.h"
24#include "uuencode.h"
Damien Miller874d77b2000-10-14 16:23:11 +110025#include "buffer.h"
26#include "bufaux.h"
Ben Lindstrom226cfa02001-01-22 05:34:40 +000027#include "pathnames.h"
28#include "log.h"
29#include "readpass.h"
Damien Miller874d77b2000-10-14 16:23:11 +110030
Damien Millereba71ba2000-04-29 23:57:08 +100031/* Number of bits in the RSA/DSA key. This value can be changed on the command line. */
Damien Millerd4a8b7e1999-10-27 13:42:43 +100032int bits = 1024;
33
Damien Miller5428f641999-11-25 11:54:57 +110034/*
35 * Flag indicating that we just want to change the passphrase. This can be
36 * set on the command line.
37 */
Damien Millerd4a8b7e1999-10-27 13:42:43 +100038int change_passphrase = 0;
39
Damien Miller5428f641999-11-25 11:54:57 +110040/*
41 * Flag indicating that we just want to change the comment. This can be set
42 * on the command line.
43 */
Damien Millerd4a8b7e1999-10-27 13:42:43 +100044int change_comment = 0;
45
46int quiet = 0;
47
Damien Miller10f6f6b1999-11-17 17:29:08 +110048/* Flag indicating that we just want to see the key fingerprint */
49int print_fingerprint = 0;
Ben Lindstrom8fd372b2001-03-12 03:02:17 +000050int print_bubblebabble = 0;
Damien Miller10f6f6b1999-11-17 17:29:08 +110051
Damien Miller431f66b1999-11-21 18:31:57 +110052/* The identity file name, given on the command line or entered by the user. */
53char identity_file[1024];
54int have_identity = 0;
Damien Millerd4a8b7e1999-10-27 13:42:43 +100055
56/* This is set to the passphrase if given on the command line. */
57char *identity_passphrase = NULL;
58
59/* This is set to the new passphrase if given on the command line. */
60char *identity_new_passphrase = NULL;
61
62/* This is set to the new comment if given on the command line. */
63char *identity_comment = NULL;
64
Damien Millereba71ba2000-04-29 23:57:08 +100065/* Dump public key file in format used by real and the original SSH 2 */
66int convert_to_ssh2 = 0;
67int convert_from_ssh2 = 0;
68int print_public = 0;
Damien Miller0bc1bd82000-11-13 22:57:25 +110069
Damien Millere39cacc2000-11-29 12:18:44 +110070/* default to RSA for SSH-1 */
71char *key_type_name = "rsa1";
Damien Millereba71ba2000-04-29 23:57:08 +100072
Damien Miller431f66b1999-11-21 18:31:57 +110073/* argv0 */
Damien Miller95def091999-11-25 00:26:21 +110074#ifdef HAVE___PROGNAME
Damien Miller431f66b1999-11-21 18:31:57 +110075extern char *__progname;
Ben Lindstrom49a79c02000-11-17 03:47:20 +000076#else
77char *__progname;
78#endif
Damien Millerd4a8b7e1999-10-27 13:42:43 +100079
Damien Millereba71ba2000-04-29 23:57:08 +100080char hostname[MAXHOSTNAMELEN];
81
Damien Miller431f66b1999-11-21 18:31:57 +110082void
83ask_filename(struct passwd *pw, const char *prompt)
Damien Millerd4a8b7e1999-10-27 13:42:43 +100084{
Damien Miller95def091999-11-25 00:26:21 +110085 char buf[1024];
Damien Millere39cacc2000-11-29 12:18:44 +110086 char *name = NULL;
87
88 switch (key_type_from_name(key_type_name)) {
89 case KEY_RSA1:
Ben Lindstrom226cfa02001-01-22 05:34:40 +000090 name = _PATH_SSH_CLIENT_IDENTITY;
Damien Millere39cacc2000-11-29 12:18:44 +110091 break;
92 case KEY_DSA:
Ben Lindstrom226cfa02001-01-22 05:34:40 +000093 name = _PATH_SSH_CLIENT_ID_DSA;
Damien Millere39cacc2000-11-29 12:18:44 +110094 break;
95 case KEY_RSA:
Ben Lindstrom226cfa02001-01-22 05:34:40 +000096 name = _PATH_SSH_CLIENT_ID_RSA;
Damien Millere39cacc2000-11-29 12:18:44 +110097 break;
98 default:
99 fprintf(stderr, "bad key type");
100 exit(1);
101 break;
102 }
103 snprintf(identity_file, sizeof(identity_file), "%s/%s", pw->pw_dir, name);
Ben Lindstrom3deda8b2000-12-22 20:27:43 +0000104 fprintf(stderr, "%s (%s): ", prompt, identity_file);
105 fflush(stderr);
Damien Miller95def091999-11-25 00:26:21 +1100106 if (fgets(buf, sizeof(buf), stdin) == NULL)
107 exit(1);
108 if (strchr(buf, '\n'))
109 *strchr(buf, '\n') = 0;
110 if (strcmp(buf, "") != 0)
111 strlcpy(identity_file, buf, sizeof(identity_file));
112 have_identity = 1;
Damien Miller10f6f6b1999-11-17 17:29:08 +1100113}
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000114
Ben Lindstromd0fca422001-03-26 13:44:06 +0000115Key *
Ben Lindstromd78ae762001-06-05 20:35:09 +0000116load_identity(char *filename)
Damien Millereba71ba2000-04-29 23:57:08 +1000117{
Ben Lindstromd0fca422001-03-26 13:44:06 +0000118 char *pass;
119 Key *prv;
120
Ben Lindstroma3700052001-04-05 23:26:32 +0000121 prv = key_load_private(filename, "", NULL);
Ben Lindstromd0fca422001-03-26 13:44:06 +0000122 if (prv == NULL) {
Ben Lindstromd78ae762001-06-05 20:35:09 +0000123 if (identity_passphrase)
124 pass = xstrdup(identity_passphrase);
125 else
126 pass = read_passphrase("Enter passphrase: ", 1);
Ben Lindstromd0fca422001-03-26 13:44:06 +0000127 prv = key_load_private(filename, pass, NULL);
Damien Millereba71ba2000-04-29 23:57:08 +1000128 memset(pass, 0, strlen(pass));
129 xfree(pass);
130 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000131 return prv;
Damien Millereba71ba2000-04-29 23:57:08 +1000132}
133
Damien Miller874d77b2000-10-14 16:23:11 +1100134#define SSH_COM_PUBLIC_BEGIN "---- BEGIN SSH2 PUBLIC KEY ----"
135#define SSH_COM_PUBLIC_END "---- END SSH2 PUBLIC KEY ----"
136#define SSH_COM_PRIVATE_BEGIN "---- BEGIN SSH2 ENCRYPTED PRIVATE KEY ----"
Kevin Stevesef4eea92001-02-05 12:42:17 +0000137#define SSH_COM_PRIVATE_KEY_MAGIC 0x3f6ff9eb
Damien Millereba71ba2000-04-29 23:57:08 +1000138
139void
140do_convert_to_ssh2(struct passwd *pw)
141{
Ben Lindstrom46c264f2001-04-24 16:56:58 +0000142 Key *k;
Damien Millereba71ba2000-04-29 23:57:08 +1000143 int len;
Ben Lindstrom46c16222000-12-22 01:43:59 +0000144 u_char *blob;
Damien Millereba71ba2000-04-29 23:57:08 +1000145 struct stat st;
146
147 if (!have_identity)
148 ask_filename(pw, "Enter file in which the key is");
149 if (stat(identity_file, &st) < 0) {
150 perror(identity_file);
151 exit(1);
152 }
Ben Lindstrom46c264f2001-04-24 16:56:58 +0000153 if ((k = key_load_public(identity_file, NULL)) == NULL) {
Ben Lindstromd78ae762001-06-05 20:35:09 +0000154 if ((k = load_identity(identity_file)) == NULL) {
Ben Lindstrom46c264f2001-04-24 16:56:58 +0000155 fprintf(stderr, "load failed\n");
156 exit(1);
157 }
Damien Millereba71ba2000-04-29 23:57:08 +1000158 }
Ben Lindstrom46c264f2001-04-24 16:56:58 +0000159 key_to_blob(k, &blob, &len);
Damien Miller874d77b2000-10-14 16:23:11 +1100160 fprintf(stdout, "%s\n", SSH_COM_PUBLIC_BEGIN);
Damien Millereba71ba2000-04-29 23:57:08 +1000161 fprintf(stdout,
Damien Miller874d77b2000-10-14 16:23:11 +1100162 "Comment: \"%d-bit %s, converted from OpenSSH by %s@%s\"\n",
Ben Lindstrom46c264f2001-04-24 16:56:58 +0000163 key_size(k), key_type(k),
Damien Millereba71ba2000-04-29 23:57:08 +1000164 pw->pw_name, hostname);
165 dump_base64(stdout, blob, len);
Damien Miller874d77b2000-10-14 16:23:11 +1100166 fprintf(stdout, "%s\n", SSH_COM_PUBLIC_END);
Ben Lindstrom46c264f2001-04-24 16:56:58 +0000167 key_free(k);
Damien Millereba71ba2000-04-29 23:57:08 +1000168 xfree(blob);
169 exit(0);
170}
171
172void
Damien Miller874d77b2000-10-14 16:23:11 +1100173buffer_get_bignum_bits(Buffer *b, BIGNUM *value)
174{
175 int bits = buffer_get_int(b);
176 int bytes = (bits + 7) / 8;
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000177
Damien Miller874d77b2000-10-14 16:23:11 +1100178 if (buffer_len(b) < bytes)
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000179 fatal("buffer_get_bignum_bits: input buffer too small: "
180 "need %d have %d", bytes, buffer_len(b));
Ben Lindstrom46c16222000-12-22 01:43:59 +0000181 BN_bin2bn((u_char *)buffer_ptr(b), bytes, value);
Damien Miller874d77b2000-10-14 16:23:11 +1100182 buffer_consume(b, bytes);
183}
184
185Key *
186do_convert_private_ssh2_from_blob(char *blob, int blen)
187{
188 Buffer b;
Damien Miller874d77b2000-10-14 16:23:11 +1100189 Key *key = NULL;
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000190 int ignore, magic, rlen, ktype;
Damien Miller874d77b2000-10-14 16:23:11 +1100191 char *type, *cipher;
192
193 buffer_init(&b);
194 buffer_append(&b, blob, blen);
195
196 magic = buffer_get_int(&b);
197 if (magic != SSH_COM_PRIVATE_KEY_MAGIC) {
198 error("bad magic 0x%x != 0x%x", magic, SSH_COM_PRIVATE_KEY_MAGIC);
199 buffer_free(&b);
200 return NULL;
201 }
202 ignore = buffer_get_int(&b);
203 type = buffer_get_string(&b, NULL);
204 cipher = buffer_get_string(&b, NULL);
205 ignore = buffer_get_int(&b);
206 ignore = buffer_get_int(&b);
207 ignore = buffer_get_int(&b);
Damien Miller874d77b2000-10-14 16:23:11 +1100208
209 if (strcmp(cipher, "none") != 0) {
210 error("unsupported cipher %s", cipher);
211 xfree(cipher);
212 buffer_free(&b);
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000213 xfree(type);
Damien Miller874d77b2000-10-14 16:23:11 +1100214 return NULL;
215 }
216 xfree(cipher);
217
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000218 if (strstr(type, "dsa")) {
219 ktype = KEY_DSA;
220 } else if (strstr(type, "rsa")) {
221 ktype = KEY_RSA;
222 } else {
223 xfree(type);
Damien Miller874d77b2000-10-14 16:23:11 +1100224 return NULL;
225 }
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000226 key = key_new_private(ktype);
227 xfree(type);
228
229 switch (key->type) {
230 case KEY_DSA:
231 buffer_get_bignum_bits(&b, key->dsa->p);
232 buffer_get_bignum_bits(&b, key->dsa->g);
233 buffer_get_bignum_bits(&b, key->dsa->q);
234 buffer_get_bignum_bits(&b, key->dsa->pub_key);
235 buffer_get_bignum_bits(&b, key->dsa->priv_key);
236 break;
237 case KEY_RSA:
238 if (!BN_set_word(key->rsa->e, (u_long) buffer_get_char(&b))) {
239 buffer_free(&b);
240 key_free(key);
241 return NULL;
242 }
243 buffer_get_bignum_bits(&b, key->rsa->d);
244 buffer_get_bignum_bits(&b, key->rsa->n);
245 buffer_get_bignum_bits(&b, key->rsa->iqmp);
246 buffer_get_bignum_bits(&b, key->rsa->q);
247 buffer_get_bignum_bits(&b, key->rsa->p);
248 generate_additional_parameters(key->rsa);
249 break;
250 }
Damien Miller874d77b2000-10-14 16:23:11 +1100251 rlen = buffer_len(&b);
252 if(rlen != 0)
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000253 error("do_convert_private_ssh2_from_blob: "
254 "remaining bytes in key blob %d", rlen);
Damien Miller874d77b2000-10-14 16:23:11 +1100255 buffer_free(&b);
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000256#ifdef DEBUG_PK
257 {
258 u_int slen;
259 u_char *sig, data[10] = "abcde12345";
260
261 key_sign(key, &sig, &slen, data, sizeof data);
262 key_verify(key, sig, slen, data, sizeof data);
Ben Lindstrom86ebcb62001-04-04 01:53:20 +0000263 xfree(sig);
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000264 }
265#endif
Damien Miller874d77b2000-10-14 16:23:11 +1100266 return key;
267}
268
269void
Damien Millereba71ba2000-04-29 23:57:08 +1000270do_convert_from_ssh2(struct passwd *pw)
271{
272 Key *k;
273 int blen;
274 char line[1024], *p;
275 char blob[8096];
276 char encoded[8096];
277 struct stat st;
Damien Miller874d77b2000-10-14 16:23:11 +1100278 int escaped = 0, private = 0, ok;
Damien Millereba71ba2000-04-29 23:57:08 +1000279 FILE *fp;
280
281 if (!have_identity)
282 ask_filename(pw, "Enter file in which the key is");
283 if (stat(identity_file, &st) < 0) {
284 perror(identity_file);
285 exit(1);
286 }
287 fp = fopen(identity_file, "r");
288 if (fp == NULL) {
289 perror(identity_file);
290 exit(1);
291 }
292 encoded[0] = '\0';
293 while (fgets(line, sizeof(line), fp)) {
Damien Miller30c3d422000-05-09 11:02:59 +1000294 if (!(p = strchr(line, '\n'))) {
295 fprintf(stderr, "input line too long.\n");
296 exit(1);
297 }
298 if (p > line && p[-1] == '\\')
299 escaped++;
Damien Millereba71ba2000-04-29 23:57:08 +1000300 if (strncmp(line, "----", 4) == 0 ||
301 strstr(line, ": ") != NULL) {
Damien Miller874d77b2000-10-14 16:23:11 +1100302 if (strstr(line, SSH_COM_PRIVATE_BEGIN) != NULL)
303 private = 1;
Ben Lindstrom30358602001-04-24 16:59:28 +0000304 /* fprintf(stderr, "ignore: %s", line); */
Damien Millereba71ba2000-04-29 23:57:08 +1000305 continue;
306 }
Damien Miller30c3d422000-05-09 11:02:59 +1000307 if (escaped) {
308 escaped--;
Ben Lindstrom30358602001-04-24 16:59:28 +0000309 /* fprintf(stderr, "escaped: %s", line); */
Damien Miller30c3d422000-05-09 11:02:59 +1000310 continue;
Damien Millereba71ba2000-04-29 23:57:08 +1000311 }
312 *p = '\0';
313 strlcat(encoded, line, sizeof(encoded));
314 }
Ben Lindstrom46c16222000-12-22 01:43:59 +0000315 blen = uudecode(encoded, (u_char *)blob, sizeof(blob));
Damien Millereba71ba2000-04-29 23:57:08 +1000316 if (blen < 0) {
317 fprintf(stderr, "uudecode failed.\n");
318 exit(1);
319 }
Damien Miller874d77b2000-10-14 16:23:11 +1100320 k = private ?
321 do_convert_private_ssh2_from_blob(blob, blen) :
Damien Miller0bc1bd82000-11-13 22:57:25 +1100322 key_from_blob(blob, blen);
Damien Miller874d77b2000-10-14 16:23:11 +1100323 if (k == NULL) {
324 fprintf(stderr, "decode blob failed.\n");
325 exit(1);
326 }
327 ok = private ?
Ben Lindstromd09fcf52001-03-29 00:29:54 +0000328 (k->type == KEY_DSA ?
329 PEM_write_DSAPrivateKey(stdout, k->dsa, NULL, NULL, 0, NULL, NULL) :
330 PEM_write_RSAPrivateKey(stdout, k->rsa, NULL, NULL, 0, NULL, NULL)) :
Damien Miller874d77b2000-10-14 16:23:11 +1100331 key_write(k, stdout);
332 if (!ok) {
333 fprintf(stderr, "key write failed");
334 exit(1);
335 }
Damien Millereba71ba2000-04-29 23:57:08 +1000336 key_free(k);
337 fprintf(stdout, "\n");
338 fclose(fp);
339 exit(0);
340}
341
342void
343do_print_public(struct passwd *pw)
344{
Ben Lindstromd0fca422001-03-26 13:44:06 +0000345 Key *prv;
Damien Millereba71ba2000-04-29 23:57:08 +1000346 struct stat st;
347
348 if (!have_identity)
349 ask_filename(pw, "Enter file in which the key is");
350 if (stat(identity_file, &st) < 0) {
351 perror(identity_file);
352 exit(1);
353 }
Ben Lindstromd78ae762001-06-05 20:35:09 +0000354 prv = load_identity(identity_file);
Ben Lindstromd0fca422001-03-26 13:44:06 +0000355 if (prv == NULL) {
Damien Millereba71ba2000-04-29 23:57:08 +1000356 fprintf(stderr, "load failed\n");
357 exit(1);
358 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000359 if (!key_write(prv, stdout))
Damien Millereba71ba2000-04-29 23:57:08 +1000360 fprintf(stderr, "key_write failed");
Ben Lindstromd0fca422001-03-26 13:44:06 +0000361 key_free(prv);
Damien Millereba71ba2000-04-29 23:57:08 +1000362 fprintf(stdout, "\n");
363 exit(0);
364}
365
Damien Miller10f6f6b1999-11-17 17:29:08 +1100366void
367do_fingerprint(struct passwd *pw)
368{
Damien Miller98c7ad62000-03-09 21:27:49 +1100369 FILE *f;
Damien Millereba71ba2000-04-29 23:57:08 +1000370 Key *public;
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000371 char *comment = NULL, *cp, *ep, line[16*1024], *fp;
Ben Lindstromd0fca422001-03-26 13:44:06 +0000372 int i, skip = 0, num = 1, invalid = 1, rep, fptype;
Damien Miller95def091999-11-25 00:26:21 +1100373 struct stat st;
Damien Miller10f6f6b1999-11-17 17:29:08 +1100374
Ben Lindstromd0fca422001-03-26 13:44:06 +0000375 fptype = print_bubblebabble ? SSH_FP_SHA1 : SSH_FP_MD5;
376 rep = print_bubblebabble ? SSH_FP_BUBBLEBABBLE : SSH_FP_HEX;
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000377
Damien Miller95def091999-11-25 00:26:21 +1100378 if (!have_identity)
379 ask_filename(pw, "Enter file in which the key is");
380 if (stat(identity_file, &st) < 0) {
381 perror(identity_file);
382 exit(1);
383 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000384 public = key_load_public(identity_file, &comment);
385 if (public != NULL) {
386 fp = key_fingerprint(public, fptype, rep);
387 printf("%d %s %s\n", key_size(public), fp, comment);
Damien Miller0bc1bd82000-11-13 22:57:25 +1100388 key_free(public);
389 xfree(comment);
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000390 xfree(fp);
Damien Miller98c7ad62000-03-09 21:27:49 +1100391 exit(0);
392 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000393 if (comment)
394 xfree(comment);
Damien Miller98c7ad62000-03-09 21:27:49 +1100395
396 f = fopen(identity_file, "r");
397 if (f != NULL) {
Damien Miller98c7ad62000-03-09 21:27:49 +1100398 while (fgets(line, sizeof(line), f)) {
399 i = strlen(line) - 1;
400 if (line[i] != '\n') {
401 error("line %d too long: %.40s...", num, line);
402 skip = 1;
403 continue;
Damien Miller95def091999-11-25 00:26:21 +1100404 }
Damien Miller98c7ad62000-03-09 21:27:49 +1100405 num++;
406 if (skip) {
407 skip = 0;
408 continue;
409 }
410 line[i] = '\0';
411
412 /* Skip leading whitespace, empty and comment lines. */
413 for (cp = line; *cp == ' ' || *cp == '\t'; cp++)
414 ;
415 if (!*cp || *cp == '\n' || *cp == '#')
416 continue ;
417 i = strtol(cp, &ep, 10);
418 if (i == 0 || ep == NULL || (*ep != ' ' && *ep != '\t')) {
419 int quoted = 0;
420 comment = cp;
421 for (; *cp && (quoted || (*cp != ' ' && *cp != '\t')); cp++) {
422 if (*cp == '\\' && cp[1] == '"')
423 cp++; /* Skip both */
424 else if (*cp == '"')
425 quoted = !quoted;
426 }
427 if (!*cp)
428 continue;
429 *cp++ = '\0';
430 }
431 ep = cp;
Ben Lindstrom2941f112000-12-29 16:50:13 +0000432 public = key_new(KEY_RSA1);
433 if (key_read(public, &cp) != 1) {
434 cp = ep;
435 key_free(public);
436 public = key_new(KEY_UNSPEC);
437 if (key_read(public, &cp) != 1) {
438 key_free(public);
439 continue;
440 }
Damien Miller98c7ad62000-03-09 21:27:49 +1100441 }
Ben Lindstrom2941f112000-12-29 16:50:13 +0000442 comment = *cp ? cp : comment;
Ben Lindstromd0fca422001-03-26 13:44:06 +0000443 fp = key_fingerprint(public, fptype, rep);
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000444 printf("%d %s %s\n", key_size(public), fp,
Ben Lindstrom2941f112000-12-29 16:50:13 +0000445 comment ? comment : "no comment");
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000446 xfree(fp);
Ben Lindstromd0fca422001-03-26 13:44:06 +0000447 key_free(public);
Ben Lindstrom2941f112000-12-29 16:50:13 +0000448 invalid = 0;
Damien Miller95def091999-11-25 00:26:21 +1100449 }
Damien Miller98c7ad62000-03-09 21:27:49 +1100450 fclose(f);
Damien Miller95def091999-11-25 00:26:21 +1100451 }
Damien Miller98c7ad62000-03-09 21:27:49 +1100452 if (invalid) {
453 printf("%s is not a valid key file.\n", identity_file);
454 exit(1);
455 }
Damien Miller95def091999-11-25 00:26:21 +1100456 exit(0);
Damien Miller10f6f6b1999-11-17 17:29:08 +1100457}
458
Damien Miller95def091999-11-25 00:26:21 +1100459/*
460 * Perform changing a passphrase. The argument is the passwd structure
461 * for the current user.
462 */
Damien Miller10f6f6b1999-11-17 17:29:08 +1100463void
464do_change_passphrase(struct passwd *pw)
465{
Damien Miller95def091999-11-25 00:26:21 +1100466 char *comment;
467 char *old_passphrase, *passphrase1, *passphrase2;
468 struct stat st;
Damien Millereba71ba2000-04-29 23:57:08 +1000469 Key *private;
Damien Miller10f6f6b1999-11-17 17:29:08 +1100470
Damien Miller95def091999-11-25 00:26:21 +1100471 if (!have_identity)
472 ask_filename(pw, "Enter file in which the key is");
Damien Miller95def091999-11-25 00:26:21 +1100473 if (stat(identity_file, &st) < 0) {
474 perror(identity_file);
475 exit(1);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000476 }
Damien Miller95def091999-11-25 00:26:21 +1100477 /* Try to load the file with empty passphrase. */
Ben Lindstromd0fca422001-03-26 13:44:06 +0000478 private = key_load_private(identity_file, "", &comment);
479 if (private == NULL) {
Damien Miller95def091999-11-25 00:26:21 +1100480 if (identity_passphrase)
481 old_passphrase = xstrdup(identity_passphrase);
482 else
483 old_passphrase = read_passphrase("Enter old passphrase: ", 1);
Ben Lindstromd0fca422001-03-26 13:44:06 +0000484 private = key_load_private(identity_file, old_passphrase , &comment);
485 memset(old_passphrase, 0, strlen(old_passphrase));
486 xfree(old_passphrase);
487 if (private == NULL) {
Damien Miller95def091999-11-25 00:26:21 +1100488 printf("Bad passphrase.\n");
489 exit(1);
490 }
Damien Miller95def091999-11-25 00:26:21 +1100491 }
492 printf("Key has comment '%s'\n", comment);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000493
Damien Miller95def091999-11-25 00:26:21 +1100494 /* Ask the new passphrase (twice). */
495 if (identity_new_passphrase) {
496 passphrase1 = xstrdup(identity_new_passphrase);
497 passphrase2 = NULL;
498 } else {
499 passphrase1 =
500 read_passphrase("Enter new passphrase (empty for no passphrase): ", 1);
501 passphrase2 = read_passphrase("Enter same passphrase again: ", 1);
502
503 /* Verify that they are the same. */
504 if (strcmp(passphrase1, passphrase2) != 0) {
505 memset(passphrase1, 0, strlen(passphrase1));
506 memset(passphrase2, 0, strlen(passphrase2));
507 xfree(passphrase1);
508 xfree(passphrase2);
509 printf("Pass phrases do not match. Try again.\n");
510 exit(1);
511 }
512 /* Destroy the other copy. */
513 memset(passphrase2, 0, strlen(passphrase2));
514 xfree(passphrase2);
515 }
516
517 /* Save the file using the new passphrase. */
Ben Lindstromd0fca422001-03-26 13:44:06 +0000518 if (!key_save_private(private, identity_file, passphrase1, comment)) {
Ben Lindstrom15f33862001-04-16 02:00:02 +0000519 printf("Saving the key failed: %s.\n", identity_file);
Damien Miller95def091999-11-25 00:26:21 +1100520 memset(passphrase1, 0, strlen(passphrase1));
521 xfree(passphrase1);
Damien Millereba71ba2000-04-29 23:57:08 +1000522 key_free(private);
Damien Miller95def091999-11-25 00:26:21 +1100523 xfree(comment);
524 exit(1);
525 }
526 /* Destroy the passphrase and the copy of the key in memory. */
527 memset(passphrase1, 0, strlen(passphrase1));
528 xfree(passphrase1);
Damien Millereba71ba2000-04-29 23:57:08 +1000529 key_free(private); /* Destroys contents */
Damien Miller95def091999-11-25 00:26:21 +1100530 xfree(comment);
531
532 printf("Your identification has been saved with the new passphrase.\n");
533 exit(0);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000534}
535
Damien Miller95def091999-11-25 00:26:21 +1100536/*
537 * Change the comment of a private key file.
538 */
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000539void
540do_change_comment(struct passwd *pw)
541{
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000542 char new_comment[1024], *comment, *passphrase;
Ben Lindstromd0fca422001-03-26 13:44:06 +0000543 Key *private;
544 Key *public;
Damien Miller95def091999-11-25 00:26:21 +1100545 struct stat st;
546 FILE *f;
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000547 int fd;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000548
Damien Miller95def091999-11-25 00:26:21 +1100549 if (!have_identity)
550 ask_filename(pw, "Enter file in which the key is");
Damien Miller95def091999-11-25 00:26:21 +1100551 if (stat(identity_file, &st) < 0) {
552 perror(identity_file);
553 exit(1);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000554 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000555 private = key_load_private(identity_file, "", &comment);
556 if (private == NULL) {
Damien Miller95def091999-11-25 00:26:21 +1100557 if (identity_passphrase)
558 passphrase = xstrdup(identity_passphrase);
559 else if (identity_new_passphrase)
560 passphrase = xstrdup(identity_new_passphrase);
561 else
562 passphrase = read_passphrase("Enter passphrase: ", 1);
563 /* Try to load using the passphrase. */
Ben Lindstromd0fca422001-03-26 13:44:06 +0000564 private = key_load_private(identity_file, passphrase, &comment);
565 if (private == NULL) {
Damien Miller95def091999-11-25 00:26:21 +1100566 memset(passphrase, 0, strlen(passphrase));
567 xfree(passphrase);
568 printf("Bad passphrase.\n");
569 exit(1);
570 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000571 } else {
572 passphrase = xstrdup("");
Damien Miller95def091999-11-25 00:26:21 +1100573 }
Ben Lindstromd0fca422001-03-26 13:44:06 +0000574 if (private->type != KEY_RSA1) {
575 fprintf(stderr, "Comments are only supported for RSA1 keys.\n");
576 key_free(private);
577 exit(1);
578 }
Damien Miller95def091999-11-25 00:26:21 +1100579 printf("Key now has comment '%s'\n", comment);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000580
Damien Miller95def091999-11-25 00:26:21 +1100581 if (identity_comment) {
582 strlcpy(new_comment, identity_comment, sizeof(new_comment));
583 } else {
584 printf("Enter new comment: ");
585 fflush(stdout);
586 if (!fgets(new_comment, sizeof(new_comment), stdin)) {
587 memset(passphrase, 0, strlen(passphrase));
Damien Millereba71ba2000-04-29 23:57:08 +1000588 key_free(private);
Damien Miller95def091999-11-25 00:26:21 +1100589 exit(1);
590 }
Damien Miller95def091999-11-25 00:26:21 +1100591 if (strchr(new_comment, '\n'))
592 *strchr(new_comment, '\n') = 0;
593 }
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000594
Damien Miller95def091999-11-25 00:26:21 +1100595 /* Save the file using the new passphrase. */
Ben Lindstromd0fca422001-03-26 13:44:06 +0000596 if (!key_save_private(private, identity_file, passphrase, new_comment)) {
Ben Lindstrom15f33862001-04-16 02:00:02 +0000597 printf("Saving the key failed: %s.\n", identity_file);
Damien Miller95def091999-11-25 00:26:21 +1100598 memset(passphrase, 0, strlen(passphrase));
599 xfree(passphrase);
Damien Millereba71ba2000-04-29 23:57:08 +1000600 key_free(private);
Damien Miller95def091999-11-25 00:26:21 +1100601 xfree(comment);
602 exit(1);
603 }
Damien Miller95def091999-11-25 00:26:21 +1100604 memset(passphrase, 0, strlen(passphrase));
605 xfree(passphrase);
Ben Lindstromd0fca422001-03-26 13:44:06 +0000606 public = key_from_private(private);
Damien Millereba71ba2000-04-29 23:57:08 +1000607 key_free(private);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000608
Damien Miller95def091999-11-25 00:26:21 +1100609 strlcat(identity_file, ".pub", sizeof(identity_file));
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000610 fd = open(identity_file, O_WRONLY | O_CREAT | O_TRUNC, 0644);
611 if (fd == -1) {
Damien Miller95def091999-11-25 00:26:21 +1100612 printf("Could not save your public key in %s\n", identity_file);
613 exit(1);
614 }
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000615 f = fdopen(fd, "w");
616 if (f == NULL) {
617 printf("fdopen %s failed", identity_file);
618 exit(1);
619 }
Damien Millereba71ba2000-04-29 23:57:08 +1000620 if (!key_write(public, f))
621 fprintf(stderr, "write key failed");
622 key_free(public);
623 fprintf(f, " %s\n", new_comment);
Damien Miller95def091999-11-25 00:26:21 +1100624 fclose(f);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000625
Damien Miller95def091999-11-25 00:26:21 +1100626 xfree(comment);
627
628 printf("The comment in your key file has been changed.\n");
629 exit(0);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000630}
631
Damien Miller431f66b1999-11-21 18:31:57 +1100632void
633usage(void)
634{
Ben Lindstrom2857d9c2001-04-22 17:19:46 +0000635 printf("Usage: %s [-ceilpqyB] [-t type] [-b bits] [-f file] [-C comment] "
Ben Lindstromb7c92322001-03-05 05:10:52 +0000636 "[-N new-pass] [-P pass]\n", __progname);
Damien Miller95def091999-11-25 00:26:21 +1100637 exit(1);
Damien Miller431f66b1999-11-21 18:31:57 +1100638}
639
Damien Miller95def091999-11-25 00:26:21 +1100640/*
641 * Main program for key management.
642 */
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000643int
644main(int ac, char **av)
645{
Damien Miller95def091999-11-25 00:26:21 +1100646 char dotsshdir[16 * 1024], comment[1024], *passphrase1, *passphrase2;
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000647 Key *private, *public;
Damien Miller95def091999-11-25 00:26:21 +1100648 struct passwd *pw;
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000649 int opt, type, fd;
Damien Miller95def091999-11-25 00:26:21 +1100650 struct stat st;
651 FILE *f;
Damien Miller0bc1bd82000-11-13 22:57:25 +1100652
Damien Miller95def091999-11-25 00:26:21 +1100653 extern int optind;
654 extern char *optarg;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000655
Ben Lindstrom49a79c02000-11-17 03:47:20 +0000656 __progname = get_progname(av[0]);
Damien Millerf9b625c2000-07-09 22:42:32 +1000657 init_rng();
Damien Miller60bc5172001-03-19 09:38:15 +1100658 seed_rng();
Damien Millerf9b625c2000-07-09 22:42:32 +1000659
Damien Millerd3a18572000-06-07 19:55:44 +1000660 SSLeay_add_all_algorithms();
Damien Millereba71ba2000-04-29 23:57:08 +1000661
Damien Miller5428f641999-11-25 11:54:57 +1100662 /* we need this for the home * directory. */
Damien Miller95def091999-11-25 00:26:21 +1100663 pw = getpwuid(getuid());
664 if (!pw) {
665 printf("You don't exist, go away!\n");
666 exit(1);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000667 }
Damien Millereba71ba2000-04-29 23:57:08 +1000668 if (gethostname(hostname, sizeof(hostname)) < 0) {
669 perror("gethostname");
670 exit(1);
671 }
Damien Miller5428f641999-11-25 11:54:57 +1100672
Ben Lindstrom5a707822001-04-22 17:15:46 +0000673 while ((opt = getopt(ac, av, "deiqpclBRxXyb:f:t:P:N:C:")) != -1) {
Damien Miller95def091999-11-25 00:26:21 +1100674 switch (opt) {
675 case 'b':
676 bits = atoi(optarg);
677 if (bits < 512 || bits > 32768) {
678 printf("Bits has bad value.\n");
679 exit(1);
680 }
681 break;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000682
Damien Miller95def091999-11-25 00:26:21 +1100683 case 'l':
684 print_fingerprint = 1;
685 break;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000686
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000687 case 'B':
688 print_bubblebabble = 1;
689 break;
690
Damien Miller95def091999-11-25 00:26:21 +1100691 case 'p':
692 change_passphrase = 1;
693 break;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000694
Damien Miller95def091999-11-25 00:26:21 +1100695 case 'c':
696 change_comment = 1;
697 break;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000698
Damien Miller95def091999-11-25 00:26:21 +1100699 case 'f':
700 strlcpy(identity_file, optarg, sizeof(identity_file));
701 have_identity = 1;
702 break;
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000703
Damien Miller95def091999-11-25 00:26:21 +1100704 case 'P':
705 identity_passphrase = optarg;
706 break;
707
708 case 'N':
709 identity_new_passphrase = optarg;
710 break;
711
712 case 'C':
713 identity_comment = optarg;
714 break;
715
716 case 'q':
717 quiet = 1;
718 break;
719
Damien Millereba71ba2000-04-29 23:57:08 +1000720 case 'R':
Damien Miller0bc1bd82000-11-13 22:57:25 +1100721 /* unused */
722 exit(0);
Damien Millereba71ba2000-04-29 23:57:08 +1000723 break;
724
Ben Lindstrom5a707822001-04-22 17:15:46 +0000725 case 'e':
Damien Millereba71ba2000-04-29 23:57:08 +1000726 case 'x':
Ben Lindstrom5a707822001-04-22 17:15:46 +0000727 /* export key */
Damien Millereba71ba2000-04-29 23:57:08 +1000728 convert_to_ssh2 = 1;
729 break;
730
Ben Lindstrom5a707822001-04-22 17:15:46 +0000731 case 'i':
Damien Millereba71ba2000-04-29 23:57:08 +1000732 case 'X':
Ben Lindstrom5a707822001-04-22 17:15:46 +0000733 /* import key */
Damien Millereba71ba2000-04-29 23:57:08 +1000734 convert_from_ssh2 = 1;
735 break;
736
737 case 'y':
738 print_public = 1;
739 break;
740
741 case 'd':
Damien Miller0bc1bd82000-11-13 22:57:25 +1100742 key_type_name = "dsa";
Damien Millereba71ba2000-04-29 23:57:08 +1000743 break;
744
Damien Miller0bc1bd82000-11-13 22:57:25 +1100745 case 't':
746 key_type_name = optarg;
Damien Miller0bc1bd82000-11-13 22:57:25 +1100747 break;
748
Damien Miller95def091999-11-25 00:26:21 +1100749 case '?':
750 default:
751 usage();
752 }
753 }
754 if (optind < ac) {
755 printf("Too many arguments.\n");
756 usage();
757 }
758 if (change_passphrase && change_comment) {
759 printf("Can only have one of -p and -c.\n");
760 usage();
761 }
Ben Lindstrom8fd372b2001-03-12 03:02:17 +0000762 if (print_fingerprint || print_bubblebabble)
Damien Miller95def091999-11-25 00:26:21 +1100763 do_fingerprint(pw);
Damien Miller95def091999-11-25 00:26:21 +1100764 if (change_passphrase)
765 do_change_passphrase(pw);
Damien Miller95def091999-11-25 00:26:21 +1100766 if (change_comment)
767 do_change_comment(pw);
Damien Millereba71ba2000-04-29 23:57:08 +1000768 if (convert_to_ssh2)
769 do_convert_to_ssh2(pw);
770 if (convert_from_ssh2)
771 do_convert_from_ssh2(pw);
772 if (print_public)
773 do_print_public(pw);
Damien Miller95def091999-11-25 00:26:21 +1100774
775 arc4random_stir();
776
Damien Millere39cacc2000-11-29 12:18:44 +1100777 type = key_type_from_name(key_type_name);
778 if (type == KEY_UNSPEC) {
779 fprintf(stderr, "unknown key type %s\n", key_type_name);
780 exit(1);
Damien Millereba71ba2000-04-29 23:57:08 +1000781 }
Damien Miller0bc1bd82000-11-13 22:57:25 +1100782 if (!quiet)
Damien Millere39cacc2000-11-29 12:18:44 +1100783 printf("Generating public/private %s key pair.\n", key_type_name);
Damien Miller0bc1bd82000-11-13 22:57:25 +1100784 private = key_generate(type, bits);
785 if (private == NULL) {
786 fprintf(stderr, "key_generate failed");
787 exit(1);
788 }
789 public = key_from_private(private);
Damien Miller95def091999-11-25 00:26:21 +1100790
791 if (!have_identity)
792 ask_filename(pw, "Enter file in which to save the key");
793
794 /* Create ~/.ssh directory if it doesn\'t already exist. */
Ben Lindstrom226cfa02001-01-22 05:34:40 +0000795 snprintf(dotsshdir, sizeof dotsshdir, "%s/%s", pw->pw_dir, _PATH_SSH_USER_DIR);
Damien Miller95def091999-11-25 00:26:21 +1100796 if (strstr(identity_file, dotsshdir) != NULL &&
797 stat(dotsshdir, &st) < 0) {
Damien Millerbe484b52000-07-15 14:14:16 +1000798 if (mkdir(dotsshdir, 0700) < 0)
Damien Miller95def091999-11-25 00:26:21 +1100799 error("Could not create directory '%s'.", dotsshdir);
800 else if (!quiet)
801 printf("Created directory '%s'.\n", dotsshdir);
802 }
803 /* If the file already exists, ask the user to confirm. */
804 if (stat(identity_file, &st) >= 0) {
805 char yesno[3];
806 printf("%s already exists.\n", identity_file);
807 printf("Overwrite (y/n)? ");
808 fflush(stdout);
809 if (fgets(yesno, sizeof(yesno), stdin) == NULL)
810 exit(1);
811 if (yesno[0] != 'y' && yesno[0] != 'Y')
812 exit(1);
813 }
814 /* Ask for a passphrase (twice). */
815 if (identity_passphrase)
816 passphrase1 = xstrdup(identity_passphrase);
817 else if (identity_new_passphrase)
818 passphrase1 = xstrdup(identity_new_passphrase);
819 else {
820passphrase_again:
821 passphrase1 =
822 read_passphrase("Enter passphrase (empty for no passphrase): ", 1);
823 passphrase2 = read_passphrase("Enter same passphrase again: ", 1);
824 if (strcmp(passphrase1, passphrase2) != 0) {
825 /* The passphrases do not match. Clear them and retry. */
826 memset(passphrase1, 0, strlen(passphrase1));
827 memset(passphrase2, 0, strlen(passphrase2));
828 xfree(passphrase1);
829 xfree(passphrase2);
830 printf("Passphrases do not match. Try again.\n");
831 goto passphrase_again;
832 }
833 /* Clear the other copy of the passphrase. */
834 memset(passphrase2, 0, strlen(passphrase2));
835 xfree(passphrase2);
836 }
837
Damien Miller95def091999-11-25 00:26:21 +1100838 if (identity_comment) {
839 strlcpy(comment, identity_comment, sizeof(comment));
840 } else {
Damien Miller4af51302000-04-16 11:18:38 +1000841 /* Create default commend field for the passphrase. */
Damien Miller95def091999-11-25 00:26:21 +1100842 snprintf(comment, sizeof comment, "%s@%s", pw->pw_name, hostname);
843 }
844
845 /* Save the key with the given passphrase and comment. */
Ben Lindstromd0fca422001-03-26 13:44:06 +0000846 if (!key_save_private(private, identity_file, passphrase1, comment)) {
Ben Lindstrom15f33862001-04-16 02:00:02 +0000847 printf("Saving the key failed: %s.\n", identity_file);
Damien Miller95def091999-11-25 00:26:21 +1100848 memset(passphrase1, 0, strlen(passphrase1));
849 xfree(passphrase1);
850 exit(1);
851 }
852 /* Clear the passphrase. */
853 memset(passphrase1, 0, strlen(passphrase1));
854 xfree(passphrase1);
855
856 /* Clear the private key and the random number generator. */
Damien Miller0bc1bd82000-11-13 22:57:25 +1100857 key_free(private);
Damien Miller95def091999-11-25 00:26:21 +1100858 arc4random_stir();
859
860 if (!quiet)
861 printf("Your identification has been saved in %s.\n", identity_file);
862
Damien Miller95def091999-11-25 00:26:21 +1100863 strlcat(identity_file, ".pub", sizeof(identity_file));
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000864 fd = open(identity_file, O_WRONLY | O_CREAT | O_TRUNC, 0644);
865 if (fd == -1) {
Damien Miller95def091999-11-25 00:26:21 +1100866 printf("Could not save your public key in %s\n", identity_file);
867 exit(1);
868 }
Ben Lindstrom5fc62702001-03-09 18:19:24 +0000869 f = fdopen(fd, "w");
870 if (f == NULL) {
871 printf("fdopen %s failed", identity_file);
872 exit(1);
873 }
Damien Millereba71ba2000-04-29 23:57:08 +1000874 if (!key_write(public, f))
875 fprintf(stderr, "write key failed");
876 fprintf(f, " %s\n", comment);
Damien Miller95def091999-11-25 00:26:21 +1100877 fclose(f);
878
879 if (!quiet) {
Ben Lindstromcfccef92001-03-13 04:57:58 +0000880 char *fp = key_fingerprint(public, SSH_FP_MD5, SSH_FP_HEX);
Damien Millereba71ba2000-04-29 23:57:08 +1000881 printf("Your public key has been saved in %s.\n",
882 identity_file);
Damien Miller95def091999-11-25 00:26:21 +1100883 printf("The key fingerprint is:\n");
Ben Lindstromcfccef92001-03-13 04:57:58 +0000884 printf("%s %s\n", fp, comment);
885 xfree(fp);
Damien Miller95def091999-11-25 00:26:21 +1100886 }
Damien Millereba71ba2000-04-29 23:57:08 +1000887
888 key_free(public);
Damien Miller95def091999-11-25 00:26:21 +1100889 exit(0);
Damien Millerd4a8b7e1999-10-27 13:42:43 +1000890}