mirror of
https://github.com/rsyslog/rsyslog.git
synced 2025-12-16 11:00:41 +01:00
This prevents message loss due to "queue full" when re-enqueueing data under quite exotic settings. see also https://github.com/rsyslog/rsyslog/issues/3941#issuecomment-549765813 closes https://github.com/rsyslog/rsyslog/issues/4020
1597 lines
53 KiB
C
1597 lines
53 KiB
C
/* glbl.c - this module holds global defintions and data items.
|
|
* These are shared among the runtime library. Their use should be
|
|
* limited to cases where it is actually needed. The main intension for
|
|
* implementing them was support for the transistion from v2 to v4
|
|
* (with fully modular design), but it turned out that there may also
|
|
* be some other good use cases besides backwards-compatibility.
|
|
*
|
|
* Module begun 2008-04-16 by Rainer Gerhards
|
|
*
|
|
* Copyright 2008-2020 Rainer Gerhards and Adiscon GmbH.
|
|
*
|
|
* This file is part of the rsyslog runtime library.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
* -or-
|
|
* see COPYING.ASL20 in the source distribution
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include <stdlib.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/wait.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <pthread.h>
|
|
#include <ctype.h>
|
|
#include <assert.h>
|
|
#include <stdint.h>
|
|
#include <errno.h>
|
|
|
|
#include "rsyslog.h"
|
|
#include "obj.h"
|
|
#include "unicode-helper.h"
|
|
#include "cfsysline.h"
|
|
#include "glbl.h"
|
|
#include "prop.h"
|
|
#include "atomic.h"
|
|
#include "errmsg.h"
|
|
#include "action.h"
|
|
#include "parserif.h"
|
|
#include "rainerscript.h"
|
|
#include "srUtils.h"
|
|
#include "operatingstate.h"
|
|
#include "net.h"
|
|
#include "rsconf.h"
|
|
#include "queue.h"
|
|
#include "dnscache.h"
|
|
|
|
#define REPORT_CHILD_PROCESS_EXITS_NONE 0
|
|
#define REPORT_CHILD_PROCESS_EXITS_ERRORS 1
|
|
#define REPORT_CHILD_PROCESS_EXITS_ALL 2
|
|
|
|
/* some defaults */
|
|
#ifndef DFLT_NETSTRM_DRVR
|
|
# define DFLT_NETSTRM_DRVR ((uchar*)"ptcp")
|
|
#endif
|
|
|
|
/* static data */
|
|
DEFobjStaticHelpers
|
|
DEFobjCurrIf(prop)
|
|
DEFobjCurrIf(net)
|
|
|
|
/* static data
|
|
* For this object, these variables are obviously what makes the "meat" of the
|
|
* class...
|
|
*/
|
|
int glblDebugOnShutdown = 0; /* start debug log when we are shut down */
|
|
#ifdef ENABLE_LIBLOGGING_STDLOG
|
|
stdlog_channel_t stdlog_hdl = NULL; /* handle to be used for stdlog */
|
|
#endif
|
|
|
|
static struct cnfobj *mainqCnfObj = NULL;/* main queue object, to be used later in startup sequence */
|
|
#ifndef DFLT_INT_MSGS_SEV_FILTER
|
|
#define DFLT_INT_MSGS_SEV_FILTER 6 /* Warning level and more important */
|
|
#endif
|
|
int glblIntMsgsSeverityFilter = DFLT_INT_MSGS_SEV_FILTER;/* filter for logging internal messages by syslog sev. */
|
|
int bProcessInternalMessages = 0; /* Should rsyslog itself process internal messages?
|
|
* 1 - yes
|
|
* 0 - send them to libstdlog (e.g. to push to journal) or syslog()
|
|
*/
|
|
static uchar *pszWorkDir = NULL;
|
|
#ifdef ENABLE_LIBLOGGING_STDLOG
|
|
static uchar *stdlog_chanspec = NULL;
|
|
#endif
|
|
static int bParseHOSTNAMEandTAG = 1; /* parser modification (based on startup params!) */
|
|
static int bPreserveFQDN = 0; /* should FQDNs always be preserved? */
|
|
static int iMaxLine = 8096; /* maximum length of a syslog message */
|
|
static uchar * oversizeMsgErrorFile = NULL; /* File where oversize messages are written to */
|
|
static int oversizeMsgInputMode = 0; /* Mode which oversize messages will be forwarded */
|
|
static int reportOversizeMsg = 1; /* shall error messages be generated for oversize messages? */
|
|
static int reportChildProcessExits = REPORT_CHILD_PROCESS_EXITS_ERRORS;
|
|
static int iGnuTLSLoglevel = 0; /* Sets GNUTLS Debug Level */
|
|
static int iDefPFFamily = PF_UNSPEC; /* protocol family (IPv4, IPv6 or both) */
|
|
static int bDropMalPTRMsgs = 0;/* Drop messages which have malicious PTR records during DNS lookup */
|
|
static int option_DisallowWarning = 1; /* complain if message from disallowed sender is received */
|
|
static int bDisableDNS = 0; /* don't look up IP addresses of remote messages */
|
|
static prop_t *propLocalIPIF = NULL;/* IP address to report for the local host (default is 127.0.0.1) */
|
|
static int propLocalIPIF_set = 0; /* is propLocalIPIF already set? */
|
|
static prop_t *propLocalHostName = NULL;/* our hostname as FQDN - read-only after startup */
|
|
static prop_t *propLocalHostNameToDelete = NULL;/* see GenerateLocalHostName function hdr comment! */
|
|
static uchar *LocalHostName = NULL;/* our hostname - read-only after startup, except HUP */
|
|
static uchar *LocalHostNameOverride = NULL;/* user-overridden hostname - read-only after startup */
|
|
static uchar *LocalFQDNName = NULL;/* our hostname as FQDN - read-only after startup, except HUP */
|
|
static uchar *LocalDomain = NULL;/* our local domain name - read-only after startup, except HUP */
|
|
static char **StripDomains = NULL;
|
|
/* these domains may be stripped before writing logs - r/o after s.u., never touched by init */
|
|
static char **LocalHosts = NULL;
|
|
/* these hosts are logged with their hostname - read-only after startup, never touched by init */
|
|
static uchar *pszDfltNetstrmDrvr = NULL; /* module name of default netstream driver */
|
|
static uchar *pszDfltNetstrmDrvrCAF = NULL; /* default CA file for the netstrm driver */
|
|
static uchar *pszDfltNetstrmDrvrKeyFile = NULL; /* default key file for the netstrm driver (server) */
|
|
static uchar *pszDfltNetstrmDrvrCertFile = NULL; /* default cert file for the netstrm driver (server) */
|
|
int bTerminateInputs = 0; /* global switch that inputs shall terminate ASAP (1=> terminate) */
|
|
static uchar cCCEscapeChar = '#'; /* character to be used to start an escape sequence for control chars */
|
|
static int bDropTrailingLF = 1; /* drop trailing LF's on reception? */
|
|
static int bEscapeCCOnRcv = 1; /* escape control characters on reception: 0 - no, 1 - yes */
|
|
static int bSpaceLFOnRcv = 0; /* replace newlines with spaces on reception: 0 - no, 1 - yes */
|
|
static int bEscape8BitChars = 0; /* escape characters > 127 on reception: 0 - no, 1 - yes */
|
|
static int bEscapeTab = 1; /* escape tab control character when doing CC escapes: 0 - no, 1 - yes */
|
|
static int bParserEscapeCCCStyle = 0; /* escape control characters in c style: 0 - no, 1 - yes */
|
|
short janitorInterval = 10; /* interval (in minutes) at which the janitor runs */
|
|
int glblReportNewSenders = 0;
|
|
int glblReportGoneAwaySenders = 0;
|
|
int glblSenderStatsTimeout = 12 * 60 * 60; /* 12 hr timeout for senders */
|
|
int glblSenderKeepTrack = 0; /* keep track of known senders? */
|
|
int glblUnloadModules = 1;
|
|
int bPermitSlashInProgramname = 0;
|
|
int glblIntMsgRateLimitItv = 5;
|
|
int glblIntMsgRateLimitBurst = 500;
|
|
char** glblDbgFiles = NULL;
|
|
size_t glblDbgFilesNum = 0;
|
|
int glblDbgWhitelist = 1;
|
|
int glblPermitCtlC = 0;
|
|
int glblInputTimeoutShutdown = 1000; /* input shutdown timeout in ms */
|
|
int glblShutdownQueueDoubleSize = 0;
|
|
static const uchar * operatingStateFile = NULL;
|
|
|
|
uint64_t glblDevOptions = 0; /* to be used by developers only */
|
|
|
|
pid_t glbl_ourpid;
|
|
#ifndef HAVE_ATOMIC_BUILTINS
|
|
DEF_ATOMIC_HELPER_MUT(mutTerminateInputs);
|
|
#endif
|
|
#ifdef USE_UNLIMITED_SELECT
|
|
static int iFdSetSize = howmany(FD_SETSIZE, __NFDBITS) * sizeof (fd_mask); /* size of select() bitmask in bytes */
|
|
#endif
|
|
static uchar *SourceIPofLocalClient = NULL; /* [ar] Source IP for local client to be used on multihomed host */
|
|
|
|
tzinfo_t *tzinfos = NULL;
|
|
static int ntzinfos;
|
|
|
|
/* tables for interfacing with the v6 config system */
|
|
static struct cnfparamdescr cnfparamdescr[] = {
|
|
{ "workdirectory", eCmdHdlrString, 0 },
|
|
{ "operatingstatefile", eCmdHdlrString, 0 },
|
|
{ "dropmsgswithmaliciousdnsptrrecords", eCmdHdlrBinary, 0 },
|
|
{ "localhostname", eCmdHdlrGetWord, 0 },
|
|
{ "preservefqdn", eCmdHdlrBinary, 0 },
|
|
{ "debug.onshutdown", eCmdHdlrBinary, 0 },
|
|
{ "debug.logfile", eCmdHdlrString, 0 },
|
|
{ "debug.gnutls", eCmdHdlrPositiveInt, 0 },
|
|
{ "debug.unloadmodules", eCmdHdlrBinary, 0 },
|
|
{ "defaultnetstreamdrivercafile", eCmdHdlrString, 0 },
|
|
{ "defaultnetstreamdriverkeyfile", eCmdHdlrString, 0 },
|
|
{ "defaultnetstreamdrivercertfile", eCmdHdlrString, 0 },
|
|
{ "defaultnetstreamdriver", eCmdHdlrString, 0 },
|
|
{ "maxmessagesize", eCmdHdlrSize, 0 },
|
|
{ "oversizemsg.errorfile", eCmdHdlrGetWord, 0 },
|
|
{ "oversizemsg.report", eCmdHdlrBinary, 0 },
|
|
{ "oversizemsg.input.mode", eCmdHdlrGetWord, 0 },
|
|
{ "reportchildprocessexits", eCmdHdlrGetWord, 0 },
|
|
{ "action.reportsuspension", eCmdHdlrBinary, 0 },
|
|
{ "action.reportsuspensioncontinuation", eCmdHdlrBinary, 0 },
|
|
{ "parser.controlcharacterescapeprefix", eCmdHdlrGetChar, 0 },
|
|
{ "parser.droptrailinglfonreception", eCmdHdlrBinary, 0 },
|
|
{ "parser.escapecontrolcharactersonreceive", eCmdHdlrBinary, 0 },
|
|
{ "parser.spacelfonreceive", eCmdHdlrBinary, 0 },
|
|
{ "parser.escape8bitcharactersonreceive", eCmdHdlrBinary, 0},
|
|
{ "parser.escapecontrolcharactertab", eCmdHdlrBinary, 0},
|
|
{ "parser.escapecontrolcharacterscstyle", eCmdHdlrBinary, 0 },
|
|
{ "parser.parsehostnameandtag", eCmdHdlrBinary, 0 },
|
|
{ "parser.permitslashinprogramname", eCmdHdlrBinary, 0 },
|
|
{ "stdlog.channelspec", eCmdHdlrString, 0 },
|
|
{ "janitor.interval", eCmdHdlrPositiveInt, 0 },
|
|
{ "senders.reportnew", eCmdHdlrBinary, 0 },
|
|
{ "senders.reportgoneaway", eCmdHdlrBinary, 0 },
|
|
{ "senders.timeoutafter", eCmdHdlrPositiveInt, 0 },
|
|
{ "senders.keeptrack", eCmdHdlrBinary, 0 },
|
|
{ "inputs.timeout.shutdown", eCmdHdlrPositiveInt, 0 },
|
|
{ "privdrop.group.keepsupplemental", eCmdHdlrBinary, 0 },
|
|
{ "net.ipprotocol", eCmdHdlrGetWord, 0 },
|
|
{ "net.acladdhostnameonfail", eCmdHdlrBinary, 0 },
|
|
{ "net.aclresolvehostname", eCmdHdlrBinary, 0 },
|
|
{ "net.enabledns", eCmdHdlrBinary, 0 },
|
|
{ "net.permitACLwarning", eCmdHdlrBinary, 0 },
|
|
{ "abortonuncleanconfig", eCmdHdlrBinary, 0 },
|
|
{ "variables.casesensitive", eCmdHdlrBinary, 0 },
|
|
{ "environment", eCmdHdlrArray, 0 },
|
|
{ "processinternalmessages", eCmdHdlrBinary, 0 },
|
|
{ "umask", eCmdHdlrFileCreateMode, 0 },
|
|
{ "security.abortonidresolutionfail", eCmdHdlrBinary, 0 },
|
|
{ "internal.developeronly.options", eCmdHdlrInt, 0 },
|
|
{ "internalmsg.ratelimit.interval", eCmdHdlrPositiveInt, 0 },
|
|
{ "internalmsg.ratelimit.burst", eCmdHdlrPositiveInt, 0 },
|
|
{ "internalmsg.severity", eCmdHdlrSeverity, 0 },
|
|
{ "errormessagestostderr.maxnumber", eCmdHdlrPositiveInt, 0 },
|
|
{ "shutdown.enable.ctlc", eCmdHdlrBinary, 0 },
|
|
{ "default.action.queue.timeoutshutdown", eCmdHdlrInt, 0 },
|
|
{ "default.action.queue.timeoutactioncompletion", eCmdHdlrInt, 0 },
|
|
{ "default.action.queue.timeoutenqueue", eCmdHdlrInt, 0 },
|
|
{ "default.action.queue.timeoutworkerthreadshutdown", eCmdHdlrInt, 0 },
|
|
{ "default.ruleset.queue.timeoutshutdown", eCmdHdlrInt, 0 },
|
|
{ "default.ruleset.queue.timeoutactioncompletion", eCmdHdlrInt, 0 },
|
|
{ "default.ruleset.queue.timeoutenqueue", eCmdHdlrInt, 0 },
|
|
{ "default.ruleset.queue.timeoutworkerthreadshutdown", eCmdHdlrInt, 0 },
|
|
{ "reverselookup.cache.ttl.default", eCmdHdlrNonNegInt, 0 },
|
|
{ "reverselookup.cache.ttl.enable", eCmdHdlrBinary, 0 },
|
|
{ "shutdown.queue.doublesize", eCmdHdlrBinary, 0 },
|
|
{ "debug.files", eCmdHdlrArray, 0 },
|
|
{ "debug.whitelist", eCmdHdlrBinary, 0 }
|
|
};
|
|
static struct cnfparamblk paramblk =
|
|
{ CNFPARAMBLK_VERSION,
|
|
sizeof(cnfparamdescr)/sizeof(struct cnfparamdescr),
|
|
cnfparamdescr
|
|
};
|
|
|
|
static struct cnfparamdescr timezonecnfparamdescr[] = {
|
|
{ "id", eCmdHdlrString, CNFPARAM_REQUIRED},
|
|
{ "offset", eCmdHdlrGetWord, CNFPARAM_REQUIRED }
|
|
};
|
|
static struct cnfparamblk timezonepblk =
|
|
{ CNFPARAMBLK_VERSION,
|
|
sizeof(timezonecnfparamdescr)/sizeof(struct cnfparamdescr),
|
|
timezonecnfparamdescr
|
|
};
|
|
|
|
static struct cnfparamvals *cnfparamvals = NULL;
|
|
/* we need to support multiple calls into our param block, so we need
|
|
* to persist the current settings. Note that this must be re-set
|
|
* each time a new config load begins (TODO: create interface?)
|
|
*/
|
|
|
|
int
|
|
glblGetMaxLine(void)
|
|
{
|
|
return(iMaxLine);
|
|
}
|
|
|
|
|
|
int
|
|
GetGnuTLSLoglevel(void)
|
|
{
|
|
return(iGnuTLSLoglevel);
|
|
}
|
|
|
|
/* define a macro for the simple properties' set and get functions
|
|
* (which are always the same). This is only suitable for pretty
|
|
* simple cases which require neither checks nor memory allocation.
|
|
*/
|
|
#define SIMP_PROP(nameFunc, nameVar, dataType) \
|
|
SIMP_PROP_GET(nameFunc, nameVar, dataType) \
|
|
SIMP_PROP_SET(nameFunc, nameVar, dataType)
|
|
#define SIMP_PROP_SET(nameFunc, nameVar, dataType) \
|
|
static rsRetVal Set##nameFunc(dataType newVal) \
|
|
{ \
|
|
nameVar = newVal; \
|
|
return RS_RET_OK; \
|
|
}
|
|
#define SIMP_PROP_GET(nameFunc, nameVar, dataType) \
|
|
static dataType Get##nameFunc(void) \
|
|
{ \
|
|
return(nameVar); \
|
|
}
|
|
|
|
SIMP_PROP(PreserveFQDN, bPreserveFQDN, int)
|
|
SIMP_PROP(mainqCnfObj, mainqCnfObj, struct cnfobj *)
|
|
SIMP_PROP(DropMalPTRMsgs, bDropMalPTRMsgs, int)
|
|
SIMP_PROP(StripDomains, StripDomains, char**)
|
|
SIMP_PROP(LocalHosts, LocalHosts, char**)
|
|
SIMP_PROP(ParserControlCharacterEscapePrefix, cCCEscapeChar, uchar)
|
|
SIMP_PROP(ParserDropTrailingLFOnReception, bDropTrailingLF, int)
|
|
SIMP_PROP(ParserEscapeControlCharactersOnReceive, bEscapeCCOnRcv, int)
|
|
SIMP_PROP(ParserSpaceLFOnReceive, bSpaceLFOnRcv, int)
|
|
SIMP_PROP(ParserEscape8BitCharactersOnReceive, bEscape8BitChars, int)
|
|
SIMP_PROP(ParserEscapeControlCharacterTab, bEscapeTab, int)
|
|
SIMP_PROP(ParserEscapeControlCharactersCStyle, bParserEscapeCCCStyle, int)
|
|
#ifdef USE_UNLIMITED_SELECT
|
|
SIMP_PROP(FdSetSize, iFdSetSize, int)
|
|
#endif
|
|
|
|
SIMP_PROP_SET(DfltNetstrmDrvr, pszDfltNetstrmDrvr, uchar*) /* TODO: use custom function which frees existing value */
|
|
SIMP_PROP_SET(DfltNetstrmDrvrCAF, pszDfltNetstrmDrvrCAF, uchar*)
|
|
/* TODO: use custom function which frees existing value */
|
|
SIMP_PROP_SET(DfltNetstrmDrvrKeyFile, pszDfltNetstrmDrvrKeyFile, uchar*)
|
|
/* TODO: use custom function which frees existing value */
|
|
SIMP_PROP_SET(DfltNetstrmDrvrCertFile, pszDfltNetstrmDrvrCertFile, uchar*)
|
|
/* TODO: use custom function which frees existing value */
|
|
|
|
#undef SIMP_PROP
|
|
#undef SIMP_PROP_SET
|
|
#undef SIMP_PROP_GET
|
|
|
|
|
|
/* return global input termination status
|
|
* rgerhards, 2009-07-20
|
|
*/
|
|
static int GetGlobalInputTermState(void)
|
|
{
|
|
return ATOMIC_FETCH_32BIT(&bTerminateInputs, &mutTerminateInputs);
|
|
}
|
|
|
|
|
|
/* set global termination state to "terminate". Note that this is a
|
|
* "once in a lifetime" action which can not be undone. -- gerhards, 2009-07-20
|
|
*/
|
|
static void SetGlobalInputTermination(void)
|
|
{
|
|
ATOMIC_STORE_1_TO_INT(&bTerminateInputs, &mutTerminateInputs);
|
|
}
|
|
|
|
|
|
/* set the local host IP address to a specific string. Helper to
|
|
* small set of functions. No checks done, caller must ensure it is
|
|
* ok to call. Most importantly, the IP address must not already have
|
|
* been set. -- rgerhards, 2012-03-21
|
|
*/
|
|
static rsRetVal
|
|
storeLocalHostIPIF(uchar *myIP)
|
|
{
|
|
DEFiRet;
|
|
if(propLocalIPIF != NULL) {
|
|
CHKiRet(prop.Destruct(&propLocalIPIF));
|
|
}
|
|
CHKiRet(prop.Construct(&propLocalIPIF));
|
|
CHKiRet(prop.SetString(propLocalIPIF, myIP, ustrlen(myIP)));
|
|
CHKiRet(prop.ConstructFinalize(propLocalIPIF));
|
|
DBGPRINTF("rsyslog/glbl: using '%s' as localhost IP\n", myIP);
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* This function is used to set the IP address that is to be
|
|
* reported for the local host. Note that in order to ease things
|
|
* for the v6 config interface, we do not allow to set this more
|
|
* than once.
|
|
* rgerhards, 2012-03-21
|
|
*/
|
|
static rsRetVal
|
|
setLocalHostIPIF(void __attribute__((unused)) *pVal, uchar *pNewVal)
|
|
{
|
|
uchar myIP[128];
|
|
rsRetVal localRet;
|
|
DEFiRet;
|
|
|
|
CHKiRet(objUse(net, CORE_COMPONENT));
|
|
|
|
if(propLocalIPIF_set) {
|
|
LogError(0, RS_RET_ERR, "$LocalHostIPIF is already set "
|
|
"and cannot be reset; place it at TOP OF rsyslog.conf!");
|
|
ABORT_FINALIZE(RS_RET_ERR);
|
|
}
|
|
|
|
localRet = net.GetIFIPAddr(pNewVal, AF_UNSPEC, myIP, (int) sizeof(myIP));
|
|
if(localRet != RS_RET_OK) {
|
|
LogError(0, RS_RET_ERR, "$LocalHostIPIF: IP address for interface "
|
|
"'%s' cannnot be obtained - ignoring directive", pNewVal);
|
|
} else {
|
|
storeLocalHostIPIF(myIP);
|
|
}
|
|
|
|
|
|
finalize_it:
|
|
free(pNewVal); /* no longer needed -> is in prop! */
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* This function is used to set the global work directory name.
|
|
* It verifies that the provided directory actually exists and
|
|
* emits an error message if not.
|
|
* rgerhards, 2011-02-16
|
|
*/
|
|
static rsRetVal setWorkDir(void __attribute__((unused)) *pVal, uchar *pNewVal)
|
|
{
|
|
size_t lenDir;
|
|
int i;
|
|
struct stat sb;
|
|
DEFiRet;
|
|
|
|
/* remove trailing slashes */
|
|
lenDir = ustrlen(pNewVal);
|
|
i = lenDir - 1;
|
|
while(i > 0 && pNewVal[i] == '/') {
|
|
--i;
|
|
}
|
|
|
|
if(i < 0) {
|
|
LogError(0, RS_RET_ERR_WRKDIR, "$WorkDirectory: empty value "
|
|
"- directive ignored");
|
|
ABORT_FINALIZE(RS_RET_ERR_WRKDIR);
|
|
}
|
|
|
|
if(i != (int) lenDir - 1) {
|
|
pNewVal[i+1] = '\0';
|
|
LogError(0, RS_RET_WRN_WRKDIR, "$WorkDirectory: trailing slashes "
|
|
"removed, new value is '%s'", pNewVal);
|
|
}
|
|
|
|
if(stat((char*) pNewVal, &sb) != 0) {
|
|
LogError(0, RS_RET_ERR_WRKDIR, "$WorkDirectory: %s can not be "
|
|
"accessed, probably does not exist - directive ignored", pNewVal);
|
|
ABORT_FINALIZE(RS_RET_ERR_WRKDIR);
|
|
}
|
|
|
|
if(!S_ISDIR(sb.st_mode)) {
|
|
LogError(0, RS_RET_ERR_WRKDIR, "$WorkDirectory: %s not a directory - directive ignored",
|
|
pNewVal);
|
|
ABORT_FINALIZE(RS_RET_ERR_WRKDIR);
|
|
}
|
|
|
|
free(pszWorkDir);
|
|
pszWorkDir = pNewVal;
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* This function is used both by legacy and RainerScript conf. It is a real setter. */
|
|
static void
|
|
setMaxLine(const int64_t iNew)
|
|
{
|
|
if(iNew < 128) {
|
|
LogError(0, RS_RET_INVALID_VALUE, "maxMessageSize tried to set "
|
|
"to %lld, but cannot be less than 128 - set to 128 "
|
|
"instead", (long long) iNew);
|
|
iMaxLine = 128;
|
|
} else if(iNew > (int64_t) INT_MAX) {
|
|
LogError(0, RS_RET_INVALID_VALUE, "maxMessageSize larger than "
|
|
"INT_MAX (%d) - reduced to INT_MAX", INT_MAX);
|
|
iMaxLine = INT_MAX;
|
|
} else {
|
|
iMaxLine = (int) iNew;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static rsRetVal
|
|
legacySetMaxMessageSize(void __attribute__((unused)) *pVal, int64_t iNew)
|
|
{
|
|
setMaxLine(iNew);
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
static rsRetVal
|
|
setDebugFile(void __attribute__((unused)) *pVal, uchar *pNewVal)
|
|
{
|
|
DEFiRet;
|
|
dbgSetDebugFile(pNewVal);
|
|
free(pNewVal);
|
|
RETiRet;
|
|
}
|
|
|
|
static rsRetVal
|
|
setDebugLevel(void __attribute__((unused)) *pVal, int level)
|
|
{
|
|
DEFiRet;
|
|
dbgSetDebugLevel(level);
|
|
dbgprintf("debug level %d set via config file\n", level);
|
|
dbgprintf("This is rsyslog version " VERSION "\n");
|
|
RETiRet;
|
|
}
|
|
|
|
static rsRetVal ATTR_NONNULL()
|
|
setOversizeMsgInputMode(const uchar *const mode)
|
|
{
|
|
DEFiRet;
|
|
if(!strcmp((char*)mode, "truncate")) {
|
|
oversizeMsgInputMode = glblOversizeMsgInputMode_Truncate;
|
|
} else if(!strcmp((char*)mode, "split")) {
|
|
oversizeMsgInputMode = glblOversizeMsgInputMode_Split;
|
|
} else if(!strcmp((char*)mode, "accept")) {
|
|
oversizeMsgInputMode = glblOversizeMsgInputMode_Accept;
|
|
} else {
|
|
oversizeMsgInputMode = glblOversizeMsgInputMode_Truncate;
|
|
}
|
|
RETiRet;
|
|
}
|
|
|
|
static rsRetVal ATTR_NONNULL()
|
|
setReportChildProcessExits(const uchar *const mode)
|
|
{
|
|
DEFiRet;
|
|
if(!strcmp((char*)mode, "none")) {
|
|
reportChildProcessExits = REPORT_CHILD_PROCESS_EXITS_NONE;
|
|
} else if(!strcmp((char*)mode, "errors")) {
|
|
reportChildProcessExits = REPORT_CHILD_PROCESS_EXITS_ERRORS;
|
|
} else if(!strcmp((char*)mode, "all")) {
|
|
reportChildProcessExits = REPORT_CHILD_PROCESS_EXITS_ALL;
|
|
} else {
|
|
LogError(0, RS_RET_CONF_PARAM_INVLD,
|
|
"invalid value '%s' for global parameter reportChildProcessExits -- ignored",
|
|
mode);
|
|
iRet = RS_RET_CONF_PARAM_INVLD;
|
|
}
|
|
RETiRet;
|
|
}
|
|
|
|
static rsRetVal
|
|
setDisableDNS(int val)
|
|
{
|
|
bDisableDNS = val;
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
static int
|
|
getDisableDNS(void)
|
|
{
|
|
return bDisableDNS;
|
|
}
|
|
|
|
static rsRetVal
|
|
setOption_DisallowWarning(int val)
|
|
{
|
|
option_DisallowWarning = val;
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
static int
|
|
getOption_DisallowWarning(void)
|
|
{
|
|
return option_DisallowWarning;
|
|
}
|
|
|
|
static rsRetVal
|
|
setParseHOSTNAMEandTAG(int val)
|
|
{
|
|
bParseHOSTNAMEandTAG = val;
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
static int
|
|
getParseHOSTNAMEandTAG(void)
|
|
{
|
|
return bParseHOSTNAMEandTAG;
|
|
}
|
|
|
|
static rsRetVal
|
|
setDefPFFamily(int level)
|
|
{
|
|
DEFiRet;
|
|
iDefPFFamily = level;
|
|
RETiRet;
|
|
}
|
|
|
|
static int
|
|
getDefPFFamily(void)
|
|
{
|
|
return iDefPFFamily;
|
|
}
|
|
|
|
/* return our local IP.
|
|
* If no local IP is set, "127.0.0.1" is selected *and* set. This
|
|
* is an intensional side effect that we do in order to keep things
|
|
* consistent and avoid config errors (this will make us not accept
|
|
* setting the local IP address once a module has obtained it - so
|
|
* it forces the $LocalHostIPIF directive high up in rsyslog.conf)
|
|
* rgerhards, 2012-03-21
|
|
*/
|
|
static prop_t*
|
|
GetLocalHostIP(void)
|
|
{
|
|
assert(propLocalIPIF != NULL);
|
|
return(propLocalIPIF);
|
|
}
|
|
|
|
|
|
/* set our local hostname. Free previous hostname, if it was already set.
|
|
* Note that we do now do this in a thread. As such, the method here
|
|
* is NOT 100% clean. If we run into issues, we need to think about
|
|
* refactoring the whole way we handle the local host name processing.
|
|
* Possibly using a PROP might be the right solution then.
|
|
*/
|
|
static rsRetVal
|
|
SetLocalHostName(uchar *const newname)
|
|
{
|
|
uchar *toFree;
|
|
if(LocalHostName == NULL || strcmp((const char*)LocalHostName, (const char*) newname)) {
|
|
toFree = LocalHostName;
|
|
LocalHostName = newname;
|
|
} else {
|
|
toFree = newname;
|
|
}
|
|
free(toFree);
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
|
|
/* return our local hostname. if it is not set, "[localhost]" is returned
|
|
*/
|
|
static uchar*
|
|
GetLocalHostName(void)
|
|
{
|
|
uchar *pszRet;
|
|
|
|
if(LocalHostNameOverride != NULL) {
|
|
pszRet = LocalHostNameOverride;
|
|
goto done;
|
|
}
|
|
|
|
if(LocalHostName == NULL)
|
|
pszRet = (uchar*) "[localhost]";
|
|
else {
|
|
if(GetPreserveFQDN() == 1)
|
|
pszRet = LocalFQDNName;
|
|
else
|
|
pszRet = LocalHostName;
|
|
}
|
|
done:
|
|
return(pszRet);
|
|
}
|
|
|
|
|
|
/* return the name of the file where oversize messages are written to
|
|
*/
|
|
uchar*
|
|
glblGetOversizeMsgErrorFile(void)
|
|
{
|
|
return oversizeMsgErrorFile;
|
|
}
|
|
|
|
const uchar*
|
|
glblGetOperatingStateFile(void)
|
|
{
|
|
return operatingStateFile;
|
|
}
|
|
|
|
/* return the mode with which oversize messages will be put forward
|
|
*/
|
|
int
|
|
glblGetOversizeMsgInputMode(void)
|
|
{
|
|
return oversizeMsgInputMode;
|
|
}
|
|
|
|
int
|
|
glblReportOversizeMessage(void)
|
|
{
|
|
return reportOversizeMsg;
|
|
}
|
|
|
|
|
|
/* logs a message indicating that a child process has terminated.
|
|
* If name != NULL, prints it as the program name.
|
|
*/
|
|
void
|
|
glblReportChildProcessExit(const uchar *name, pid_t pid, int status)
|
|
{
|
|
DBGPRINTF("waitpid for child %ld returned status: %2.2x\n", (long) pid, status);
|
|
|
|
if(reportChildProcessExits == REPORT_CHILD_PROCESS_EXITS_NONE
|
|
|| (reportChildProcessExits == REPORT_CHILD_PROCESS_EXITS_ERRORS
|
|
&& WIFEXITED(status) && WEXITSTATUS(status) == 0)) {
|
|
return;
|
|
}
|
|
|
|
if(WIFEXITED(status)) {
|
|
int severity = WEXITSTATUS(status) == 0 ? LOG_INFO : LOG_WARNING;
|
|
if(name != NULL) {
|
|
LogMsg(0, NO_ERRCODE, severity, "program '%s' (pid %ld) exited with status %d",
|
|
name, (long) pid, WEXITSTATUS(status));
|
|
} else {
|
|
LogMsg(0, NO_ERRCODE, severity, "child process (pid %ld) exited with status %d",
|
|
(long) pid, WEXITSTATUS(status));
|
|
}
|
|
} else if(WIFSIGNALED(status)) {
|
|
if(name != NULL) {
|
|
LogMsg(0, NO_ERRCODE, LOG_WARNING, "program '%s' (pid %ld) terminated by signal %d",
|
|
name, (long) pid, WTERMSIG(status));
|
|
} else {
|
|
LogMsg(0, NO_ERRCODE, LOG_WARNING, "child process (pid %ld) terminated by signal %d",
|
|
(long) pid, WTERMSIG(status));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* set our local domain name. Free previous domain, if it was already set.
|
|
*/
|
|
static rsRetVal
|
|
SetLocalDomain(uchar *newname)
|
|
{
|
|
free(LocalDomain);
|
|
LocalDomain = newname;
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
|
|
/* return our local hostname. if it is not set, "[localhost]" is returned
|
|
*/
|
|
static uchar*
|
|
GetLocalDomain(void)
|
|
{
|
|
return LocalDomain;
|
|
}
|
|
|
|
|
|
/* generate the local hostname property. This must be done after the hostname info
|
|
* has been set as well as PreserveFQDN.
|
|
* rgerhards, 2009-06-30
|
|
* NOTE: This function tries to avoid locking by not destructing the previous value
|
|
* immediately. This is so that current readers can continue to use the previous name.
|
|
* Otherwise, we would need to use read/write locks to protect the update process.
|
|
* In order to do so, we save the previous value and delete it when we are called again
|
|
* the next time. Note that this in theory is racy and can lead to a double-free.
|
|
* In practice, however, the window of exposure to trigger this is extremely short
|
|
* and as this functions is very infrequently being called (on HUP), the trigger
|
|
* condition for this bug is so highly unlikely that it never occurs in practice.
|
|
* Probably if you HUP rsyslog every few milliseconds, but who does that...
|
|
* To further reduce risk potential, we do only update the property when there
|
|
* actually is a hostname change, which makes it even less likely.
|
|
* rgerhards, 2013-10-28
|
|
*/
|
|
static rsRetVal
|
|
GenerateLocalHostNameProperty(void)
|
|
{
|
|
uchar *pszPrev;
|
|
int lenPrev;
|
|
prop_t *hostnameNew;
|
|
uchar *pszName;
|
|
DEFiRet;
|
|
|
|
if(propLocalHostNameToDelete != NULL)
|
|
prop.Destruct(&propLocalHostNameToDelete);
|
|
|
|
if(LocalHostNameOverride == NULL) {
|
|
if(LocalHostName == NULL)
|
|
pszName = (uchar*) "[localhost]";
|
|
else {
|
|
if(GetPreserveFQDN() == 1)
|
|
pszName = LocalFQDNName;
|
|
else
|
|
pszName = LocalHostName;
|
|
}
|
|
} else { /* local hostname is overriden via config */
|
|
pszName = LocalHostNameOverride;
|
|
}
|
|
DBGPRINTF("GenerateLocalHostName uses '%s'\n", pszName);
|
|
|
|
if(propLocalHostName == NULL)
|
|
pszPrev = (uchar*)""; /* make sure strcmp() below does not match */
|
|
else
|
|
prop.GetString(propLocalHostName, &pszPrev, &lenPrev);
|
|
|
|
if(ustrcmp(pszPrev, pszName)) {
|
|
/* we need to update */
|
|
CHKiRet(prop.Construct(&hostnameNew));
|
|
CHKiRet(prop.SetString(hostnameNew, pszName, ustrlen(pszName)));
|
|
CHKiRet(prop.ConstructFinalize(hostnameNew));
|
|
propLocalHostNameToDelete = propLocalHostName;
|
|
propLocalHostName = hostnameNew;
|
|
}
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* return our local hostname as a string property
|
|
*/
|
|
static prop_t*
|
|
GetLocalHostNameProp(void)
|
|
{
|
|
return(propLocalHostName);
|
|
}
|
|
|
|
|
|
static rsRetVal
|
|
SetLocalFQDNName(uchar *newname)
|
|
{
|
|
free(LocalFQDNName);
|
|
LocalFQDNName = newname;
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
/* return the current localhost name as FQDN (requires FQDN to be set)
|
|
* TODO: we should set the FQDN ourselfs in here!
|
|
*/
|
|
static uchar*
|
|
GetLocalFQDNName(void)
|
|
{
|
|
return(LocalFQDNName == NULL ? (uchar*) "[localhost]" : LocalFQDNName);
|
|
}
|
|
|
|
|
|
/* return the current working directory */
|
|
static uchar*
|
|
GetWorkDir(void)
|
|
{
|
|
return(pszWorkDir == NULL ? (uchar*) "" : pszWorkDir);
|
|
}
|
|
|
|
/* return the "raw" working directory, which means
|
|
* NULL if unset.
|
|
*/
|
|
const uchar *
|
|
glblGetWorkDirRaw(void)
|
|
{
|
|
return pszWorkDir;
|
|
}
|
|
|
|
/* return the current default netstream driver */
|
|
static uchar*
|
|
GetDfltNetstrmDrvr(void)
|
|
{
|
|
return(pszDfltNetstrmDrvr == NULL ? DFLT_NETSTRM_DRVR : pszDfltNetstrmDrvr);
|
|
}
|
|
|
|
|
|
/* return the current default netstream driver CA File */
|
|
static uchar*
|
|
GetDfltNetstrmDrvrCAF(void)
|
|
{
|
|
return(pszDfltNetstrmDrvrCAF);
|
|
}
|
|
|
|
|
|
/* return the current default netstream driver key File */
|
|
static uchar*
|
|
GetDfltNetstrmDrvrKeyFile(void)
|
|
{
|
|
return(pszDfltNetstrmDrvrKeyFile);
|
|
}
|
|
|
|
|
|
/* return the current default netstream driver certificate File */
|
|
static uchar*
|
|
GetDfltNetstrmDrvrCertFile(void)
|
|
{
|
|
return(pszDfltNetstrmDrvrCertFile);
|
|
}
|
|
|
|
|
|
/* [ar] Source IP for local client to be used on multihomed host */
|
|
static rsRetVal
|
|
SetSourceIPofLocalClient(uchar *newname)
|
|
{
|
|
if(SourceIPofLocalClient != NULL) {
|
|
free(SourceIPofLocalClient); }
|
|
SourceIPofLocalClient = newname;
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
static uchar*
|
|
GetSourceIPofLocalClient(void)
|
|
{
|
|
return(SourceIPofLocalClient);
|
|
}
|
|
|
|
|
|
/* queryInterface function
|
|
* rgerhards, 2008-02-21
|
|
*/
|
|
BEGINobjQueryInterface(glbl)
|
|
CODESTARTobjQueryInterface(glbl)
|
|
if(pIf->ifVersion != glblCURR_IF_VERSION) { /* check for current version, increment on each change */
|
|
ABORT_FINALIZE(RS_RET_INTERFACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* ok, we have the right interface, so let's fill it
|
|
* Please note that we may also do some backwards-compatibility
|
|
* work here (if we can support an older interface version - that,
|
|
* of course, also affects the "if" above).
|
|
*/
|
|
pIf->GetWorkDir = GetWorkDir;
|
|
pIf->GenerateLocalHostNameProperty = GenerateLocalHostNameProperty;
|
|
pIf->GetLocalHostNameProp = GetLocalHostNameProp;
|
|
pIf->GetLocalHostIP = GetLocalHostIP;
|
|
pIf->SetGlobalInputTermination = SetGlobalInputTermination;
|
|
pIf->GetGlobalInputTermState = GetGlobalInputTermState;
|
|
pIf->GetSourceIPofLocalClient = GetSourceIPofLocalClient; /* [ar] */
|
|
pIf->SetSourceIPofLocalClient = SetSourceIPofLocalClient; /* [ar] */
|
|
pIf->SetDefPFFamily = setDefPFFamily;
|
|
pIf->GetDefPFFamily = getDefPFFamily;
|
|
pIf->SetDisableDNS = setDisableDNS;
|
|
pIf->GetDisableDNS = getDisableDNS;
|
|
pIf->GetMaxLine = glblGetMaxLine;
|
|
pIf->SetOption_DisallowWarning = setOption_DisallowWarning;
|
|
pIf->GetOption_DisallowWarning = getOption_DisallowWarning;
|
|
pIf->SetParseHOSTNAMEandTAG = setParseHOSTNAMEandTAG;
|
|
pIf->GetParseHOSTNAMEandTAG = getParseHOSTNAMEandTAG;
|
|
#define SIMP_PROP(name) \
|
|
pIf->Get##name = Get##name; \
|
|
pIf->Set##name = Set##name;
|
|
SIMP_PROP(PreserveFQDN);
|
|
SIMP_PROP(DropMalPTRMsgs);
|
|
SIMP_PROP(mainqCnfObj);
|
|
SIMP_PROP(LocalFQDNName)
|
|
SIMP_PROP(LocalHostName)
|
|
SIMP_PROP(LocalDomain)
|
|
SIMP_PROP(StripDomains)
|
|
SIMP_PROP(LocalHosts)
|
|
SIMP_PROP(ParserControlCharacterEscapePrefix)
|
|
SIMP_PROP(ParserDropTrailingLFOnReception)
|
|
SIMP_PROP(ParserEscapeControlCharactersOnReceive)
|
|
SIMP_PROP(ParserSpaceLFOnReceive)
|
|
SIMP_PROP(ParserEscape8BitCharactersOnReceive)
|
|
SIMP_PROP(ParserEscapeControlCharacterTab)
|
|
SIMP_PROP(ParserEscapeControlCharactersCStyle)
|
|
SIMP_PROP(DfltNetstrmDrvr)
|
|
SIMP_PROP(DfltNetstrmDrvrCAF)
|
|
SIMP_PROP(DfltNetstrmDrvrKeyFile)
|
|
SIMP_PROP(DfltNetstrmDrvrCertFile)
|
|
#ifdef USE_UNLIMITED_SELECT
|
|
SIMP_PROP(FdSetSize)
|
|
#endif
|
|
#undef SIMP_PROP
|
|
finalize_it:
|
|
ENDobjQueryInterface(glbl)
|
|
|
|
/* Reset config variables to default values.
|
|
* rgerhards, 2008-04-17
|
|
*/
|
|
static rsRetVal resetConfigVariables(uchar __attribute__((unused)) *pp, void __attribute__((unused)) *pVal)
|
|
{
|
|
free(pszDfltNetstrmDrvr);
|
|
pszDfltNetstrmDrvr = NULL;
|
|
free(pszDfltNetstrmDrvrCAF);
|
|
pszDfltNetstrmDrvrCAF = NULL;
|
|
free(pszDfltNetstrmDrvrKeyFile);
|
|
pszDfltNetstrmDrvrKeyFile = NULL;
|
|
free(pszDfltNetstrmDrvrCertFile);
|
|
pszDfltNetstrmDrvrCertFile = NULL;
|
|
free(LocalHostNameOverride);
|
|
LocalHostNameOverride = NULL;
|
|
free(oversizeMsgErrorFile);
|
|
oversizeMsgErrorFile = NULL;
|
|
oversizeMsgInputMode = glblOversizeMsgInputMode_Accept;
|
|
reportChildProcessExits = REPORT_CHILD_PROCESS_EXITS_ERRORS;
|
|
free(pszWorkDir);
|
|
pszWorkDir = NULL;
|
|
free((void*)operatingStateFile);
|
|
operatingStateFile = NULL;
|
|
bDropMalPTRMsgs = 0;
|
|
bPreserveFQDN = 0;
|
|
iMaxLine = 8192;
|
|
cCCEscapeChar = '#';
|
|
bDropTrailingLF = 1;
|
|
reportOversizeMsg = 1;
|
|
bEscapeCCOnRcv = 1; /* default is to escape control characters */
|
|
bSpaceLFOnRcv = 0;
|
|
bEscape8BitChars = 0; /* default is not to escape control characters */
|
|
bEscapeTab = 1; /* default is to escape tab characters */
|
|
bParserEscapeCCCStyle = 0;
|
|
#ifdef USE_UNLIMITED_SELECT
|
|
iFdSetSize = howmany(FD_SETSIZE, __NFDBITS) * sizeof (fd_mask);
|
|
#endif
|
|
return RS_RET_OK;
|
|
}
|
|
|
|
|
|
/* Prepare for new config
|
|
*/
|
|
void
|
|
glblPrepCnf(void)
|
|
{
|
|
free(mainqCnfObj);
|
|
mainqCnfObj = NULL;
|
|
free(cnfparamvals);
|
|
cnfparamvals = NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
freeTimezoneInfo(void)
|
|
{
|
|
int i;
|
|
for(i = 0 ; i < ntzinfos ; ++i)
|
|
free(tzinfos[i].id);
|
|
free(tzinfos);
|
|
tzinfos = NULL;
|
|
}
|
|
|
|
static void
|
|
displayTzinfos(void)
|
|
{
|
|
int i;
|
|
if(!Debug)
|
|
return;
|
|
for(i = 0 ; i < ntzinfos ; ++i)
|
|
dbgprintf("tzinfo: '%s':%c%2.2d:%2.2d\n",
|
|
tzinfos[i].id, tzinfos[i].offsMode,
|
|
tzinfos[i].offsHour, tzinfos[i].offsMin);
|
|
}
|
|
|
|
|
|
/* Note: this function is NOT thread-safe!
|
|
* This is currently not needed as used only during
|
|
* initialization.
|
|
*/
|
|
static rsRetVal
|
|
addTimezoneInfo(uchar *tzid, char offsMode, int8_t offsHour, int8_t offsMin)
|
|
{
|
|
DEFiRet;
|
|
tzinfo_t *newti;
|
|
CHKmalloc(newti = realloc(tzinfos, (ntzinfos+1)*sizeof(tzinfo_t)));
|
|
if((newti[ntzinfos].id = strdup((char*)tzid)) == NULL) {
|
|
free(newti);
|
|
DBGPRINTF("addTimezoneInfo: strdup failed with OOM\n");
|
|
ABORT_FINALIZE(RS_RET_OUT_OF_MEMORY);
|
|
}
|
|
newti[ntzinfos].offsMode = offsMode;
|
|
newti[ntzinfos].offsHour = offsHour;
|
|
newti[ntzinfos].offsMin = offsMin;
|
|
++ntzinfos, tzinfos = newti;
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
static int
|
|
bs_arrcmp_tzinfo(const void *s1, const void *s2)
|
|
{
|
|
return strcmp((char*)s1, (char*)((tzinfo_t*)s2)->id);
|
|
}
|
|
/* returns matching timezone info or NULL if no entry exists */
|
|
tzinfo_t*
|
|
glblFindTimezoneInfo(char *id)
|
|
{
|
|
return (tzinfo_t*) bsearch(id, tzinfos, ntzinfos, sizeof(tzinfo_t), bs_arrcmp_tzinfo);
|
|
}
|
|
|
|
/* handle the timezone() object. Each incarnation adds one additional
|
|
* zone info to the global table of time zones.
|
|
*/
|
|
|
|
int
|
|
bs_arrcmp_glblDbgFiles(const void *s1, const void *s2)
|
|
{
|
|
return strcmp((char*)s1, *(char**)s2);
|
|
}
|
|
|
|
void
|
|
glblProcessTimezone(struct cnfobj *o)
|
|
{
|
|
struct cnfparamvals *pvals;
|
|
uchar *id = NULL;
|
|
uchar *offset = NULL;
|
|
char offsMode;
|
|
int8_t offsHour;
|
|
int8_t offsMin;
|
|
int i;
|
|
|
|
pvals = nvlstGetParams(o->nvlst, &timezonepblk, NULL);
|
|
if(pvals == NULL) {
|
|
LogError(0, RS_RET_MISSING_CNFPARAMS, "error processing timezone "
|
|
"config parameters");
|
|
goto done;
|
|
}
|
|
if(Debug) {
|
|
dbgprintf("timezone param blk after glblProcessTimezone:\n");
|
|
cnfparamsPrint(&timezonepblk, pvals);
|
|
}
|
|
|
|
for(i = 0 ; i < timezonepblk.nParams ; ++i) {
|
|
if(!pvals[i].bUsed)
|
|
continue;
|
|
if(!strcmp(timezonepblk.descr[i].name, "id")) {
|
|
id = (uchar*) es_str2cstr(pvals[i].val.d.estr, NULL);
|
|
} else if(!strcmp(timezonepblk.descr[i].name, "offset")) {
|
|
offset = (uchar*) es_str2cstr(pvals[i].val.d.estr, NULL);
|
|
} else {
|
|
dbgprintf("glblProcessTimezone: program error, non-handled "
|
|
"param '%s'\n", timezonepblk.descr[i].name);
|
|
}
|
|
}
|
|
|
|
/* note: the following two checks for NULL are not strictly necessary
|
|
* as these are required parameters for the config block. But we keep
|
|
* them to make the clang static analyzer happy, which also helps
|
|
* guard against logic errors.
|
|
*/
|
|
if(offset == NULL) {
|
|
parser_errmsg("offset parameter missing (logic error?), timezone config ignored");
|
|
goto done;
|
|
}
|
|
if(id == NULL) {
|
|
parser_errmsg("id parameter missing (logic error?), timezone config ignored");
|
|
goto done;
|
|
}
|
|
|
|
if( strlen((char*)offset) != 6
|
|
|| !(offset[0] == '-' || offset[0] == '+')
|
|
|| !(isdigit(offset[1]) && isdigit(offset[2]))
|
|
|| offset[3] != ':'
|
|
|| !(isdigit(offset[4]) && isdigit(offset[5]))
|
|
) {
|
|
parser_errmsg("timezone offset has invalid format. Must be +/-hh:mm, e.g. \"-07:00\".");
|
|
goto done;
|
|
}
|
|
|
|
offsHour = (offset[1] - '0') * 10 + offset[2] - '0';
|
|
offsMin = (offset[4] - '0') * 10 + offset[5] - '0';
|
|
offsMode = offset[0];
|
|
|
|
if(offsHour > 12 || offsMin > 59) {
|
|
parser_errmsg("timezone offset outside of supported range (hours 0..12, minutes 0..59)");
|
|
goto done;
|
|
}
|
|
|
|
addTimezoneInfo(id, offsMode, offsHour, offsMin);
|
|
|
|
done:
|
|
cnfparamvalsDestruct(pvals, &timezonepblk);
|
|
free(id);
|
|
free(offset);
|
|
}
|
|
|
|
/* handle a global config object. Note that multiple global config statements
|
|
* are permitted (because of plugin support), so once we got a param block,
|
|
* we need to hold to it.
|
|
* rgerhards, 2011-07-19
|
|
*/
|
|
void
|
|
glblProcessCnf(struct cnfobj *o)
|
|
{
|
|
int i;
|
|
|
|
cnfparamvals = nvlstGetParams(o->nvlst, ¶mblk, cnfparamvals);
|
|
if(cnfparamvals == NULL) {
|
|
LogError(0, RS_RET_MISSING_CNFPARAMS, "error processing global "
|
|
"config parameters [global(...)]");
|
|
goto done;
|
|
}
|
|
if(Debug) {
|
|
dbgprintf("glbl param blk after glblProcessCnf:\n");
|
|
cnfparamsPrint(¶mblk, cnfparamvals);
|
|
}
|
|
|
|
/* The next thing is a bit hackish and should be changed in higher
|
|
* versions. There are a select few parameters which we need to
|
|
* act on immediately. These are processed here.
|
|
*/
|
|
for(i = 0 ; i < paramblk.nParams ; ++i) {
|
|
if(!cnfparamvals[i].bUsed)
|
|
continue;
|
|
if(!strcmp(paramblk.descr[i].name, "processinternalmessages")) {
|
|
bProcessInternalMessages = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "internal.developeronly.options")) {
|
|
glblDevOptions = (uint64_t) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "stdlog.channelspec")) {
|
|
#ifndef ENABLE_LIBLOGGING_STDLOG
|
|
LogError(0, RS_RET_ERR, "rsyslog wasn't "
|
|
"compiled with liblogging-stdlog support. "
|
|
"The 'stdlog.channelspec' parameter "
|
|
"is ignored. Note: the syslog API is used instead.\n");
|
|
#else
|
|
stdlog_chanspec = (uchar*)
|
|
es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
/* we need to re-open with the new channel */
|
|
stdlog_close(stdlog_hdl);
|
|
stdlog_hdl = stdlog_open("rsyslogd", 0, STDLOG_SYSLOG,
|
|
(char*) stdlog_chanspec);
|
|
#endif
|
|
} else if(!strcmp(paramblk.descr[i].name, "operatingstatefile")) {
|
|
if(operatingStateFile != NULL) {
|
|
LogError(errno, RS_RET_PARAM_ERROR,
|
|
"error: operatingStateFile already set to '%s' - "
|
|
"new valule ignored", operatingStateFile);
|
|
} else {
|
|
operatingStateFile = (uchar*) es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
osf_open();
|
|
}
|
|
}
|
|
}
|
|
done: return;
|
|
}
|
|
|
|
/* Set mainq parameters. Note that when this is not called, we'll use the
|
|
* legacy parameter config. mainq parameters can only be set once.
|
|
*/
|
|
void
|
|
glblProcessMainQCnf(struct cnfobj *o)
|
|
{
|
|
if(mainqCnfObj == NULL) {
|
|
mainqCnfObj = o;
|
|
} else {
|
|
LogError(0, RS_RET_ERR, "main_queue() object can only be specified "
|
|
"once - all but first ignored\n");
|
|
}
|
|
}
|
|
|
|
/* destruct the main q cnf object after it is no longer needed. This is
|
|
* also used to do some final checks.
|
|
*/
|
|
void
|
|
glblDestructMainqCnfObj(void)
|
|
{
|
|
/* Only destruct if not NULL! */
|
|
if (mainqCnfObj != NULL) {
|
|
nvlstChkUnused(mainqCnfObj->nvlst);
|
|
cnfobjDestruct(mainqCnfObj);
|
|
mainqCnfObj = NULL;
|
|
}
|
|
}
|
|
|
|
/* comparison function for qsort() and string array compare
|
|
* this is for the string lookup table type
|
|
*/
|
|
static int
|
|
qs_arrcmp_tzinfo(const void *s1, const void *s2)
|
|
{
|
|
return strcmp(((tzinfo_t*)s1)->id, ((tzinfo_t*)s2)->id);
|
|
}
|
|
|
|
static int
|
|
qs_arrcmp_glblDbgFiles(const void *s1, const void *s2)
|
|
{
|
|
return strcmp(*((char**)s1), *((char**)s2));
|
|
}
|
|
|
|
/* set an environment variable */
|
|
static rsRetVal
|
|
do_setenv(const char *const var)
|
|
{
|
|
char varname[128];
|
|
const char *val = var;
|
|
size_t i;
|
|
DEFiRet;
|
|
|
|
for(i = 0 ; *val != '=' ; ++i, ++val) {
|
|
if(i == sizeof(varname)-i) {
|
|
parser_errmsg("environment variable name too long "
|
|
"[max %zu chars] or malformed entry: '%s'",
|
|
sizeof(varname)-1, var);
|
|
ABORT_FINALIZE(RS_RET_ERR_SETENV);
|
|
}
|
|
if(*val == '\0') {
|
|
parser_errmsg("environment variable entry is missing "
|
|
"equal sign (for value): '%s'", var);
|
|
ABORT_FINALIZE(RS_RET_ERR_SETENV);
|
|
}
|
|
varname[i] = *val;
|
|
}
|
|
varname[i] = '\0';
|
|
++val;
|
|
DBGPRINTF("do_setenv, var '%s', val '%s'\n", varname, val);
|
|
|
|
if(setenv(varname, val, 1) != 0) {
|
|
char errStr[1024];
|
|
rs_strerror_r(errno, errStr, sizeof(errStr));
|
|
parser_errmsg("error setting environment variable "
|
|
"'%s' to '%s': %s", varname, val, errStr);
|
|
ABORT_FINALIZE(RS_RET_ERR_SETENV);
|
|
}
|
|
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* This processes the "regular" parameters which are to be set after the
|
|
* config has been fully loaded.
|
|
*/
|
|
rsRetVal
|
|
glblDoneLoadCnf(void)
|
|
{
|
|
int i;
|
|
unsigned char *cstr;
|
|
FILE *fp;
|
|
DEFiRet;
|
|
CHKiRet(objUse(net, CORE_COMPONENT));
|
|
|
|
if(ntzinfos > 0) {
|
|
qsort(tzinfos, ntzinfos, sizeof(tzinfo_t), qs_arrcmp_tzinfo);
|
|
}
|
|
DBGPRINTF("Timezone information table (%d entries):\n", ntzinfos);
|
|
displayTzinfos();
|
|
|
|
if(cnfparamvals == NULL)
|
|
goto finalize_it;
|
|
|
|
for(i = 0 ; i < paramblk.nParams ; ++i) {
|
|
if(!cnfparamvals[i].bUsed)
|
|
continue;
|
|
if(!strcmp(paramblk.descr[i].name, "workdirectory")) {
|
|
cstr = (uchar*) es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
setWorkDir(NULL, cstr);
|
|
} else if(!strcmp(paramblk.descr[i].name, "variables.casesensitive")) {
|
|
const int val = (int) cnfparamvals[i].val.d.n;
|
|
fjson_global_do_case_sensitive_comparison(val);
|
|
DBGPRINTF("global/config: set case sensitive variables to %d\n",
|
|
val);
|
|
} else if(!strcmp(paramblk.descr[i].name, "localhostname")) {
|
|
free(LocalHostNameOverride);
|
|
LocalHostNameOverride = (uchar*)
|
|
es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
} else if(!strcmp(paramblk.descr[i].name, "defaultnetstreamdriverkeyfile")) {
|
|
free(pszDfltNetstrmDrvrKeyFile);
|
|
uchar *const fn = (uchar*) es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
fp = fopen((const char*)fn, "r");
|
|
if(fp == NULL) {
|
|
LogError(errno, RS_RET_NO_FILE_ACCESS,
|
|
"error: defaultnetstreamdriverkeyfile '%s' "
|
|
"could not be accessed", fn);
|
|
} else {
|
|
fclose(fp);
|
|
pszDfltNetstrmDrvrKeyFile = fn;
|
|
}
|
|
} else if(!strcmp(paramblk.descr[i].name, "defaultnetstreamdrivercertfile")) {
|
|
free(pszDfltNetstrmDrvrCertFile);
|
|
uchar *const fn = (uchar*) es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
fp = fopen((const char*)fn, "r");
|
|
if(fp == NULL) {
|
|
LogError(errno, RS_RET_NO_FILE_ACCESS,
|
|
"error: defaultnetstreamdrivercertfile '%s' "
|
|
"could not be accessed", fn);
|
|
} else {
|
|
fclose(fp);
|
|
pszDfltNetstrmDrvrCertFile = fn;
|
|
}
|
|
} else if(!strcmp(paramblk.descr[i].name, "defaultnetstreamdrivercafile")) {
|
|
free(pszDfltNetstrmDrvrCAF);
|
|
uchar *const fn = (uchar*) es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
fp = fopen((const char*)fn, "r");
|
|
if(fp == NULL) {
|
|
LogError(errno, RS_RET_NO_FILE_ACCESS,
|
|
"error: defaultnetstreamdrivercafile file '%s' "
|
|
"could not be accessed", fn);
|
|
} else {
|
|
fclose(fp);
|
|
pszDfltNetstrmDrvrCAF = fn;
|
|
}
|
|
} else if(!strcmp(paramblk.descr[i].name, "defaultnetstreamdriver")) {
|
|
free(pszDfltNetstrmDrvr);
|
|
pszDfltNetstrmDrvr = (uchar*)
|
|
es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
} else if(!strcmp(paramblk.descr[i].name, "preservefqdn")) {
|
|
bPreserveFQDN = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name,
|
|
"dropmsgswithmaliciousdnsptrrecords")) {
|
|
bDropMalPTRMsgs = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "action.reportsuspension")) {
|
|
bActionReportSuspension = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "action.reportsuspensioncontinuation")) {
|
|
bActionReportSuspensionCont = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "maxmessagesize")) {
|
|
setMaxLine(cnfparamvals[i].val.d.n);
|
|
} else if(!strcmp(paramblk.descr[i].name, "oversizemsg.errorfile")) {
|
|
free(oversizeMsgErrorFile);
|
|
oversizeMsgErrorFile = (uchar*)es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
} else if(!strcmp(paramblk.descr[i].name, "oversizemsg.report")) {
|
|
reportOversizeMsg = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "oversizemsg.input.mode")) {
|
|
const char *const tmp = es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
setOversizeMsgInputMode((uchar*) tmp);
|
|
free((void*)tmp);
|
|
} else if(!strcmp(paramblk.descr[i].name, "reportchildprocessexits")) {
|
|
const char *const tmp = es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
setReportChildProcessExits((uchar*) tmp);
|
|
free((void*)tmp);
|
|
} else if(!strcmp(paramblk.descr[i].name, "debug.onshutdown")) {
|
|
glblDebugOnShutdown = (int) cnfparamvals[i].val.d.n;
|
|
LogError(0, RS_RET_OK, "debug: onShutdown set to %d", glblDebugOnShutdown);
|
|
} else if(!strcmp(paramblk.descr[i].name, "debug.gnutls")) {
|
|
iGnuTLSLoglevel = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "debug.unloadmodules")) {
|
|
glblUnloadModules = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.controlcharacterescapeprefix")) {
|
|
uchar* tmp = (uchar*) es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
cCCEscapeChar = tmp[0];
|
|
free(tmp);
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.droptrailinglfonreception")) {
|
|
bDropTrailingLF = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.escapecontrolcharactersonreceive")) {
|
|
bEscapeCCOnRcv = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.spacelfonreceive")) {
|
|
bSpaceLFOnRcv = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.escape8bitcharactersonreceive")) {
|
|
bEscape8BitChars = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.escapecontrolcharactertab")) {
|
|
bEscapeTab = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.escapecontrolcharacterscstyle")) {
|
|
bParserEscapeCCCStyle = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.parsehostnameandtag")) {
|
|
bParseHOSTNAMEandTAG = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "parser.permitslashinprogramname")) {
|
|
bPermitSlashInProgramname = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "debug.logfile")) {
|
|
if(pszAltDbgFileName == NULL) {
|
|
pszAltDbgFileName = es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
/* can actually happen if debug system also opened altdbg */
|
|
if(altdbg != -1) {
|
|
close(altdbg);
|
|
}
|
|
if((altdbg = open(pszAltDbgFileName, O_WRONLY|O_CREAT|O_TRUNC|O_NOCTTY
|
|
|O_CLOEXEC, S_IRUSR|S_IWUSR)) == -1) {
|
|
LogError(0, RS_RET_ERR, "debug log file '%s' could not be opened",
|
|
pszAltDbgFileName);
|
|
}
|
|
}
|
|
LogError(0, RS_RET_OK, "debug log file is '%s', fd %d", pszAltDbgFileName, altdbg);
|
|
} else if(!strcmp(paramblk.descr[i].name, "janitor.interval")) {
|
|
janitorInterval = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "net.ipprotocol")) {
|
|
char *proto = es_str2cstr(cnfparamvals[i].val.d.estr, NULL);
|
|
if(!strcmp(proto, "unspecified")) {
|
|
iDefPFFamily = PF_UNSPEC;
|
|
} else if(!strcmp(proto, "ipv4-only")) {
|
|
iDefPFFamily = PF_INET;
|
|
} else if(!strcmp(proto, "ipv6-only")) {
|
|
iDefPFFamily = PF_INET6;
|
|
} else{
|
|
LogError(0, RS_RET_ERR, "invalid net.ipprotocol "
|
|
"parameter '%s' -- ignored", proto);
|
|
}
|
|
free(proto);
|
|
} else if(!strcmp(paramblk.descr[i].name, "senders.reportnew")) {
|
|
glblReportNewSenders = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "senders.reportgoneaway")) {
|
|
glblReportGoneAwaySenders = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "senders.timeoutafter")) {
|
|
glblSenderStatsTimeout = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "senders.keeptrack")) {
|
|
glblSenderKeepTrack = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "inputs.timeout.shutdown")) {
|
|
glblInputTimeoutShutdown = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "privdrop.group.keepsupplemental")) {
|
|
loadConf->globals.gidDropPrivKeepSupplemental = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "security.abortonidresolutionfail")) {
|
|
loadConf->globals.abortOnIDResolutionFail = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "net.acladdhostnameonfail")) {
|
|
*(net.pACLAddHostnameOnFail) = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "net.aclresolvehostname")) {
|
|
*(net.pACLDontResolve) = !((int) cnfparamvals[i].val.d.n);
|
|
} else if(!strcmp(paramblk.descr[i].name, "net.enabledns")) {
|
|
setDisableDNS(!((int) cnfparamvals[i].val.d.n));
|
|
} else if(!strcmp(paramblk.descr[i].name, "net.permitwarning")) {
|
|
setOption_DisallowWarning(!((int) cnfparamvals[i].val.d.n));
|
|
} else if(!strcmp(paramblk.descr[i].name, "abortonuncleanconfig")) {
|
|
loadConf->globals.bAbortOnUncleanConfig = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "internalmsg.ratelimit.burst")) {
|
|
glblIntMsgRateLimitBurst = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "internalmsg.ratelimit.interval")) {
|
|
glblIntMsgRateLimitItv = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "internalmsg.severity")) {
|
|
glblIntMsgsSeverityFilter = (int) cnfparamvals[i].val.d.n;
|
|
if((glblIntMsgsSeverityFilter < 0) || (glblIntMsgsSeverityFilter > 7)) {
|
|
parser_errmsg("invalid internalmsg.severity value");
|
|
glblIntMsgsSeverityFilter = DFLT_INT_MSGS_SEV_FILTER;
|
|
}
|
|
} else if(!strcmp(paramblk.descr[i].name, "environment")) {
|
|
for(int j = 0 ; j < cnfparamvals[i].val.d.ar->nmemb ; ++j) {
|
|
char *const var = es_str2cstr(cnfparamvals[i].val.d.ar->arr[j], NULL);
|
|
do_setenv(var);
|
|
free(var);
|
|
}
|
|
} else if(!strcmp(paramblk.descr[i].name, "errormessagestostderr.maxnumber")) {
|
|
loadConf->globals.maxErrMsgToStderr = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "debug.files")) {
|
|
free(glblDbgFiles); /* "fix" Coverity false positive */
|
|
glblDbgFilesNum = cnfparamvals[i].val.d.ar->nmemb;
|
|
glblDbgFiles = (char**) malloc(cnfparamvals[i].val.d.ar->nmemb * sizeof(char*));
|
|
for(int j = 0 ; j < cnfparamvals[i].val.d.ar->nmemb ; ++j) {
|
|
glblDbgFiles[j] = es_str2cstr(cnfparamvals[i].val.d.ar->arr[j], NULL);
|
|
}
|
|
qsort(glblDbgFiles, glblDbgFilesNum, sizeof(char*), qs_arrcmp_glblDbgFiles);
|
|
} else if(!strcmp(paramblk.descr[i].name, "debug.whitelist")) {
|
|
glblDbgWhitelist = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "shutdown.queue.doublesize")) {
|
|
glblShutdownQueueDoubleSize = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "umask")) {
|
|
loadConf->globals.umask = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "shutdown.enable.ctlc")) {
|
|
glblPermitCtlC = (int) cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.action.queue.timeoutshutdown")) {
|
|
actq_dflt_toQShutdown = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.action.queue.timeoutactioncompletion")) {
|
|
actq_dflt_toActShutdown = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.action.queue.timeoutenqueue")) {
|
|
actq_dflt_toEnq = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.action.queue.timeoutworkerthreadshutdown")) {
|
|
actq_dflt_toWrkShutdown = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.ruleset.queue.timeoutshutdown")) {
|
|
ruleset_dflt_toQShutdown = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.ruleset.queue.timeoutactioncompletion")) {
|
|
ruleset_dflt_toActShutdown = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.ruleset.queue.timeoutenqueue")) {
|
|
ruleset_dflt_toEnq = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "default.ruleset.queue.timeoutworkerthreadshutdown")) {
|
|
ruleset_dflt_toWrkShutdown = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "reverselookup.cache.ttl.default")) {
|
|
dnscacheDefaultTTL = cnfparamvals[i].val.d.n;
|
|
} else if(!strcmp(paramblk.descr[i].name, "reverselookup.cache.ttl.enable")) {
|
|
dnscacheEnableTTL = cnfparamvals[i].val.d.n;
|
|
} else {
|
|
dbgprintf("glblDoneLoadCnf: program error, non-handled "
|
|
"param '%s'\n", paramblk.descr[i].name);
|
|
}
|
|
}
|
|
|
|
if(glblDebugOnShutdown && Debug != DEBUG_FULL) {
|
|
Debug = DEBUG_ONDEMAND;
|
|
stddbg = -1;
|
|
}
|
|
|
|
finalize_it: RETiRet;
|
|
}
|
|
|
|
|
|
/* Initialize the glbl class. Must be called as the very first method
|
|
* before anything else is called inside this class.
|
|
* rgerhards, 2008-02-19
|
|
*/
|
|
BEGINAbstractObjClassInit(glbl, 1, OBJ_IS_CORE_MODULE) /* class, version */
|
|
/* request objects we use */
|
|
CHKiRet(objUse(prop, CORE_COMPONENT));
|
|
|
|
/* intialize properties */
|
|
storeLocalHostIPIF((uchar*)"127.0.0.1");
|
|
|
|
/* config handlers are never unregistered and need not be - we are always loaded ;) */
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"debugfile", 0, eCmdHdlrGetWord, setDebugFile, NULL, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"debuglevel", 0, eCmdHdlrInt, setDebugLevel, NULL, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"workdirectory", 0, eCmdHdlrGetWord, setWorkDir, NULL, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"dropmsgswithmaliciousdnsptrrecords", 0, eCmdHdlrBinary, NULL,
|
|
&bDropMalPTRMsgs, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"defaultnetstreamdriver", 0, eCmdHdlrGetWord, NULL, &pszDfltNetstrmDrvr,
|
|
NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"defaultnetstreamdrivercafile", 0, eCmdHdlrGetWord, NULL,
|
|
&pszDfltNetstrmDrvrCAF, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"defaultnetstreamdriverkeyfile", 0, eCmdHdlrGetWord, NULL,
|
|
&pszDfltNetstrmDrvrKeyFile, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"defaultnetstreamdrivercertfile", 0, eCmdHdlrGetWord, NULL,
|
|
&pszDfltNetstrmDrvrCertFile, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"localhostname", 0, eCmdHdlrGetWord, NULL, &LocalHostNameOverride, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"localhostipif", 0, eCmdHdlrGetWord, setLocalHostIPIF, NULL, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"optimizeforuniprocessor", 0, eCmdHdlrGoneAway, NULL, NULL, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"preservefqdn", 0, eCmdHdlrBinary, NULL, &bPreserveFQDN, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"maxmessagesize", 0, eCmdHdlrSize, legacySetMaxMessageSize, NULL, NULL));
|
|
|
|
/* Deprecated parser config options */
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"controlcharacterescapeprefix", 0, eCmdHdlrGetChar, NULL,
|
|
&cCCEscapeChar, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"droptrailinglfonreception", 0, eCmdHdlrBinary, NULL,
|
|
&bDropTrailingLF, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"escapecontrolcharactersonreceive", 0, eCmdHdlrBinary, NULL,
|
|
&bEscapeCCOnRcv, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"spacelfonreceive", 0, eCmdHdlrBinary, NULL, &bSpaceLFOnRcv, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"escape8bitcharactersonreceive", 0, eCmdHdlrBinary, NULL,
|
|
&bEscape8BitChars, NULL));
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"escapecontrolcharactertab", 0, eCmdHdlrBinary, NULL, &bEscapeTab, NULL));
|
|
|
|
CHKiRet(regCfSysLineHdlr((uchar *)"resetconfigvariables", 1, eCmdHdlrCustomHandler,
|
|
resetConfigVariables, NULL, NULL));
|
|
|
|
INIT_ATOMIC_HELPER_MUT(mutTerminateInputs);
|
|
ENDObjClassInit(glbl)
|
|
|
|
|
|
/* Exit the glbl class.
|
|
* rgerhards, 2008-04-17
|
|
*/
|
|
BEGINObjClassExit(glbl, OBJ_IS_CORE_MODULE) /* class, version */
|
|
free(pszDfltNetstrmDrvr);
|
|
free(pszDfltNetstrmDrvrCAF);
|
|
free(pszDfltNetstrmDrvrKeyFile);
|
|
free(pszDfltNetstrmDrvrCertFile);
|
|
free(pszWorkDir);
|
|
free(LocalDomain);
|
|
free(LocalHostName);
|
|
free(LocalHostNameOverride);
|
|
free(oversizeMsgErrorFile);
|
|
free(LocalFQDNName);
|
|
freeTimezoneInfo();
|
|
objRelease(prop, CORE_COMPONENT);
|
|
if(propLocalHostNameToDelete != NULL)
|
|
prop.Destruct(&propLocalHostNameToDelete);
|
|
DESTROY_ATOMIC_HELPER_MUT(mutTerminateInputs);
|
|
ENDObjClassExit(glbl)
|