X-Git-Url: https://git.octo.it/?a=blobdiff_plain;f=src%2Fexec.c;h=77b137556ef0f9813e679c6b3fe99af7f36e6c06;hb=cb6b575607f0fd698ed4ef1ba1e3313139c61c21;hp=2c99ce2a7f296395b3e682d2f85ada85af95a5a9;hpb=436dacdb577300ad6ed4692c95089f22ae98de3c;p=collectd.git diff --git a/src/exec.c b/src/exec.c index 2c99ce2a..77b13755 100644 --- a/src/exec.c +++ b/src/exec.c @@ -39,7 +39,6 @@ #include #include -#include #ifdef HAVE_SYS_CAPABILITY_H #include #endif @@ -346,6 +345,65 @@ static void close_pipe(int fd_pipe[2]) /* {{{ */ } /* }}} void close_pipe */ /* + * Get effective group ID from group name. + * Input arguments: + * pl :program list struct with group name + * gid :group id to fallback in case egid cannot be determined. + * Returns: + * egid effective group id if successfull, + * -1 if group is not defined/not found. + * -2 for any buffer allocation error. + */ +static int getegr_id(program_list_t *pl, int gid) /* {{{ */ +{ + if (pl->group == NULL) { + return -1; + } + if (strcmp(pl->group, "") == 0) { + return gid; + } + struct group *gr_ptr = NULL; + struct group gr; + + long int grbuf_size = sysconf(_SC_GETGR_R_SIZE_MAX); + if (grbuf_size <= 0) + grbuf_size = sysconf(_SC_PAGESIZE); + if (grbuf_size <= 0) + grbuf_size = 4096; + + char *temp = NULL; + char *grbuf = NULL; + + do { + temp = realloc(grbuf, grbuf_size); + if (temp == NULL) { + ERROR("exec plugin: getegr_id for %s: realloc buffer[%ld] failed ", + pl->group, grbuf_size); + sfree(grbuf); + return -2; + } + grbuf = temp; + if (getgrnam_r(pl->group, &gr, grbuf, grbuf_size, &gr_ptr) == 0) { + sfree(grbuf); + if (gr_ptr == NULL) { + ERROR("exec plugin: No such group: `%s'", pl->group); + return -1; + } + return gr.gr_gid; + } else if (errno == ERANGE) { + grbuf_size += grbuf_size; // increment buffer size and try again + } else { + ERROR("exec plugin: getegr_id failed %s", STRERRNO); + sfree(grbuf); + return -2; + } + } while (grbuf_size <= MAX_GRBUF_SIZE); + ERROR("exec plugin: getegr_id Max grbuf size reached for %s", pl->group); + sfree(grbuf); + return -2; +} + +/* * Creates three pipes (one for reading, one for writing and one for errors), * forks a child, sets up the pipes so that fd_in is connected to STDIN of * the child and fd_out is connected to STDOUT and fd_err is connected to STDERR @@ -403,80 +461,10 @@ static int fork_child(program_list_t *pl, int *fd_in, int *fd_out, /* The group configured in the configfile is set as effective group, because * this way the forked process can (re-)gain the user's primary group. */ - egid = -1; - if (pl->group != NULL) { - if (*pl->group != '\0') { - struct group *gr_ptr = NULL; - struct group gr; - - long int grbuf_size = sysconf(_SC_GETGR_R_SIZE_MAX); - if (grbuf_size <= 0) - grbuf_size = sysconf(_SC_PAGESIZE); - if (grbuf_size <= 0) - grbuf_size = 4096; - - long int size; - size = grbuf_size; - char *temp = NULL; - char *grbuf = NULL; - int getgr_failed = 0; - grbuf = malloc(size); - if (grbuf == NULL) { - ERROR("exec plugin: get group information for '%s' failed: buffer " - "malloc() failed", - pl->group); - getgr_failed = 1; - goto gr_finally; - } - int status; - while ((status = getgrnam_r(pl->group, &gr, grbuf, size, &gr_ptr)) != 0) { - switch (errno) { - case ERANGE: - if ((size + grbuf_size) < size || - (size + grbuf_size) > MAX_GRBUF_SIZE) { - ERROR("exec plugin: get group information for '%s' max buffer " - "limit (%ld) reached \n", - pl->group, MAX_GRBUF_SIZE); - getgr_failed = 1; - goto gr_finally; - } - /* grow the buffer by 'grbuf_size' each time getgrnamr fails */ - size += grbuf_size; - temp = realloc(grbuf, size); - if (temp == NULL) { - ERROR("exec plugin: get group information for '%s' realloc() " - "buffer to (%ld) failed ", - pl->group, size); - getgr_failed = 1; - goto gr_finally; - } - grbuf = temp; - break; - default: - ERROR("exec plugin: default errno: get group information for '%s' " - "failed : %s", - pl->group, sstrerror(status, errbuf, sizeof(errbuf))); - getgr_failed = 1; - goto gr_finally; - } - } - if (gr_ptr == NULL) { - ERROR("exec plugin: No such group: `%s'", pl->group); - getgr_failed = 1; - goto gr_finally; - } - egid = gr.gr_gid; - gr_finally: - free(grbuf); - DEBUG("exec plugin: release grbuf memory "); - grbuf = NULL; - if (getgr_failed > 0) { - goto failed; - } - } else { - egid = gid; - } - } /* if (pl->group == NULL) */ + egid = getegr_id(pl, gid); + if (egid == -2) { + goto failed; + } pid = fork(); if (pid < 0) {