X-Git-Url: https://mattmccutchen.net/rsync/rsync.git/blobdiff_plain/3b83a22057b71e7df2b960d3997fb4474910f30e..60c25caa64235ebc82f4684df8759811700da03c:/uidlist.c diff --git a/uidlist.c b/uidlist.c index 5d3d1f38..f771fb6b 100644 --- a/uidlist.c +++ b/uidlist.c @@ -74,7 +74,7 @@ static struct idlist *add_to_list(struct idlist **root, id_t id, const char *nam } /* turn a uid into a user name */ -static const char *uid_to_name(uid_t uid) +char *uid_to_user(uid_t uid) { struct passwd *pass = getpwuid(uid); if (pass) @@ -83,7 +83,7 @@ static const char *uid_to_name(uid_t uid) } /* turn a gid into a group name */ -static const char *gid_to_name(gid_t gid) +char *gid_to_group(gid_t gid) { struct group *grp = getgrgid(gid); if (grp) @@ -91,6 +91,38 @@ static const char *gid_to_name(gid_t gid) return NULL; } +/* Parse a user name or (optionally) a number into a uid */ +int user_to_uid(const char *name, uid_t *uid_p, BOOL num_ok) +{ + struct passwd *pass; + if (!name || !*name) + return 0; + if (num_ok && name[strspn(name, "0123456789")] == '\0') { + *uid_p = atol(name); + return 1; + } + if (!(pass = getpwnam(name))) + return 0; + *uid_p = pass->pw_uid; + return 1; +} + +/* Parse a group name or (optionally) a number into a gid */ +int group_to_gid(const char *name, gid_t *gid_p, BOOL num_ok) +{ + struct group *grp; + if (!name || !*name) + return 0; + if (num_ok && name[strspn(name, "0123456789")] == '\0') { + *gid_p = atol(name); + return 1; + } + if (!(grp = getgrnam(name))) + return 0; + *gid_p = grp->gr_gid; + return 1; +} + static int is_in_group(gid_t gid) { #ifdef HAVE_GETGROUPS @@ -162,7 +194,7 @@ static struct idlist *recv_add_id(struct idlist **idlist_ptr, struct idlist *idm if (strcmp(node->name, name) != 0) continue; } else if (node->name) { - if (id < node->id || id > (unsigned long)node->name) + if (id < node->id || (unsigned long)id > (unsigned long)node->name) continue; } else { if (node->id != id) @@ -253,7 +285,7 @@ const char *add_uid(uid_t uid) return NULL; } - node = add_to_list(&uidlist, uid, uid_to_name(uid), 0, 0); + node = add_to_list(&uidlist, uid, uid_to_user(uid), 0, 0); return node->name; } @@ -271,7 +303,7 @@ const char *add_gid(gid_t gid) return NULL; } - node = add_to_list(&gidlist, gid, gid_to_name(gid), 0, 0); + node = add_to_list(&gidlist, gid, gid_to_group(gid), 0, 0); return node->name; } @@ -457,5 +489,29 @@ void parse_name_map(char *map, BOOL usernames) /* The 0 user/group doesn't get its name sent, so add it explicitly. */ recv_add_id(idlist_ptr, *idmap_ptr, 0, - numeric_ids ? NULL : usernames ? uid_to_name(0) : gid_to_name(0)); + numeric_ids ? NULL : usernames ? uid_to_user(0) : gid_to_group(0)); } + +#ifdef HAVE_GETGROUPLIST +const char *getallgroups(uid_t uid, gid_t *gid_list, int *size_ptr) +{ + struct passwd *pw; + if ((pw = getpwuid(uid)) == NULL) + return "getpwuid failed"; + /* Get all the process's groups, with the pw_gid group first. */ + if (getgrouplist(pw->pw_name, pw->pw_gid, gid_list, size_ptr) < 0) + return "getgrouplist failed"; + /* Paranoia: is the default group not first in the list? */ + if (gid_list[0] != pw->pw_gid) { + int j; + for (j = 0; j < *size_ptr; j++) { + if (gid_list[j] == pw->pw_gid) { + gid_list[j] = gid_list[0]; + gid_list[0] = pw->pw_gid; + break; + } + } + } + return NULL; +} +#endif