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>