File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / spawn-fcgi / src / spawn-fcgi.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Mon Oct 14 15:24:39 2013 UTC (11 years, 5 months ago) by misho
Branches: spawn-fcgi, MAIN
CVS tags: v1_6_3, HEAD
1.6.3

    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>