/* Copyright information is at end of file */ #include #include #include #include #include #include #ifndef WIN32 #include #endif #include "xmlrpc_config.h" #include "bool.h" #include "girmath.h" #include "mallocvar.h" #include "xmlrpc-c/string_int.h" #include "xmlrpc-c/sleep_int.h" #include "xmlrpc-c/abyss.h" #include "trace.h" #include "session.h" #include "file.h" #include "conn.h" #include "chanswitch.h" #include "channel.h" #include "socket.h" #ifdef WIN32 #include "socket_win.h" #else #include "socket_unix.h" #endif #include "http.h" #include "handler.h" #include "server.h" void ServerTerminate(TServer * const serverP) { struct _TServer * const srvP = serverP->srvP; srvP->terminationRequested = true; if (srvP->chanSwitchP) { ChanSwitchInterrupt(srvP->chanSwitchP); ChanSwitchDestroy(srvP->chanSwitchP); } } void ServerResetTerminate(TServer * const serverP) { struct _TServer * const srvP = serverP->srvP; srvP->terminationRequested = false; } static void initUnixStuff(struct _TServer * const srvP) { #ifndef WIN32 srvP->pidfileP = NULL; srvP->uid = srvP->gid = -1; #endif } static bool logOpen(struct _TServer * const srvP) { bool success; success = FileOpenCreate(&srvP->logfileP, srvP->logfilename, O_WRONLY | O_APPEND); if (success) { bool success; success = MutexCreate(&srvP->logmutexP); if (success) srvP->logfileisopen = TRUE; else TraceMsg("Can't create mutex for log file"); if (!success) FileClose(srvP->logfileP); } else TraceMsg("Can't open log file '%s'", srvP->logfilename); return success; } static void logClose(struct _TServer * const srvP) { if (srvP->logfileisopen) { FileClose(srvP->logfileP); MutexDestroy(srvP->logmutexP); srvP->logfileisopen = FALSE; } } static void initChanSwitchStuff(struct _TServer * const srvP, bool const noAccept, TChanSwitch * const userSwitchP, unsigned short const port, const char ** const errorP) { if (userSwitchP) { *errorP = NULL; srvP->serverAcceptsConnections = TRUE; srvP->chanSwitchP = userSwitchP; } else if (noAccept) { *errorP = NULL; srvP->serverAcceptsConnections = FALSE; srvP->chanSwitchP = NULL; } else { *errorP = NULL; srvP->serverAcceptsConnections = TRUE; srvP->chanSwitchP = NULL; srvP->port = port; } srvP->weCreatedChanSwitch = FALSE; } static void createServer(struct _TServer ** const srvPP, bool const noAccept, TChanSwitch * const userChanSwitchP, unsigned short const portNumber, const char ** const errorP) { struct _TServer * srvP; MALLOCVAR(srvP); if (srvP == NULL) { xmlrpc_asprintf(errorP, "Unable to allocate space for server descriptor"); } else { srvP->terminationRequested = false; initChanSwitchStuff(srvP, noAccept, userChanSwitchP, portNumber, errorP); if (!*errorP) { srvP->builtinHandlerP = HandlerCreate(); if (!srvP->builtinHandlerP) xmlrpc_asprintf(errorP, "Unable to allocate space for " "builtin handler descriptor"); else { srvP->defaultHandler = HandlerDefaultBuiltin; srvP->defaultHandlerContext = srvP->builtinHandlerP; srvP->name = strdup("unnamed"); srvP->logfilename = NULL; srvP->keepalivetimeout = 15; srvP->keepalivemaxconn = 30; srvP->timeout = 15; srvP->advertise = TRUE; srvP->useSigchld = FALSE; initUnixStuff(srvP); ListInitAutoFree(&srvP->handlers); srvP->logfileisopen = FALSE; *errorP = NULL; if (*errorP) HandlerDestroy(srvP->builtinHandlerP); } } if (*errorP) free(srvP); } *srvPP = srvP; } static void setNamePathLog(TServer * const serverP, const char * const name, const char * const filesPath, const char * const logFileName) { /*---------------------------------------------------------------------------- This odd function exists to help with backward compatibility. Today, we have the expandable model where you create a server with default parameters, then use ServerSet... functions to choose non-default parameters. But before, you specified these three parameters right in the arguments of various create functions. -----------------------------------------------------------------------------*/ if (name) ServerSetName(serverP, name); if (filesPath) ServerSetFilesPath(serverP, filesPath); if (logFileName) ServerSetLogFileName(serverP, logFileName); } abyss_bool ServerCreate(TServer * const serverP, const char * const name, xmlrpc_uint16_t const portNumber, const char * const filesPath, const char * const logFileName) { bool const noAcceptFalse = FALSE; bool success; const char * error; createServer(&serverP->srvP, noAcceptFalse, NULL, portNumber, &error); if (error) { TraceMsg(error); xmlrpc_strfree(error); success = FALSE; } else { success = TRUE; setNamePathLog(serverP, name, filesPath, logFileName); } return success; } static void createSwitchFromOsSocket(TOsSocket const osSocket, TChanSwitch ** const chanSwitchPP, const char ** const errorP) { #ifdef WIN32 ChanSwitchWinCreateWinsock(osSocket, chanSwitchPP, errorP); #else ChanSwitchUnixCreateFd(osSocket, chanSwitchPP, errorP); #endif } static void createChannelFromOsSocket(TOsSocket const osSocket, TChannel ** const channelPP, void ** const channelInfoPP, const char ** const errorP) { #ifdef WIN32 ChannelWinCreateWinsock(osSocket, channelPP, (struct abyss_win_chaninfo**)channelInfoPP, errorP); #else ChannelUnixCreateFd(osSocket, channelPP, (struct abyss_unix_chaninfo**)channelInfoPP, errorP); #endif } abyss_bool ServerCreateSocket(TServer * const serverP, const char * const name, TOsSocket const socketFd, const char * const filesPath, const char * const logFileName) { bool success; TChanSwitch * chanSwitchP; const char * error; createSwitchFromOsSocket(socketFd, &chanSwitchP, &error); if (error) { TraceMsg(error); success = FALSE; xmlrpc_strfree(error); } else { bool const noAcceptFalse = FALSE; const char * error; createServer(&serverP->srvP, noAcceptFalse, chanSwitchP, 0, &error); if (error) { TraceMsg(error); success = FALSE; xmlrpc_strfree(error); } else { success = TRUE; setNamePathLog(serverP, name, filesPath, logFileName); } } return success; } abyss_bool ServerCreateNoAccept(TServer * const serverP, const char * const name, const char * const filesPath, const char * const logFileName) { bool const noAcceptTrue = TRUE; bool success; const char * error; createServer(&serverP->srvP, noAcceptTrue, NULL, 0, &error); if (error) { TraceMsg(error); success = FALSE; xmlrpc_strfree(error); } else { success = TRUE; setNamePathLog(serverP, name, filesPath, logFileName); } return success; } void ServerCreateSwitch(TServer * const serverP, TChanSwitch * const chanSwitchP, const char ** const errorP) { bool const noAcceptFalse = FALSE; assert(serverP); assert(chanSwitchP); createServer(&serverP->srvP, noAcceptFalse, chanSwitchP, 0, errorP); } void ServerCreateSocket2(TServer * const serverP, TSocket * const socketP, const char ** const errorP) { TChanSwitch * const chanSwitchP = SocketGetChanSwitch(socketP); assert(socketP); if (!chanSwitchP) xmlrpc_asprintf( errorP, "Socket is not a listening socket. " "You should use ServerCreateSwitch() instead, anyway."); else ServerCreateSwitch(serverP, chanSwitchP, errorP); } static void terminateHandlers(TList * const handlersP) { /*---------------------------------------------------------------------------- Terminate all handlers in the list '*handlersP'. I.e. call each handler's terminate function. -----------------------------------------------------------------------------*/ if (handlersP->item) { unsigned int i; for (i = handlersP->size; i > 0; --i) { URIHandler2 * const handlerP = handlersP->item[i-1]; if (handlerP->term) handlerP->term(handlerP->userdata); } } } void ServerFree(TServer * const serverP) { struct _TServer * const srvP = serverP->srvP; if (srvP->weCreatedChanSwitch) ChanSwitchDestroy(srvP->chanSwitchP); xmlrpc_strfree(srvP->name); terminateHandlers(&srvP->handlers); ListFree(&srvP->handlers); HandlerDestroy(srvP->builtinHandlerP); logClose(srvP); if (srvP->logfilename) xmlrpc_strfree(srvP->logfilename); free(srvP); } void ServerSetName(TServer * const serverP, const char * const name) { xmlrpc_strfree(serverP->srvP->name); serverP->srvP->name = strdup(name); } void ServerSetFilesPath(TServer * const serverP, const char * const filesPath) { HandlerSetFilesPath(serverP->srvP->builtinHandlerP, filesPath); } void ServerSetLogFileName(TServer * const serverP, const char * const logFileName) { struct _TServer * const srvP = serverP->srvP; if (srvP->logfilename) xmlrpc_strfree(srvP->logfilename); srvP->logfilename = strdup(logFileName); } void ServerSetKeepaliveTimeout(TServer * const serverP, xmlrpc_uint32_t const keepaliveTimeout) { serverP->srvP->keepalivetimeout = keepaliveTimeout; } void ServerSetKeepaliveMaxConn(TServer * const serverP, xmlrpc_uint32_t const keepaliveMaxConn) { serverP->srvP->keepalivemaxconn = keepaliveMaxConn; } void ServerSetTimeout(TServer * const serverP, xmlrpc_uint32_t const timeout) { serverP->srvP->timeout = timeout; } void ServerSetAdvertise(TServer * const serverP, abyss_bool const advertise) { serverP->srvP->advertise = advertise; } void ServerSetMimeType(TServer * const serverP, MIMEType * const mimeTypeP) { HandlerSetMimeType(serverP->srvP->builtinHandlerP, mimeTypeP); } static void runUserHandler(TSession * const sessionP, struct _TServer * const srvP) { abyss_bool handled; int i; for (i = srvP->handlers.size-1, handled = FALSE; i >= 0 && !handled; --i) { URIHandler2 * const handlerP = srvP->handlers.item[i]; if (handlerP->handleReq2) handlerP->handleReq2(handlerP, sessionP, &handled); else if (handlerP->handleReq1) handled = handlerP->handleReq1(sessionP); } assert(srvP->defaultHandler); if (!handled) srvP->defaultHandler(sessionP); } static void processDataFromClient(TConn * const connectionP, bool const lastReqOnConn, uint32_t const timeout, bool * const keepAliveP) { TSession session = {0}; /* initilization, an afforadble alternative to random memory being misinterpreted! */ RequestInit(&session, connectionP); session.serverDeniesKeepalive = lastReqOnConn; RequestRead(&session, timeout); if (session.status == 0) { if (session.version.major >= 2) ResponseStatus(&session, 505); else if (!RequestValidURI(&session)) ResponseStatus(&session, 400); else runUserHandler(&session, connectionP->server->srvP); } assert(session.status != 0); if (session.responseStarted) HTTPWriteEndChunk(&session); else ResponseError(&session); *keepAliveP = HTTPKeepalive(&session); SessionLog(&session); RequestFree(&session); } static TThreadProc serverFunc; static void serverFunc(void * const userHandle) { /*---------------------------------------------------------------------------- Do server stuff on one connection. At its simplest, this means do one HTTP request. But with keepalive, it can be many requests. -----------------------------------------------------------------------------*/ TConn * const connectionP = userHandle; struct _TServer * const srvP = connectionP->server->srvP; unsigned int requestCount; /* Number of requests we've handled so far on this connection */ bool connectionDone; /* No more need for this HTTP connection */ requestCount = 0; connectionDone = FALSE; while (!connectionDone) { bool success; /* Wait to read until timeout */ success = ConnRead(connectionP, srvP->keepalivetimeout); if (!success) connectionDone = TRUE; else { bool const lastReqOnConn = requestCount + 1 >= srvP->keepalivemaxconn; bool keepalive; processDataFromClient(connectionP, lastReqOnConn, srvP->timeout, &keepalive); ++requestCount; if (!keepalive) connectionDone = TRUE; /**************** Must adjust the read buffer *****************/ ConnReadInit(connectionP); } } } static void createSwitchFromPortNum(unsigned short const portNumber, TChanSwitch ** const chanSwitchPP, const char ** const errorP) { #ifdef WIN32 ChanSwitchWinCreate(portNumber, chanSwitchPP, errorP); #else ChanSwitchUnixCreate(portNumber, chanSwitchPP, errorP); #endif } static void createChanSwitch(struct _TServer * const srvP, const char ** const errorP) { TChanSwitch * chanSwitchP; const char * error; /* Not valid to call this when channel switch already exists: */ assert(srvP->chanSwitchP == NULL); createSwitchFromPortNum(srvP->port, &chanSwitchP, &error); if (error) { xmlrpc_asprintf(errorP, "Can't create channel switch. %s", error); xmlrpc_strfree(error); } else { *errorP = NULL; srvP->weCreatedChanSwitch = TRUE; srvP->chanSwitchP = chanSwitchP; } } int ServerInit(TServer * const serverP) { /*---------------------------------------------------------------------------- Initialize a server to accept connections. Do not confuse this with creating the server -- ServerCreate(). Not necessary or valid with a server that doesn't accept connections (i.e. user supplies the channels (TCP connections)). -----------------------------------------------------------------------------*/ struct _TServer * const srvP = serverP->srvP; const char * retError; if (!srvP->serverAcceptsConnections) xmlrpc_asprintf(&retError, "ServerInit() is not valid on a server that doesn't " "accept connections " "(i.e. created with ServerCreateNoAccept)"); else { retError = NULL; /* initial value */ if (!srvP->chanSwitchP) { const char * error; createChanSwitch(srvP, &error); if (error) { xmlrpc_asprintf(&retError, "Unable to create a channel switch " "for the server. %s", error); xmlrpc_strfree(error); } } if (!retError) { const char * error; assert(srvP->chanSwitchP); ChanSwitchListen(srvP->chanSwitchP, MAX_CONN, &error); if (error) { xmlrpc_asprintf(&retError, "Failed to listen on bound socket. %s", error); xmlrpc_strfree(error); } } } if (retError) { TraceMsg("ServerInit() failed. %s", retError); return 0; xmlrpc_strfree(retError); } return 1; } /* We don't do any locking on the outstanding connections list, so we must make sure that only the master thread (the one that listens for connections) ever accesses it. That's why when a thread completes, it places the connection in "finished" status, but doesn't destroy the connection. */ typedef struct { TConn * firstP; unsigned int count; /* Redundant with 'firstP', for quick access */ } outstandingConnList; static void createOutstandingConnList(outstandingConnList ** const listPP) { outstandingConnList * listP; MALLOCVAR_NOFAIL(listP); listP->firstP = NULL; /* empty list */ listP->count = 0; *listPP = listP; } static void destroyOutstandingConnList(outstandingConnList * const listP) { assert(listP->firstP == NULL); assert(listP->count == 0); free(listP); } static void addToOutstandingConnList(outstandingConnList * const listP, TConn * const connectionP) { connectionP->nextOutstandingP = listP->firstP; listP->firstP = connectionP; ++listP->count; } static void freeFinishedConns(outstandingConnList * const listP) { /*---------------------------------------------------------------------------- Garbage-collect the resources associated with connections that are finished with their jobs. Thread resources, connection pool descriptor, etc. -----------------------------------------------------------------------------*/ TConn ** pp; pp = &listP->firstP; while (*pp) { TConn * const connectionP = (*pp); ThreadUpdateStatus(connectionP->threadP); if (connectionP->finished) { /* Take it out of the list */ *pp = connectionP->nextOutstandingP; --listP->count; ConnWaitAndRelease(connectionP); } else { /* Move to next connection in list */ pp = &connectionP->nextOutstandingP; } } } static void waitForConnectionFreed(outstandingConnList * const outstandingConnListP ATTR_UNUSED) { /*---------------------------------------------------------------------------- Wait for a connection descriptor in 'connectionPool' to be probably freed. -----------------------------------------------------------------------------*/ /* TODO: We should do something more sophisticated here. For pthreads, we can have a thread signal us by semaphore when it terminates. For fork, we might be able to use the "done" handler argument to ConnCreate() to get interrupted when the death of a child signal happens. */ xmlrpc_millisecond_sleep(2000); } static void waitForNoConnections(outstandingConnList * const outstandingConnListP) { while (outstandingConnListP->firstP) { freeFinishedConns(outstandingConnListP); if (outstandingConnListP->firstP) waitForConnectionFreed(outstandingConnListP); } } static void waitForConnectionCapacity(outstandingConnList * const outstandingConnListP) { /*---------------------------------------------------------------------------- Wait until there are fewer than the maximum allowed connections in progress. -----------------------------------------------------------------------------*/ /* We need to make this number configurable. Note that MAX_CONN (16) is also the backlog limit on the TCP socket, and they really aren't related. As it stands, we can have 16 connections in progress inside Abyss plus 16 waiting in the the channel switch. */ while (outstandingConnListP->count >= MAX_CONN) { freeFinishedConns(outstandingConnListP); if (outstandingConnListP->firstP) waitForConnectionFreed(outstandingConnListP); } } #ifndef WIN32 void ServerHandleSigchld(pid_t const pid) { ThreadHandleSigchld(pid); } #endif void ServerUseSigchld(TServer * const serverP) { struct _TServer * const srvP = serverP->srvP; srvP->useSigchld = TRUE; } static TThreadDoneFn destroyChannel; static void destroyChannel(void * const userHandle) { /*---------------------------------------------------------------------------- This is a "connection done" function for the connection the server serves. It gets called some time after the connection has done its thing. Its job is to clean up stuff the server created for use by the connection, but the server can't clean up because the connection might be processed asynchronously in a background thread. To wit, we destroy the connection's channel and release the memory for the information about that channel. -----------------------------------------------------------------------------*/ TConn * const connectionP = userHandle; ChannelDestroy(connectionP->channelP); free(connectionP->channelInfoP); } static void acceptAndProcessNextConnection( TServer * const serverP, outstandingConnList * const outstandingConnListP) { struct _TServer * const srvP = serverP->srvP; TConn * connectionP; const char * error; TChannel * channelP; void * channelInfoP; ChanSwitchAccept(srvP->chanSwitchP, &channelP, &channelInfoP, &error); if (error) { TraceMsg("Failed to accept the next connection from a client " "at the channel level. %s", error); xmlrpc_strfree(error); } else { if (channelP) { const char * error; freeFinishedConns(outstandingConnListP); waitForConnectionCapacity(outstandingConnListP); ConnCreate(&connectionP, serverP, channelP, channelInfoP, &serverFunc, &destroyChannel, ABYSS_BACKGROUND, srvP->useSigchld, &error); if (!error) { addToOutstandingConnList(outstandingConnListP, connectionP); ConnProcess(connectionP); /* When connection is done (which could be later, courtesy of a background thread), destroyChannel() will destroy *channelP. */ } else { xmlrpc_strfree(error); ChannelDestroy(channelP); free(channelInfoP); } } else { /* Accept function was interrupted before it got a connection */ } } } static void serverRun2(TServer * const serverP) { struct _TServer * const srvP = serverP->srvP; outstandingConnList * outstandingConnListP; createOutstandingConnList(&outstandingConnListP); while (!srvP->terminationRequested) acceptAndProcessNextConnection(serverP, outstandingConnListP); waitForNoConnections(outstandingConnListP); destroyOutstandingConnList(outstandingConnListP); } void ServerRun(TServer * const serverP) { struct _TServer * const srvP = serverP->srvP; if (!srvP->chanSwitchP) TraceMsg("This server is not set up to accept connections " "on its own, so you can't use ServerRun(). " "Try ServerRunConn() or ServerInit()"); else serverRun2(serverP); } static void serverRunChannel(TServer * const serverP, TChannel * const channelP, void * const channelInfoP, const char ** const errorP) { /*---------------------------------------------------------------------------- Do the HTTP transaction on the channel 'channelP'. (channel must be at the beginning of the HTTP request -- nothing having been read or written yet). channelInfoP == NULL means no channel info supplied. -----------------------------------------------------------------------------*/ struct _TServer * const srvP = serverP->srvP; TConn * connectionP; const char * error; srvP->keepalivemaxconn = 1; ConnCreate(&connectionP, serverP, channelP, channelInfoP, &serverFunc, NULL, ABYSS_FOREGROUND, srvP->useSigchld, &error); if (error) { xmlrpc_asprintf(errorP, "Couldn't create HTTP connection out of " "channel (connected socket). %s", error); xmlrpc_strfree(error); } else { *errorP = NULL; ConnProcess(connectionP); ConnWaitAndRelease(connectionP); } } void ServerRunChannel(TServer * const serverP, TChannel * const channelP, void * const channelInfoP, const char ** const errorP) { /*---------------------------------------------------------------------------- Do the HTTP transaction on the channel 'channelP'. (channel must be at the beginning of the HTTP request -- nothing having been read or written yet). -----------------------------------------------------------------------------*/ struct _TServer * const srvP = serverP->srvP; if (srvP->serverAcceptsConnections) xmlrpc_asprintf(errorP, "This server is configured to accept connections on " "its own socket. " "Try ServerRun() or ServerCreateNoAccept()."); else serverRunChannel(serverP, channelP, channelInfoP, errorP); } void ServerRunConn2(TServer * const serverP, TSocket * const connectedSocketP, const char ** const errorP) { /*---------------------------------------------------------------------------- Do the HTTP transaction on the TCP connection on the socket *connectedSocketP. (socket must be connected state, with nothing having been read or written on the connection yet). -----------------------------------------------------------------------------*/ TChannel * const channelP = SocketGetChannel(connectedSocketP); if (!channelP) xmlrpc_asprintf(errorP, "The socket supplied is not a connected " "socket. You should use ServerRunChannel() instead, " "anyway."); else ServerRunChannel(serverP, channelP, SocketGetChannelInfo(connectedSocketP), errorP); } void ServerRunConn(TServer * const serverP, TOsSocket const connectedOsSocket) { TChannel * channelP; void * channelInfoP; const char * error; createChannelFromOsSocket(connectedOsSocket, &channelP, &channelInfoP, &error); if (error) { TraceExit("Unable to use supplied socket"); xmlrpc_strfree(error); } else { const char * error; ServerRunChannel(serverP, channelP, channelInfoP, &error); if (error) { TraceExit("Failed to run server on connection on file " "descriptor %d. %s", connectedOsSocket, error); xmlrpc_strfree(error); } ChannelDestroy(channelP); free(channelInfoP); } } void ServerRunOnce(TServer * const serverP) { /*---------------------------------------------------------------------------- Accept a connection from the channel switch and do the HTTP transaction that comes over it. If no connection is presently waiting at the switch, wait for one. But return immediately if we receive a signal during the wait. -----------------------------------------------------------------------------*/ struct _TServer * const srvP = serverP->srvP; if (!srvP->chanSwitchP) TraceMsg("This server is not set up to accept connections " "on its own, so you can't use ServerRunOnce(). " "Try ServerRunChannel() or ServerInit()"); else { const char * error; TChannel * channelP; void * channelInfoP; srvP->keepalivemaxconn = 1; ChanSwitchAccept(srvP->chanSwitchP, &channelP, &channelInfoP, &error); if (error) { TraceMsg("Failed to accept the next connection from a client " "at the channel level. %s", error); xmlrpc_strfree(error); } else { if (channelP) { const char * error; serverRunChannel(serverP, channelP, channelInfoP, &error); if (error) { const char * peerDesc; ChannelFormatPeerInfo(channelP, &peerDesc); TraceExit("Got a connection from '%s', but failed to " "run server on it. %s", peerDesc, error); xmlrpc_strfree(peerDesc); xmlrpc_strfree(error); } ChannelDestroy(channelP); free(channelInfoP); } } } } void ServerRunOnce2(TServer * const serverP, enum abyss_foreback const foregroundBackground ATTR_UNUSED) { /*---------------------------------------------------------------------------- This is a backward compatibility interface to ServerRunOnce(). 'foregroundBackground' is meaningless. We always process the connection in the foreground. The parameter exists because we once thought we could do them in the background, but we really can't do that in any clean way. If Caller wants background execution, he can spin his own thread or process to call us. It makes much more sense in Caller's context. -----------------------------------------------------------------------------*/ ServerRunOnce(serverP); } static void setGroups(void) { #if HAVE_SETGROUPS if (setgroups(0, NULL) == (-1)) TraceExit("Failed to setup the group."); #endif } void ServerDaemonize(TServer * const serverP) { /*---------------------------------------------------------------------------- Turn Caller into a daemon (i.e. fork a child, then exit; the child returns to Caller). NOTE: It's ridiculous, but conventional, for us to do this. It's ridiculous because the task of daemonizing is not something particular to Abyss. It ought to be done by a higher level. In fact, it should be done before the Abyss server program is even execed. The user should run a "daemonize" program that creates a daemon which execs the Abyss server program. -----------------------------------------------------------------------------*/ struct _TServer * const srvP = serverP->srvP; #ifndef _WIN32 /* Become a daemon */ switch (fork()) { case 0: break; case -1: TraceExit("Unable to become a daemon"); default: /* We are the parent */ exit(0); } setsid(); /* Change the current user if we are root */ if (getuid()==0) { if (srvP->uid == (uid_t)-1) TraceExit("Can't run under root privileges. " "Please add a User option in your " "Abyss configuration file."); setGroups(); if (srvP->gid != (gid_t)-1) if (setgid(srvP->gid)==(-1)) TraceExit("Failed to change the group."); if (setuid(srvP->uid) == -1) TraceExit("Failed to change the user."); } if (srvP->pidfileP) { char z[16]; sprintf(z, "%d", getpid()); FileWrite(srvP->pidfileP, z, strlen(z)); FileClose(srvP->pidfileP); } #endif /* _WIN32 */ } void ServerAddHandler2(TServer * const serverP, URIHandler2 * const handlerArgP, abyss_bool * const successP) { URIHandler2 * handlerP; MALLOCVAR(handlerP); if (handlerP == NULL) *successP = FALSE; else { *handlerP = *handlerArgP; if (handlerP->init == NULL) *successP = TRUE; else handlerP->init(handlerP, successP); if (*successP) *successP = ListAdd(&serverP->srvP->handlers, handlerP); if (!*successP) free(handlerP); } } static URIHandler2 * createHandler(URIHandler const function) { URIHandler2 * handlerP; MALLOCVAR(handlerP); if (handlerP != NULL) { handlerP->init = NULL; handlerP->term = NULL; handlerP->userdata = NULL; handlerP->handleReq2 = NULL; handlerP->handleReq1 = function; } return handlerP; } abyss_bool ServerAddHandler(TServer * const serverP, URIHandler const function) { URIHandler2 * handlerP; bool success; handlerP = createHandler(function); if (handlerP == NULL) success = FALSE; else { success = ListAdd(&serverP->srvP->handlers, handlerP); if (!success) free(handlerP); } return success; } void ServerDefaultHandler(TServer * const serverP, URIHandler const handler) { struct _TServer * const srvP = serverP->srvP; if (handler) srvP->defaultHandler = handler; else { srvP->defaultHandler = HandlerDefaultBuiltin; srvP->defaultHandlerContext = srvP->builtinHandlerP; } } void LogWrite(TServer * const serverP, const char * const msg) { struct _TServer * const srvP = serverP->srvP; if (!srvP->logfileisopen && srvP->logfilename) logOpen(srvP); if (srvP->logfileisopen) { bool success; success = MutexLock(srvP->logmutexP); if (success) { const char * const lbr = "\n"; FileWrite(srvP->logfileP, msg, strlen(msg)); FileWrite(srvP->logfileP, lbr, strlen(lbr)); MutexUnlock(srvP->logmutexP); } } } /******************************************************************************* ** ** server.c ** ** This file is part of the ABYSS Web server project. ** ** Copyright (C) 2000 by Moez Mahfoudh . ** All rights reserved. ** ** Redistribution and use in source and binary forms, with or without ** modification, are permitted provided that the following conditions ** are met: ** 1. Redistributions of source code must retain the above copyright ** notice, this list of conditions and the following disclaimer. ** 2. Redistributions in binary form must reproduce the above copyright ** notice, this list of conditions and the following disclaimer in the ** documentation and/or other materials provided with the distribution. ** 3. The name of the author may not be used to endorse or promote products ** derived from this software without specific prior written permission. ** ** THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ** ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE ** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS ** OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) ** HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT ** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY ** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF ** SUCH DAMAGE. ** ******************************************************************************/