diff options
author | des <des@FreeBSD.org> | 2002-05-08 00:48:39 +0000 |
---|---|---|
committer | des <des@FreeBSD.org> | 2002-05-08 00:48:39 +0000 |
commit | bc41fe09348fcdbb228776ca0336ea8ec789c81b (patch) | |
tree | 7af707159fd83dc1cdec07c08ec9a050e6aed030 | |
parent | ffc89486bfbf0dbba369b8bc4d622dc1257cb7a9 (diff) | |
download | FreeBSD-src-bc41fe09348fcdbb228776ca0336ea8ec789c81b.zip FreeBSD-src-bc41fe09348fcdbb228776ca0336ea8ec789c81b.tar.gz |
Add ypclnt_havepasswd(), and guard against a possible segfault.
Sponsored by: DARPA, NAI Labs
-rw-r--r-- | lib/libypclnt/ypclnt.h | 1 | ||||
-rw-r--r-- | lib/libypclnt/ypclnt_passwd.c | 44 |
2 files changed, 33 insertions, 12 deletions
diff --git a/lib/libypclnt/ypclnt.h b/lib/libypclnt/ypclnt.h index 76bca52..476981c 100644 --- a/lib/libypclnt/ypclnt.h +++ b/lib/libypclnt/ypclnt.h @@ -50,6 +50,7 @@ ypclnt_t *ypclnt_new(const char *, const char *, const char *); void ypclnt_free(ypclnt_t *); void ypclnt_error(ypclnt_t *, const char *, const char *, ...); int ypclnt_connect(ypclnt_t *); +int ypclnt_havepasswdd(ypclnt_t *); int ypclnt_passwd(ypclnt_t *, const struct passwd *, const char *); #if defined(DEBUG) && defined(__GNUC__) diff --git a/lib/libypclnt/ypclnt_passwd.c b/lib/libypclnt/ypclnt_passwd.c index 61e6f7b..0dd6569 100644 --- a/lib/libypclnt/ypclnt_passwd.c +++ b/lib/libypclnt/ypclnt_passwd.c @@ -56,8 +56,13 @@ static int yppasswd_remote(ypclnt_t *, const struct passwd *, const char *); static int yppasswd_local(ypclnt_t *, const struct passwd *, const char *); +/* + * Determines the availability of rpc.yppasswdd. Returns -1 for not + * available (or unable to determine), 0 for available, 1 for available in + * master mode. + */ int -ypclnt_passwd(ypclnt_t *ypclnt, const struct passwd *pwd, const char *passwd) +ypclnt_havepasswdd(ypclnt_t *ypclnt) { struct addrinfo hints, *res; int sd; @@ -71,7 +76,7 @@ ypclnt_passwd(ypclnt_t *ypclnt, const struct passwd *pwd, const char *passwd) /* if we're not root, use remote method */ if (getuid() != 0) - goto remote; + return (0); /* try to determine if we are the server */ memset(&hints, 0, sizeof(hints)); @@ -79,24 +84,39 @@ ypclnt_passwd(ypclnt_t *ypclnt, const struct passwd *pwd, const char *passwd) hints.ai_socktype = SOCK_STREAM; hints.ai_protocol = 0; if (getaddrinfo(ypclnt->server, NULL, &hints, &res) != 0) - goto remote; + return (0); sd = socket(res->ai_family, res->ai_socktype, res->ai_protocol); if (sd == -1) { freeaddrinfo(res); - goto remote; + return (0); } if (bind(sd, res->ai_addr, res->ai_addrlen) == -1) { close(sd); freeaddrinfo(res); - goto remote; + return (0); } freeaddrinfo(res); close(sd); - YPCLNT_DEBUG("using local update method"); - return (yppasswd_local(ypclnt, pwd, passwd)); - remote: - YPCLNT_DEBUG("using remote update method"); - return (yppasswd_remote(ypclnt, pwd, passwd)); + return (1); +} + +/* + * Updates the NIS user information for the specified user. + */ +int +ypclnt_passwd(ypclnt_t *ypclnt, const struct passwd *pwd, const char *passwd) +{ + switch (ypclnt_havepasswdd(ypclnt)) { + case 0: + YPCLNT_DEBUG("using remote update method"); + return (yppasswd_remote(ypclnt, pwd, passwd)); + case 1: + YPCLNT_DEBUG("using local update method"); + return (yppasswd_local(ypclnt, pwd, passwd)); + default: + YPCLNT_DEBUG("no rpc.yppasswdd"); + return (-1); + } } /* @@ -127,7 +147,7 @@ yppasswd_local(ypclnt_t *ypclnt, const struct passwd *pwd, const char *passwd) (yppwd.newpw.pw_gecos = strdup(pwd->pw_gecos)) == NULL || (yppwd.newpw.pw_dir = strdup(pwd->pw_dir)) == NULL || (yppwd.newpw.pw_shell = strdup(pwd->pw_shell)) == NULL || - (yppwd.oldpass = strdup(passwd)) == NULL) { + (yppwd.oldpass = strdup(passwd ? passwd : "")) == NULL) { ypclnt_error(ypclnt, __func__, strerror(errno)); ret = -1; goto done; @@ -210,7 +230,7 @@ yppasswd_remote(ypclnt_t *ypclnt, const struct passwd *pwd, const char *passwd) (yppwd.newpw.pw_gecos = strdup(pwd->pw_gecos)) == NULL || (yppwd.newpw.pw_dir = strdup(pwd->pw_dir)) == NULL || (yppwd.newpw.pw_shell = strdup(pwd->pw_shell)) == NULL || - (yppwd.oldpass = strdup(passwd)) == NULL) { + (yppwd.oldpass = strdup(passwd ? passwd : "")) == NULL) { ypclnt_error(ypclnt, __func__, strerror(errno)); ret = -1; goto done; |