2000-05-15 05:24:25 +00:00
|
|
|
/* $FreeBSD$ */
|
|
|
|
|
2000-02-24 14:29:47 +00:00
|
|
|
#include "includes.h"
|
2000-05-15 04:37:24 +00:00
|
|
|
RCSID("$Id: auth-skey.c,v 1.6 2000/04/14 10:30:29 markus Exp $");
|
2000-02-24 14:29:47 +00:00
|
|
|
|
2000-05-15 05:24:25 +00:00
|
|
|
#include <sys/sysctl.h>
|
2000-02-24 14:29:47 +00:00
|
|
|
#include "ssh.h"
|
|
|
|
#include "packet.h"
|
2000-05-15 05:24:25 +00:00
|
|
|
#include <sha.h>
|
2000-02-24 14:29:47 +00:00
|
|
|
|
2000-05-15 04:37:24 +00:00
|
|
|
/*
|
2000-02-24 14:29:47 +00:00
|
|
|
* try skey authentication,
|
2000-05-15 04:37:24 +00:00
|
|
|
* return 1 on success, 0 on failure, -1 if skey is not available
|
2000-02-24 14:29:47 +00:00
|
|
|
*/
|
|
|
|
|
2000-05-15 04:37:24 +00:00
|
|
|
int
|
2000-02-24 14:29:47 +00:00
|
|
|
auth_skey_password(struct passwd * pw, const char *password)
|
|
|
|
{
|
|
|
|
if (strncasecmp(password, "s/key", 5) == 0) {
|
2000-05-15 05:24:25 +00:00
|
|
|
char *skeyinfo = opie_keyinfo(pw->pw_name);
|
2000-02-24 14:29:47 +00:00
|
|
|
if (skeyinfo == NULL) {
|
|
|
|
debug("generating fake skeyinfo for %.100s.",
|
|
|
|
pw->pw_name);
|
|
|
|
skeyinfo = skey_fake_keyinfo(pw->pw_name);
|
|
|
|
}
|
|
|
|
if (skeyinfo != NULL)
|
|
|
|
packet_send_debug(skeyinfo);
|
|
|
|
/* Try again. */
|
|
|
|
return 0;
|
2000-05-15 05:24:25 +00:00
|
|
|
} else if (opie_haskey(pw->pw_name) == 0 &&
|
|
|
|
opie_passverify(pw->pw_name, (char *) password) != -1) {
|
2000-02-24 14:29:47 +00:00
|
|
|
/* Authentication succeeded. */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
/* Fall back to ordinary passwd authentication. */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* from %OpenBSD: skeylogin.c,v 1.32 1999/08/16 14:46:56 millert Exp % */
|
|
|
|
|
|
|
|
#define ROUND(x) (((x)[0] << 24) + (((x)[1]) << 16) + (((x)[2]) << 8) + \
|
|
|
|
((x)[3]))
|
|
|
|
|
|
|
|
/*
|
|
|
|
* hash_collapse()
|
|
|
|
*/
|
|
|
|
static u_int32_t
|
|
|
|
hash_collapse(s)
|
2000-05-15 04:37:24 +00:00
|
|
|
u_char *s;
|
2000-02-24 14:29:47 +00:00
|
|
|
{
|
2000-05-15 04:37:24 +00:00
|
|
|
int len, target;
|
2000-02-24 14:29:47 +00:00
|
|
|
u_int32_t i;
|
|
|
|
|
|
|
|
if ((strlen(s) % sizeof(u_int32_t)) == 0)
|
2000-05-15 04:37:24 +00:00
|
|
|
target = strlen(s); /* Multiple of 4 */
|
2000-02-24 14:29:47 +00:00
|
|
|
else
|
|
|
|
target = strlen(s) - (strlen(s) % sizeof(u_int32_t));
|
2000-05-15 04:37:24 +00:00
|
|
|
|
2000-02-24 14:29:47 +00:00
|
|
|
for (i = 0, len = 0; len < target; len += 4)
|
2000-05-15 04:37:24 +00:00
|
|
|
i ^= ROUND(s + len);
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
|
|
|
skey_fake_keyinfo(char *username)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
u_int ptr;
|
2000-05-15 05:24:25 +00:00
|
|
|
u_char hseed[OPIE_SEED_MAX], flg = 1, *up;
|
|
|
|
char pbuf[OPIE_SECRET_MAX+1];
|
|
|
|
static char skeyprompt[OPIE_CHALLENGE_MAX+1];
|
2000-02-24 14:29:47 +00:00
|
|
|
char *secret = NULL;
|
|
|
|
size_t secretlen = 0;
|
|
|
|
SHA1_CTX ctx;
|
|
|
|
char *p, *u;
|
2000-05-15 05:24:25 +00:00
|
|
|
int mib[2];
|
|
|
|
size_t size;
|
|
|
|
struct timeval boottime;
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
/*
|
2000-05-15 05:24:25 +00:00
|
|
|
* Base first 2 chars of seed on hostname.
|
2000-02-24 14:29:47 +00:00
|
|
|
* Add some filler for short hostnames if necessary.
|
|
|
|
*/
|
|
|
|
if (gethostname(pbuf, sizeof(pbuf)) == -1)
|
|
|
|
*(p = pbuf) = '.';
|
|
|
|
else
|
|
|
|
for (p = pbuf; *p && isalnum(*p); p++)
|
|
|
|
if (isalpha(*p) && isupper(*p))
|
|
|
|
*p = tolower(*p);
|
2000-05-15 05:24:25 +00:00
|
|
|
if (*p && pbuf - p < 2)
|
|
|
|
(void)strncpy(p, "asjd", 2 - (pbuf - p));
|
|
|
|
pbuf[2] = '\0';
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
/* Hash the username if possible */
|
2000-05-15 05:24:25 +00:00
|
|
|
if ((up = SHA1_Data(username, strlen(username), NULL)) != NULL) {
|
2000-02-24 14:29:47 +00:00
|
|
|
struct stat sb;
|
|
|
|
time_t t;
|
|
|
|
|
|
|
|
/* Collapse the hash */
|
|
|
|
ptr = hash_collapse(up);
|
|
|
|
memset(up, 0, strlen(up));
|
|
|
|
|
2000-05-15 05:24:25 +00:00
|
|
|
/*
|
|
|
|
* Seed the fake challenge with the system boot time,
|
|
|
|
* otherwise use ctime.
|
|
|
|
*
|
|
|
|
* XXX This should be a random source which is constant
|
|
|
|
* over short time periods, but changes over timescales on
|
|
|
|
* the order of a week.
|
|
|
|
*/
|
|
|
|
mib[0] = CTL_KERN;
|
|
|
|
mib[1] = KERN_BOOTTIME;
|
|
|
|
size = sizeof(boottime);
|
2000-09-03 07:58:35 +00:00
|
|
|
bzero(&boottime, sizeof(boottime));
|
2000-05-15 05:24:25 +00:00
|
|
|
if (sysctl(mib, 2, &boottime, &size, NULL, 0) != -1 &&
|
|
|
|
boottime.tv_sec != 0) {
|
|
|
|
secret = (char *)&boottime;
|
|
|
|
secretlen = size/sizeof(char);
|
2000-02-24 14:29:47 +00:00
|
|
|
flg = 0;
|
|
|
|
} else if (!stat(_PATH_MEM, &sb) || !stat("/", &sb)) {
|
|
|
|
t = sb.st_ctime;
|
|
|
|
secret = ctime(&t);
|
|
|
|
secretlen = strlen(secret);
|
|
|
|
flg = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Put that in your pipe and smoke it */
|
|
|
|
if (flg == 0) {
|
|
|
|
/* Hash secret value with username */
|
2000-05-15 05:24:25 +00:00
|
|
|
SHA1_Init(&ctx);
|
|
|
|
SHA1_Update(&ctx, secret, secretlen);
|
|
|
|
SHA1_Update(&ctx, username, strlen(username));
|
|
|
|
SHA1_End(&ctx, up);
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
/* Zero out */
|
|
|
|
memset(secret, 0, secretlen);
|
|
|
|
|
|
|
|
/* Now hash the hash */
|
2000-05-15 05:24:25 +00:00
|
|
|
SHA1_Init(&ctx);
|
|
|
|
SHA1_Update(&ctx, up, strlen(up));
|
|
|
|
SHA1_End(&ctx, up);
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
ptr = hash_collapse(up + 4);
|
|
|
|
|
2000-05-15 05:24:25 +00:00
|
|
|
for (i = 2; i < 6; i++) {
|
2000-02-24 14:29:47 +00:00
|
|
|
pbuf[i] = (ptr % 10) + '0';
|
|
|
|
ptr /= 10;
|
|
|
|
}
|
|
|
|
pbuf[i] = '\0';
|
|
|
|
|
|
|
|
/* Sequence number */
|
2000-05-15 05:24:25 +00:00
|
|
|
ptr = ((up[2] + up[3]) % 499) + 1;
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
memset(up, 0, 20); /* SHA1 specific */
|
|
|
|
free(up);
|
|
|
|
|
|
|
|
(void)snprintf(skeyprompt, sizeof skeyprompt,
|
2000-09-02 04:41:33 +00:00
|
|
|
"otp-%.*s %d %.*s ext",
|
2000-05-15 05:24:25 +00:00
|
|
|
OPIE_HASHNAME_MAX,
|
|
|
|
opie_get_algorithm(),
|
|
|
|
ptr, OPIE_SEED_MAX,
|
2000-02-24 14:29:47 +00:00
|
|
|
pbuf);
|
|
|
|
} else {
|
2000-05-15 05:24:25 +00:00
|
|
|
/* Base last 4 chars of seed on username */
|
2000-02-24 14:29:47 +00:00
|
|
|
u = username;
|
2000-05-15 05:24:25 +00:00
|
|
|
i = 4;
|
|
|
|
p = &pbuf[2];
|
2000-02-24 14:29:47 +00:00
|
|
|
do {
|
|
|
|
if (*u == 0) {
|
|
|
|
/* Pad remainder with zeros */
|
|
|
|
while (--i >= 0)
|
|
|
|
*p++ = '0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p++ = (*u++ % 10) + '0';
|
|
|
|
} while (--i != 0);
|
2000-05-15 05:24:25 +00:00
|
|
|
pbuf[6] = '\0';
|
2000-02-24 14:29:47 +00:00
|
|
|
|
|
|
|
(void)snprintf(skeyprompt, sizeof skeyprompt,
|
2000-09-02 04:41:33 +00:00
|
|
|
"otp-md5 %d %.*s ext",
|
2000-05-15 05:24:25 +00:00
|
|
|
499, OPIE_SEED_MAX, pbuf);
|
2000-02-24 14:29:47 +00:00
|
|
|
}
|
|
|
|
return skeyprompt;
|
|
|
|
}
|