Annotation of embedaddon/spawn-fcgi/src/spawn-fcgi.c, revision 1.1.1.1

1.1       misho       1: #ifdef HAVE_CONFIG_H
                      2: # include "config.h"
                      3: #endif
                      4: 
                      5: #include <sys/types.h>
                      6: #include <sys/time.h>
                      7: #include <sys/stat.h>
                      8: 
                      9: #include <stdlib.h>
                     10: #include <string.h>
                     11: #include <errno.h>
                     12: #include <stdio.h>
                     13: #include <unistd.h>
                     14: #include <fcntl.h>
                     15: 
                     16: #ifdef HAVE_PWD_H
                     17: # include <grp.h>
                     18: # include <pwd.h>
                     19: #endif
                     20: 
                     21: #ifdef HAVE_GETOPT_H
                     22: # include <getopt.h>
                     23: #endif
                     24: 
                     25: #define FCGI_LISTENSOCK_FILENO 0
                     26: 
                     27: /* "sys-socket.h" */
                     28: #ifdef __WIN32
                     29: 
                     30: # include <winsock2.h>
                     31: 
                     32: # define ECONNRESET WSAECONNRESET
                     33: # define EINPROGRESS WSAEINPROGRESS
                     34: # define EALREADY WSAEALREADY
                     35: # define ECONNABORTED WSAECONNABORTED
                     36: # define ioctl ioctlsocket
                     37: # define hstrerror(x) ""
                     38: 
                     39: #else /* _WIN32 */
                     40: 
                     41: # include <sys/socket.h>
                     42: # include <sys/ioctl.h>
                     43: # include <netinet/in.h>
                     44: # include <netinet/tcp.h>
                     45: # include <sys/un.h>
                     46: # include <arpa/inet.h>
                     47: 
                     48: # include <netdb.h>
                     49: 
                     50: #endif /* _WIN32 */
                     51: /* end "sys-socket.h" */
                     52: 
                     53: #ifdef HAVE_SYS_WAIT_H
                     54: # include <sys/wait.h>
                     55: #endif
                     56: 
                     57: /* for solaris 2.5 and netbsd 1.3.x */
                     58: #ifndef HAVE_SOCKLEN_T
                     59: typedef int socklen_t;
                     60: #endif
                     61: 
                     62: #ifndef HAVE_ISSETUGID
                     63: static int issetugid() {
                     64:        return (geteuid() != getuid() || getegid() != getgid());
                     65: }
                     66: #endif
                     67: 
                     68: #if defined(HAVE_IPV6) && defined(HAVE_INET_PTON)
                     69: # define USE_IPV6
                     70: #endif
                     71: 
                     72: #ifdef USE_IPV6
                     73: #define PACKAGE_FEATURES " (ipv6)"
                     74: #else
                     75: #define PACKAGE_FEATURES ""
                     76: #endif
                     77: 
                     78: #define PACKAGE_DESC "spawn-fcgi v" PACKAGE_VERSION PACKAGE_FEATURES " - spawns FastCGI processes\n"
                     79: 
                     80: #define CONST_STR_LEN(s) s, sizeof(s) - 1
                     81: 
                     82: static int bind_socket(const char *addr, unsigned short port, const char *unixsocket, uid_t uid, gid_t gid, int mode) {
                     83:        int fcgi_fd, socket_type, val;
                     84: 
                     85:        struct sockaddr_un fcgi_addr_un;
                     86:        struct sockaddr_in fcgi_addr_in;
                     87: #ifdef USE_IPV6
                     88:        struct sockaddr_in6 fcgi_addr_in6;
                     89: #endif
                     90:        struct sockaddr *fcgi_addr;
                     91: 
                     92:        socklen_t servlen;
                     93: 
                     94:        if (unixsocket) {
                     95:                memset(&fcgi_addr_un, 0, sizeof(fcgi_addr_un));
                     96: 
                     97:                fcgi_addr_un.sun_family = AF_UNIX;
                     98:                strcpy(fcgi_addr_un.sun_path, unixsocket);
                     99: 
                    100: #ifdef SUN_LEN
                    101:                servlen = SUN_LEN(&fcgi_addr_un);
                    102: #else
                    103:                /* stevens says: */
                    104:                servlen = strlen(fcgi_addr_un.sun_path) + sizeof(fcgi_addr_un.sun_family);
                    105: #endif
                    106:                socket_type = AF_UNIX;
                    107:                fcgi_addr = (struct sockaddr *) &fcgi_addr_un;
                    108: 
                    109:                /* check if some backend is listening on the socket
                    110:                 * as if we delete the socket-file and rebind there will be no "socket already in use" error
                    111:                 */
                    112:                if (-1 == (fcgi_fd = socket(socket_type, SOCK_STREAM, 0))) {
                    113:                        fprintf(stderr, "spawn-fcgi: couldn't create socket: %s\n", strerror(errno));
                    114:                        return -1;
                    115:                }
                    116: 
                    117:                if (0 == connect(fcgi_fd, fcgi_addr, servlen)) {
                    118:                        fprintf(stderr, "spawn-fcgi: socket is already in use, can't spawn\n");
                    119:                        close(fcgi_fd);
                    120:                        return -1;
                    121:                }
                    122: 
                    123:                /* cleanup previous socket if it exists */
                    124:                if (-1 == unlink(unixsocket)) {
                    125:                        switch (errno) {
                    126:                        case ENOENT:
                    127:                                break;
                    128:                        default:
                    129:                                fprintf(stderr, "spawn-fcgi: removing old socket failed: %s\n", strerror(errno));
                    130:                                return -1;
                    131:                        }
                    132:                }
                    133: 
                    134:                close(fcgi_fd);
                    135:        } else {
                    136:                memset(&fcgi_addr_in, 0, sizeof(fcgi_addr_in));
                    137:                fcgi_addr_in.sin_family = AF_INET;
                    138:                fcgi_addr_in.sin_port = htons(port);
                    139: 
                    140:                servlen = sizeof(fcgi_addr_in);
                    141:                socket_type = AF_INET;
                    142:                fcgi_addr = (struct sockaddr *) &fcgi_addr_in;
                    143: 
                    144: #ifdef USE_IPV6
                    145:                memset(&fcgi_addr_in6, 0, sizeof(fcgi_addr_in6));
                    146:                fcgi_addr_in6.sin6_family = AF_INET6;
                    147:                fcgi_addr_in6.sin6_port = fcgi_addr_in.sin_port;
                    148: #endif
                    149: 
                    150:                if (addr == NULL) {
                    151:                        fcgi_addr_in.sin_addr.s_addr = htonl(INADDR_ANY);
                    152: #ifdef HAVE_INET_PTON
                    153:                } else if (1 == inet_pton(AF_INET, addr, &fcgi_addr_in.sin_addr)) {
                    154:                        /* nothing to do */
                    155: #ifdef HAVE_IPV6
                    156:                } else if (1 == inet_pton(AF_INET6, addr, &fcgi_addr_in6.sin6_addr)) {
                    157:                        servlen = sizeof(fcgi_addr_in6);
                    158:                        socket_type = AF_INET6;
                    159:                        fcgi_addr = (struct sockaddr *) &fcgi_addr_in6;
                    160: #endif
                    161:                } else {
                    162:                        fprintf(stderr, "spawn-fcgi: '%s' is not a valid IP address\n", addr);
                    163:                        return -1;
                    164: #else
                    165:                } else {
                    166:                        if ((in_addr_t)(-1) == (fcgi_addr_in.sin_addr.s_addr = inet_addr(addr))) {
                    167:                                fprintf(stderr, "spawn-fcgi: '%s' is not a valid IPv4 address\n", addr);
                    168:                                return -1;
                    169:                        }
                    170: #endif
                    171:                }
                    172:        }
                    173: 
                    174: 
                    175:        if (-1 == (fcgi_fd = socket(socket_type, SOCK_STREAM, 0))) {
                    176:                fprintf(stderr, "spawn-fcgi: couldn't create socket: %s\n", strerror(errno));
                    177:                return -1;
                    178:        }
                    179: 
                    180:        val = 1;
                    181:        if (setsockopt(fcgi_fd, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val)) < 0) {
                    182:                fprintf(stderr, "spawn-fcgi: couldn't set SO_REUSEADDR: %s\n", strerror(errno));
                    183:                return -1;
                    184:        }
                    185: 
                    186:        if (-1 == bind(fcgi_fd, fcgi_addr, servlen)) {
                    187:                fprintf(stderr, "spawn-fcgi: bind failed: %s\n", strerror(errno));
                    188:                return -1;
                    189:        }
                    190: 
                    191:        if (unixsocket) {
                    192:                if (0 != uid || 0 != gid) {
                    193:                        if (0 == uid) uid = -1;
                    194:                        if (0 == gid) gid = -1;
                    195:                        if (-1 == chown(unixsocket, uid, gid)) {
                    196:                                fprintf(stderr, "spawn-fcgi: couldn't chown socket: %s\n", strerror(errno));
                    197:                                close(fcgi_fd);
                    198:                                unlink(unixsocket);
                    199:                                return -1;
                    200:                        }
                    201:                }
                    202: 
                    203:                if (-1 != mode && -1 == chmod(unixsocket, mode)) {
                    204:                        fprintf(stderr, "spawn-fcgi: couldn't chmod socket: %s\n", strerror(errno));
                    205:                        close(fcgi_fd);
                    206:                        unlink(unixsocket);
                    207:                        return -1;
                    208:                }
                    209:        }
                    210: 
                    211:        if (-1 == listen(fcgi_fd, 1024)) {
                    212:                fprintf(stderr, "spawn-fcgi: listen failed: %s\n", strerror(errno));
                    213:                return -1;
                    214:        }
                    215: 
                    216:        return fcgi_fd;
                    217: }
                    218: 
                    219: static int fcgi_spawn_connection(char *appPath, char **appArgv, int fcgi_fd, int fork_count, int child_count, int pid_fd, int nofork) {
                    220:        int status, rc = 0;
                    221:        struct timeval tv = { 0, 100 * 1000 };
                    222: 
                    223:        pid_t child;
                    224: 
                    225:        while (fork_count-- > 0) {
                    226: 
                    227:                if (!nofork) {
                    228:                        child = fork();
                    229:                } else {
                    230:                        child = 0;
                    231:                }
                    232: 
                    233:                switch (child) {
                    234:                case 0: {
                    235:                        char cgi_childs[64];
                    236:                        int max_fd = 0;
                    237: 
                    238:                        int i = 0;
                    239: 
                    240:                        if (child_count >= 0) {
                    241:                                snprintf(cgi_childs, sizeof(cgi_childs), "PHP_FCGI_CHILDREN=%d", child_count);
                    242:                                putenv(cgi_childs);
                    243:                        }
                    244: 
                    245:                        if(fcgi_fd != FCGI_LISTENSOCK_FILENO) {
                    246:                                close(FCGI_LISTENSOCK_FILENO);
                    247:                                dup2(fcgi_fd, FCGI_LISTENSOCK_FILENO);
                    248:                                close(fcgi_fd);
                    249:                        }
                    250: 
                    251:                        /* loose control terminal */
                    252:                        if (!nofork) {
                    253:                                setsid();
                    254: 
                    255:                                max_fd = open("/dev/null", O_RDWR);
                    256:                                if (-1 != max_fd) {
                    257:                                        if (max_fd != STDOUT_FILENO) dup2(max_fd, STDOUT_FILENO);
                    258:                                        if (max_fd != STDERR_FILENO) dup2(max_fd, STDERR_FILENO);
                    259:                                        if (max_fd != STDOUT_FILENO && max_fd != STDERR_FILENO) close(max_fd);
                    260:                                } else {
                    261:                                        fprintf(stderr, "spawn-fcgi: couldn't open and redirect stdout/stderr to '/dev/null': %s\n", strerror(errno));
                    262:                                }
                    263:                        }
                    264: 
                    265:                        /* we don't need the client socket */
                    266:                        for (i = 3; i < max_fd; i++) {
                    267:                                if (i != FCGI_LISTENSOCK_FILENO) close(i);
                    268:                        }
                    269: 
                    270:                        /* fork and replace shell */
                    271:                        if (appArgv) {
                    272:                                execv(appArgv[0], appArgv);
                    273: 
                    274:                        } else {
                    275:                                char *b = malloc((sizeof("exec ") - 1) + strlen(appPath) + 1);
                    276:                                strcpy(b, "exec ");
                    277:                                strcat(b, appPath);
                    278: 
                    279:                                /* exec the cgi */
                    280:                                execl("/bin/sh", "sh", "-c", b, (char *)NULL);
                    281:                        }
                    282: 
                    283:                        /* in nofork mode stderr is still open */
                    284:                        fprintf(stderr, "spawn-fcgi: exec failed: %s\n", strerror(errno));
                    285:                        exit(errno);
                    286: 
                    287:                        break;
                    288:                }
                    289:                case -1:
                    290:                        /* error */
                    291:                        fprintf(stderr, "spawn-fcgi: fork failed: %s\n", strerror(errno));
                    292:                        break;
                    293:                default:
                    294:                        /* father */
                    295: 
                    296:                        /* wait */
                    297:                        select(0, NULL, NULL, NULL, &tv);
                    298: 
                    299:                        switch (waitpid(child, &status, WNOHANG)) {
                    300:                        case 0:
                    301:                                fprintf(stdout, "spawn-fcgi: child spawned successfully: PID: %d\n", child);
                    302: 
                    303:                                /* write pid file */
                    304:                                if (pid_fd != -1) {
                    305:                                        /* assume a 32bit pid_t */
                    306:                                        char pidbuf[12];
                    307: 
                    308:                                        snprintf(pidbuf, sizeof(pidbuf) - 1, "%d", child);
                    309: 
                    310:                                        write(pid_fd, pidbuf, strlen(pidbuf));
                    311:                                        /* avoid eol for the last one */
                    312:                                        if (fork_count != 0) {
                    313:                                                write(pid_fd, "\n", 1);
                    314:                                        }
                    315:                                }
                    316: 
                    317:                                break;
                    318:                        case -1:
                    319:                                break;
                    320:                        default:
                    321:                                if (WIFEXITED(status)) {
                    322:                                        fprintf(stderr, "spawn-fcgi: child exited with: %d\n",
                    323:                                                WEXITSTATUS(status));
                    324:                                        rc = WEXITSTATUS(status);
                    325:                                } else if (WIFSIGNALED(status)) {
                    326:                                        fprintf(stderr, "spawn-fcgi: child signaled: %d\n",
                    327:                                                WTERMSIG(status));
                    328:                                        rc = 1;
                    329:                                } else {
                    330:                                        fprintf(stderr, "spawn-fcgi: child died somehow: exit status = %d\n",
                    331:                                                status);
                    332:                                        rc = status;
                    333:                                }
                    334:                        }
                    335: 
                    336:                        break;
                    337:                }
                    338:        }
                    339:        close(pid_fd);
                    340: 
                    341:        close(fcgi_fd);
                    342: 
                    343:        return rc;
                    344: }
                    345: 
                    346: static int find_user_group(const char *user, const char *group, uid_t *uid, gid_t *gid, const char **username) {
                    347:        uid_t my_uid = 0;
                    348:        gid_t my_gid = 0;
                    349:        struct passwd *my_pwd = NULL;
                    350:        struct group *my_grp = NULL;
                    351:        char *endptr = NULL;
                    352:        *uid = 0; *gid = 0;
                    353:        if (username) *username = NULL;
                    354: 
                    355:        if (user) {
                    356:                my_uid = strtol(user, &endptr, 10);
                    357: 
                    358:                if (my_uid <= 0 || *endptr) {
                    359:                        if (NULL == (my_pwd = getpwnam(user))) {
                    360:                                fprintf(stderr, "spawn-fcgi: can't find user name %s\n", user);
                    361:                                return -1;
                    362:                        }
                    363:                        my_uid = my_pwd->pw_uid;
                    364: 
                    365:                        if (my_uid == 0) {
                    366:                                fprintf(stderr, "spawn-fcgi: I will not set uid to 0\n");
                    367:                                return -1;
                    368:                        }
                    369: 
                    370:                        if (username) *username = user;
                    371:                } else {
                    372:                        my_pwd = getpwuid(my_uid);
                    373:                        if (username && my_pwd) *username = my_pwd->pw_name;
                    374:                }
                    375:        }
                    376: 
                    377:        if (group) {
                    378:                my_gid = strtol(group, &endptr, 10);
                    379: 
                    380:                if (my_gid <= 0 || *endptr) {
                    381:                        if (NULL == (my_grp = getgrnam(group))) {
                    382:                                fprintf(stderr, "spawn-fcgi: can't find group name %s\n", group);
                    383:                                return -1;
                    384:                        }
                    385:                        my_gid = my_grp->gr_gid;
                    386: 
                    387:                        if (my_gid == 0) {
                    388:                                fprintf(stderr, "spawn-fcgi: I will not set gid to 0\n");
                    389:                                return -1;
                    390:                        }
                    391:                }
                    392:        } else if (my_pwd) {
                    393:                my_gid = my_pwd->pw_gid;
                    394: 
                    395:                if (my_gid == 0) {
                    396:                        fprintf(stderr, "spawn-fcgi: I will not set gid to 0\n");
                    397:                        return -1;
                    398:                }
                    399:        }
                    400: 
                    401:        *uid = my_uid;
                    402:        *gid = my_gid;
                    403:        return 0;
                    404: }
                    405: 
                    406: static void show_version () {
                    407:        write(1, CONST_STR_LEN(
                    408:                PACKAGE_DESC \
                    409:                "Build-Date: " __DATE__ " " __TIME__ "\n"
                    410:        ));
                    411: }
                    412: 
                    413: static void show_help () {
                    414:        write(1, CONST_STR_LEN(
                    415:                "Usage: spawn-fcgi [options] [-- <fcgiapp> [fcgi app arguments]]\n" \
                    416:                "\n" \
                    417:                PACKAGE_DESC \
                    418:                "\n" \
                    419:                "Options:\n" \
                    420:                " -f <path>      filename of the fcgi-application (deprecated; ignored if\n" \
                    421:                "                <fcgiapp> is given; needs /bin/sh)\n" \
                    422:                " -d <directory> chdir to directory before spawning\n" \
                    423:                " -a <address>   bind to IPv4/IPv6 address (defaults to 0.0.0.0)\n" \
                    424:                " -p <port>      bind to TCP-port\n" \
                    425:                " -s <path>      bind to Unix domain socket\n" \
                    426:                " -M <mode>      change Unix domain socket mode\n" \
                    427:                " -C <children>  (PHP only) numbers of childs to spawn (default: not setting\n" \
                    428:                "                the PHP_FCGI_CHILDREN environment variable - PHP defaults to 0)\n" \
                    429:                " -F <children>  number of children to fork (default 1)\n" \
                    430:                " -P <path>      name of PID-file for spawned process (ignored in no-fork mode)\n" \
                    431:                " -n             no fork (for daemontools)\n" \
                    432:                " -v             show version\n" \
                    433:                " -?, -h         show this help\n" \
                    434:                "(root only)\n" \
                    435:                " -c <directory> chroot to directory\n" \
                    436:                " -S             create socket before chroot() (default is to create the socket\n" \
                    437:                "                in the chroot)\n" \
                    438:                " -u <user>      change to user-id\n" \
                    439:                " -g <group>     change to group-id (default: primary group of user if -u\n" \
                    440:                "                is given)\n" \
                    441:                " -U <user>      change Unix domain socket owner to user-id\n" \
                    442:                " -G <group>     change Unix domain socket group to group-id\n" \
                    443:        ));
                    444: }
                    445: 
                    446: 
                    447: int main(int argc, char **argv) {
                    448:        char *fcgi_app = NULL, *changeroot = NULL, *username = NULL,
                    449:             *groupname = NULL, *unixsocket = NULL, *pid_file = NULL,
                    450:             *sockusername = NULL, *sockgroupname = NULL, *fcgi_dir = NULL,
                    451:             *addr = NULL;
                    452:        char **fcgi_app_argv = { NULL };
                    453:        char *endptr = NULL;
                    454:        unsigned short port = 0;
                    455:        int sockmode = -1;
                    456:        int child_count = -1;
                    457:        int fork_count = 1;
                    458:        int i_am_root, o;
                    459:        int pid_fd = -1;
                    460:        int nofork = 0;
                    461:        int sockbeforechroot = 0;
                    462:        struct sockaddr_un un;
                    463:        int fcgi_fd = -1;
                    464: 
                    465:        if (argc < 2) { /* no arguments given */
                    466:                show_help();
                    467:                return -1;
                    468:        }
                    469: 
                    470:        i_am_root = (getuid() == 0);
                    471: 
                    472:        while (-1 != (o = getopt(argc, argv, "c:d:f:g:?hna:p:u:vC:F:s:P:U:G:M:S"))) {
                    473:                switch(o) {
                    474:                case 'f': fcgi_app = optarg; break;
                    475:                case 'd': fcgi_dir = optarg; break;
                    476:                case 'a': addr = optarg;/* ip addr */ break;
                    477:                case 'p': port = strtol(optarg, &endptr, 10);/* port */
                    478:                        if (*endptr) {
                    479:                                fprintf(stderr, "spawn-fcgi: invalid port: %u\n", (unsigned int) port);
                    480:                                return -1;
                    481:                        }
                    482:                        break;
                    483:                case 'C': child_count = strtol(optarg, NULL, 10);/*  */ break;
                    484:                case 'F': fork_count = strtol(optarg, NULL, 10);/*  */ break;
                    485:                case 's': unixsocket = optarg; /* unix-domain socket */ break;
                    486:                case 'c': if (i_am_root) { changeroot = optarg; }/* chroot() */ break;
                    487:                case 'u': if (i_am_root) { username = optarg; } /* set user */ break;
                    488:                case 'g': if (i_am_root) { groupname = optarg; } /* set group */ break;
                    489:                case 'U': if (i_am_root) { sockusername = optarg; } /* set socket user */ break;
                    490:                case 'G': if (i_am_root) { sockgroupname = optarg; } /* set socket group */ break;
                    491:                case 'S': if (i_am_root) { sockbeforechroot = 1; } /* open socket before chroot() */ break;
                    492:                case 'M': sockmode = strtol(optarg, NULL, 0); /* set socket mode */ break;
                    493:                case 'n': nofork = 1; break;
                    494:                case 'P': pid_file = optarg; /* PID file */ break;
                    495:                case 'v': show_version(); return 0;
                    496:                case '?':
                    497:                case 'h': show_help(); return 0;
                    498:                default:
                    499:                        show_help();
                    500:                        return -1;
                    501:                }
                    502:        }
                    503: 
                    504:        if (optind < argc) {
                    505:                fcgi_app_argv = &argv[optind];
                    506:        }
                    507: 
                    508:        if (NULL == fcgi_app && NULL == fcgi_app_argv) {
                    509:                fprintf(stderr, "spawn-fcgi: no FastCGI application given\n");
                    510:                return -1;
                    511:        }
                    512: 
                    513:        if (0 == port && NULL == unixsocket) {
                    514:                fprintf(stderr, "spawn-fcgi: no socket given (use either -p or -s)\n");
                    515:                return -1;
                    516:        } else if (0 != port && NULL != unixsocket) {
                    517:                fprintf(stderr, "spawn-fcgi: either a Unix domain socket or a TCP-port, but not both\n");
                    518:                return -1;
                    519:        }
                    520: 
                    521:        if (unixsocket && strlen(unixsocket) > sizeof(un.sun_path) - 1) {
                    522:                fprintf(stderr, "spawn-fcgi: path of the Unix domain socket is too long\n");
                    523:                return -1;
                    524:        }
                    525: 
                    526:        /* SUID handling */
                    527:        if (!i_am_root && issetugid()) {
                    528:                fprintf(stderr, "spawn-fcgi: Are you nuts? Don't apply a SUID bit to this binary\n");
                    529:                return -1;
                    530:        }
                    531: 
                    532:        if (nofork) pid_file = NULL; /* ignore pid file in no-fork mode */
                    533: 
                    534:        if (pid_file &&
                    535:            (-1 == (pid_fd = open(pid_file, O_WRONLY | O_CREAT | O_EXCL | O_TRUNC, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)))) {
                    536:                struct stat st;
                    537:                if (errno != EEXIST) {
                    538:                        fprintf(stderr, "spawn-fcgi: opening PID-file '%s' failed: %s\n",
                    539:                                pid_file, strerror(errno));
                    540:                        return -1;
                    541:                }
                    542: 
                    543:                /* ok, file exists */
                    544: 
                    545:                if (0 != stat(pid_file, &st)) {
                    546:                        fprintf(stderr, "spawn-fcgi: stating PID-file '%s' failed: %s\n",
                    547:                                pid_file, strerror(errno));
                    548:                        return -1;
                    549:                }
                    550: 
                    551:                /* is it a regular file ? */
                    552: 
                    553:                if (!S_ISREG(st.st_mode)) {
                    554:                        fprintf(stderr, "spawn-fcgi: PID-file exists and isn't regular file: '%s'\n",
                    555:                                pid_file);
                    556:                        return -1;
                    557:                }
                    558: 
                    559:                if (-1 == (pid_fd = open(pid_file, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH))) {
                    560:                        fprintf(stderr, "spawn-fcgi: opening PID-file '%s' failed: %s\n",
                    561:                                pid_file, strerror(errno));
                    562:                        return -1;
                    563:                }
                    564:        }
                    565: 
                    566:        if (i_am_root) {
                    567:                uid_t uid, sockuid;
                    568:                gid_t gid, sockgid;
                    569:                const char* real_username;
                    570: 
                    571:                if (-1 == find_user_group(username, groupname, &uid, &gid, &real_username))
                    572:                        return -1;
                    573: 
                    574:                if (-1 == find_user_group(sockusername, sockgroupname, &sockuid, &sockgid, NULL))
                    575:                        return -1;
                    576: 
                    577:                if (uid != 0 && gid == 0) {
                    578:                        fprintf(stderr, "spawn-fcgi: WARNING: couldn't find the user for uid %i and no group was specified, so only the user privileges will be dropped\n", (int) uid);
                    579:                }
                    580: 
                    581:                if (0 == sockuid) sockuid = uid;
                    582:                if (0 == sockgid) sockgid = gid;
                    583: 
                    584:                if (sockbeforechroot && -1 == (fcgi_fd = bind_socket(addr, port, unixsocket, sockuid, sockgid, sockmode)))
                    585:                        return -1;
                    586: 
                    587:                /* Change group before chroot, when we have access
                    588:                 * to /etc/group
                    589:                 */
                    590:                if (gid != 0) {
                    591:                        setgid(gid);
                    592:                        setgroups(0, NULL);
                    593:                        if (real_username) {
                    594:                                initgroups(real_username, gid);
                    595:                        }
                    596:                }
                    597: 
                    598:                if (changeroot) {
                    599:                        if (-1 == chroot(changeroot)) {
                    600:                                fprintf(stderr, "spawn-fcgi: chroot('%s') failed: %s\n", changeroot, strerror(errno));
                    601:                                return -1;
                    602:                        }
                    603:                        if (-1 == chdir("/")) {
                    604:                                fprintf(stderr, "spawn-fcgi: chdir('/') failed: %s\n", strerror(errno));
                    605:                                return -1;
                    606:                        }
                    607:                }
                    608: 
                    609:                if (!sockbeforechroot && -1 == (fcgi_fd = bind_socket(addr, port, unixsocket, sockuid, sockgid, sockmode)))
                    610:                        return -1;
                    611: 
                    612:                /* drop root privs */
                    613:                if (uid != 0) {
                    614:                        setuid(uid);
                    615:                }
                    616:        } else {
                    617:                if (-1 == (fcgi_fd = bind_socket(addr, port, unixsocket, 0, 0, sockmode)))
                    618:                        return -1;
                    619:        }
                    620: 
                    621:        if (fcgi_dir && -1 == chdir(fcgi_dir)) {
                    622:                fprintf(stderr, "spawn-fcgi: chdir('%s') failed: %s\n", fcgi_dir, strerror(errno));
                    623:                return -1;
                    624:        }
                    625: 
                    626:        return fcgi_spawn_connection(fcgi_app, fcgi_app_argv, fcgi_fd, fork_count, child_count, pid_fd, nofork);
                    627: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>