Anthony Minessale b66e1bad4c fix xml_rpc socket shutdown
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@9796 d0543943-73ff-0310-b7d9-9358b9ac24b2
2008-10-02 18:50:09 +00:00

1371 lines
36 KiB
C

/* Copyright information is at end of file */
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <errno.h>
#ifndef WIN32
#include <grp.h>
#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 <mmoez@bigfoot.com>.
** 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.
**
******************************************************************************/