blob: b7cb3a7bda6a7a7ccb626bd9bb781c7600fd40bc [file] [log] [blame]
#include <net-snmp/net-snmp-config.h>
#if HAVE_STDLIB_H
#include <stdlib.h>
#endif
#if HAVE_UNISTD_H
#include <unistd.h>
#endif
#include <stdio.h>
#if HAVE_STRING_H
#include <string.h>
#else
#include <strings.h>
#endif
#include <ctype.h>
#include <sys/types.h>
#if HAVE_WINSOCK_H
#include <winsock.h>
#else
#include <netinet/in.h>
#include <netdb.h>
#endif
#if HAVE_SYS_WAIT_H
#include <sys/wait.h>
#endif
#include <net-snmp/config_api.h>
#include <net-snmp/output_api.h>
#include <net-snmp/mib_api.h>
#include <net-snmp/utilities.h>
#include <net-snmp/net-snmp-includes.h>
#include <net-snmp/agent/net-snmp-agent-includes.h>
#include "utilities/execute.h"
#include "snmptrapd_handlers.h"
#include "snmptrapd_log.h"
char *syslog_format1 = NULL;
char *syslog_format2 = NULL;
char *print_format1 = NULL;
char *print_format2 = NULL;
const char *trap1_std_str = "%.4y-%.2m-%.2l %.2h:%.2j:%.2k %B [%b] (via %A [%a]): %N\n\t%W Trap (%q) Uptime: %#T\n%v\n";
const char *trap2_std_str = "%.4y-%.2m-%.2l %.2h:%.2j:%.2k %B [%b]:\n%v\n";
const char *
trap_description(int trap)
{
switch (trap) {
case SNMP_TRAP_COLDSTART:
return "Cold Start";
case SNMP_TRAP_WARMSTART:
return "Warm Start";
case SNMP_TRAP_LINKDOWN:
return "Link Down";
case SNMP_TRAP_LINKUP:
return "Link Up";
case SNMP_TRAP_AUTHFAIL:
return "Authentication Failure";
case SNMP_TRAP_EGPNEIGHBORLOSS:
return "EGP Neighbor Loss";
case SNMP_TRAP_ENTERPRISESPECIFIC:
return "Enterprise Specific";
default:
return "Unknown Type";
}
}
void
snmptrapd_parse_traphandle(const char *token, char *line)
{
char buf[STRINGMAX];
oid obuf[MAX_OID_LEN];
size_t olen = MAX_OID_LEN;
char *cptr;
netsnmp_trapd_handler *traph;
cptr = copy_nword(line, buf, sizeof(buf));
DEBUGMSGTL(("read_config:traphandle", "registering handler for: "));
if (!strcmp(buf, "default")) {
DEBUGMSG(("read_config:traphandle", "default"));
traph = netsnmp_add_default_traphandler( command_handler );
} else {
if (!read_objid(buf, obuf, &olen)) {
char buf1[STRINGMAX];
snprintf(buf1, sizeof(buf1),
"Bad trap OID in traphandle directive: %s", buf);
buf1[ sizeof(buf1)-1 ] = 0;
config_perror(buf1);
return;
}
DEBUGMSGOID(("read_config:traphandle", obuf, olen));
traph = netsnmp_add_traphandler( command_handler, obuf, olen );
}
DEBUGMSG(("read_config:traphandle", "\n"));
if (traph) {
traph->token = strdup(cptr);
}
}
static void
parse_forward(const char *token, char *line)
{
char buf[STRINGMAX];
oid obuf[MAX_OID_LEN];
size_t olen = MAX_OID_LEN;
char *cptr;
netsnmp_trapd_handler *traph;
cptr = copy_nword(line, buf, sizeof(buf));
DEBUGMSGTL(("read_config:forward", "registering forward for: "));
if (!strcmp(buf, "default")) {
DEBUGMSG(("read_config:forward", "default"));
traph = netsnmp_add_default_traphandler( forward_handler );
} else {
if (!read_objid(buf, obuf, &olen)) {
char buf1[STRINGMAX];
snprintf(buf1, sizeof(buf1),
"Bad trap OID in forward directive: %s", buf);
buf1[ sizeof(buf1)-1 ] = 0;
config_perror(buf1);
return;
}
DEBUGMSGOID(("read_config:forward", obuf, olen));
traph = netsnmp_add_traphandler( forward_handler, obuf, olen );
}
DEBUGMSG(("read_config:forward", "\n"));
if (traph) {
traph->token = strdup(cptr);
}
}
static void
parse_format(const char *token, char *line)
{
char *cp;
/*
* Extract the first token from the value
* which tells us which style of format this is
*/
cp = line;
while (*cp && !isspace(*cp))
cp++;
if (!(*cp)) {
/*
* If we haven't got anything left,
* then this entry is malformed.
* So report this, and give up
*/
return;
}
*cp = '\0';
cp++;
/*
* OK - now "line" contains the format type,
* and cp points to the actual format string.
* So update the appropriate pointer(s).
*
* XXX - the previous values really need to be freed too
*/
if (!strcmp( line, "print1"))
print_format1 = strdup(cp);
else if (!strcmp( line, "print2"))
print_format2 = strdup(cp);
else if (!strcmp( line, "print")) {
print_format1 = strdup(cp);
print_format2 = strdup(cp);
} else if (!strcmp( line, "syslog1"))
syslog_format1 = strdup(cp);
else if (!strcmp( line, "syslog2"))
syslog_format2 = strdup(cp);
else if (!strcmp( line, "syslog")) {
syslog_format1 = strdup(cp);
syslog_format2 = strdup(cp);
}
}
static void
parse_trap1_fmt(const char *token, char *line)
{
print_format1 = strdup(line);
}
void
free_trap1_fmt(void)
{
if (print_format1 && print_format1 != trap1_std_str)
free((char *) print_format1);
print_format1 = NULL;
}
static void
parse_trap2_fmt(const char *token, char *line)
{
print_format2 = strdup(line);
}
void
free_trap2_fmt(void)
{
if (print_format2 && print_format2 != trap2_std_str)
free((char *) print_format2);
print_format2 = NULL;
}
void
snmptrapd_register_configs( void )
{
register_config_handler("snmptrapd", "traphandle",
snmptrapd_parse_traphandle, NULL,
"oid|\"default\" program [args ...] ");
register_config_handler("snmptrapd", "format1",
parse_trap1_fmt, free_trap1_fmt, "format");
register_config_handler("snmptrapd", "format2",
parse_trap2_fmt, free_trap2_fmt, "format");
register_config_handler("snmptrapd", "format",
parse_format, NULL,
"[print{,1,2}|syslog{,1,2}] format");
register_config_handler("snmptrapd", "forward",
parse_forward, NULL, "OID|\"default\" destination");
}
/*-----------------------------
*
* Routines to implement a "registry" of trap handlers
*
*-----------------------------*/
netsnmp_trapd_handler *netsnmp_auth_global_traphandlers = NULL;
netsnmp_trapd_handler *netsnmp_pre_global_traphandlers = NULL;
netsnmp_trapd_handler *netsnmp_post_global_traphandlers = NULL;
netsnmp_trapd_handler *netsnmp_default_traphandlers = NULL;
netsnmp_trapd_handler *netsnmp_specific_traphandlers = NULL;
/*
* Register a new "global" traphandler,
* to be applied to *all* incoming traps
*/
netsnmp_trapd_handler *
netsnmp_add_global_traphandler(int list, Netsnmp_Trap_Handler handler) {
netsnmp_trapd_handler *traph;
if ( !handler )
return NULL;
traph = SNMP_MALLOC_TYPEDEF(netsnmp_trapd_handler);
if ( !traph )
return NULL;
/*
* Add this new handler to the front of the appropriate global list
* (or should it go on the end?)
*/
traph->handler = handler;
switch (list) {
case NETSNMPTRAPD_AUTH_HANDLER:
traph->nexth = netsnmp_auth_global_traphandlers;
netsnmp_auth_global_traphandlers = traph;
break;
case NETSNMPTRAPD_PRE_HANDLER:
traph->nexth = netsnmp_pre_global_traphandlers;
netsnmp_pre_global_traphandlers = traph;
break;
case NETSNMPTRAPD_POST_HANDLER:
traph->nexth = netsnmp_post_global_traphandlers;
netsnmp_post_global_traphandlers = traph;
break;
default:
free( traph );
return NULL;
}
return traph;
}
/*
* Register a new "default" traphandler, to be applied to all
* traps with no specific trap handlers of their own.
*/
netsnmp_trapd_handler *
netsnmp_add_default_traphandler( Netsnmp_Trap_Handler handler) {
netsnmp_trapd_handler *traph;
if ( !handler )
return NULL;
traph = SNMP_MALLOC_TYPEDEF(netsnmp_trapd_handler);
if ( !traph )
return NULL;
/*
* Add this new handler to the front of the default list
* (or should it go on the end?)
*/
traph->handler = handler;
traph->nexth = netsnmp_default_traphandlers;
netsnmp_default_traphandlers = traph;
return traph;
}
/*
* Register a new trap-specific traphandler
*/
netsnmp_trapd_handler *
netsnmp_add_traphandler(Netsnmp_Trap_Handler handler,
oid *trapOid, int trapOidLen ) {
netsnmp_trapd_handler *traph, *traph2;
if ( !handler )
return NULL;
traph = SNMP_MALLOC_TYPEDEF(netsnmp_trapd_handler);
if ( !traph )
return NULL;
/*
* Populate this new handler with the trap information
* (NB: the OID fields were not used in the default/global lists)
*/
traph->handler = handler;
traph->trapoid_len = trapOidLen;
memdup((u_char **)&(traph->trapoid), (u_char *)trapOid,
sizeof(oid) * trapOidLen);
/*
* Now try to find the appropriate place in the trap-specific
* list for this particular trap OID. If there's a matching OID
* already, then find it. Otherwise find the one that follows.
* If we run out of entried, the new one should be tacked onto the end.
*/
for (traph2 = netsnmp_specific_traphandlers;
traph2; traph2 = traph2->nextt) {
/* XXX - check this! */
if (snmp_oid_compare(traph2->trapoid, traph2->trapoid_len,
trapOid, trapOidLen) <= 0)
break;
}
if (traph2) {
/*
* OK - We've either got an exact match, or we've found the
* entry *after* where the new one should go.
*/
if (!snmp_oid_compare(traph->trapoid, traph->trapoid_len,
traph2->trapoid, traph2->trapoid_len)) {
/*
* Exact match, so find the end of the *handler* list
* and tack on this new entry...
*/
while (traph2->nexth)
traph2 = traph2->nexth;
traph2->nexth = traph;
traph->nextt = traph2->nextt; /* Might as well... */
traph->prevt = traph2->prevt;
} else {
/*
* .. or the following entry, so insert the new one before it.
*/
traph->prevt = traph2->prevt;
if (traph2->prevt)
traph2->prevt->nextt = traph;
else
netsnmp_specific_traphandlers = traph;
traph2->prevt = traph;
traph->nextt = traph2;
}
} else {
/*
* If we've run out of entries without finding a suitable spot,
* the new one should be tacked onto the end.....
*/
if (netsnmp_specific_traphandlers) {
traph2 = netsnmp_specific_traphandlers;
while (traph2->nextt)
traph2 = traph2->nextt;
traph2->nextt = traph;
traph->prevt = traph2;
} else {
/*
* .... unless this is the very first entry, of course!
*/
netsnmp_specific_traphandlers = traph;
}
}
return traph;
}
/*
* Locate the list of handlers for this particular Trap OID
* Returns NULL if there are no relevant traps
*/
netsnmp_trapd_handler *
netsnmp_get_traphandler( oid *trapOid, int trapOidLen ) {
netsnmp_trapd_handler *traph;
if (!trapOid || !trapOidLen)
return NULL;
/*
* Look for a matching OID, and return that list...
*/
for (traph = netsnmp_specific_traphandlers;
traph; traph=traph->nextt ) {
if ( !snmp_oid_compare(traph->trapoid, traph->trapoid_len,
trapOid, trapOidLen)) {
DEBUGMSGTL(( "snmptrapd", "get_traphandler matched (%x)\n", traph));
return traph;
}
}
/*
* .... or failing that, return the "default" list (which may be NULL)
*/
DEBUGMSGTL(( "snmptrapd", "get_traphandler default (%x)\n",
netsnmp_default_traphandlers));
return netsnmp_default_traphandlers;
}
/*-----------------------------
*
* Standard traphandlers for the common requirements
*
*-----------------------------*/
#define SYSLOG_V1_STANDARD_FORMAT "%a: %W Trap (%q) Uptime: %#T%#v\n"
#define SYSLOG_V1_ENTERPRISE_FORMAT "%a: %W Trap (%q) Uptime: %#T%#v\n" /* XXX - (%q) become (.N) ??? */
#define SYSLOG_V23_NOTIFICATION_FORMAT "%B [%b]: Trap %#v\n" /* XXX - introduces a leading " ," */
/*
* Trap handler for logging via syslog
*/
int syslog_handler( netsnmp_pdu *pdu,
netsnmp_transport *transport,
netsnmp_trapd_handler *handler)
{
u_char *rbuf = NULL;
size_t r_len = 64, o_len = 0;
int trunc = 0;
extern int SyslogTrap;
DEBUGMSGTL(( "snmptrapd", "syslog_handler\n"));
if (SyslogTrap)
return NETSNMPTRAPD_HANDLER_OK;
if ((rbuf = (u_char *) calloc(r_len, 1)) == NULL) {
snmp_log(LOG_ERR, "couldn't display trap -- malloc failed\n");
return NETSNMPTRAPD_HANDLER_FAIL; /* Failed but keep going */
}
/*
* If there's a format string registered for this trap, then use it.
*/
if (handler && handler->format) {
DEBUGMSGTL(( "snmptrapd", "format = '%s'\n", handler->format));
if (*handler->format) {
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
handler->format, pdu, transport);
} else {
free(rbuf);
return NETSNMPTRAPD_HANDLER_OK; /* A 0-length format string means don't log */
}
/*
* Otherwise (i.e. a NULL handler format string),
* use a standard output format setting
* either configurable, or hardwired
*
* XXX - v1 traps use a different hardwired formats for
* standard and enterprise specific traps
* Do we actually need this?
*/
} else {
if ( pdu->command == SNMP_MSG_TRAP ) {
if (syslog_format1) {
DEBUGMSGTL(( "snmptrapd", "syslog_format v1 = '%s'\n", syslog_format1));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
syslog_format1, pdu, transport);
} else if (pdu->trap_type == SNMP_TRAP_ENTERPRISESPECIFIC) {
DEBUGMSGTL(( "snmptrapd", "v1 enterprise format\n"));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
SYSLOG_V1_ENTERPRISE_FORMAT,
pdu, transport);
} else {
DEBUGMSGTL(( "snmptrapd", "v1 standard trap format\n"));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
SYSLOG_V1_STANDARD_FORMAT,
pdu, transport);
}
} else { /* SNMPv2/3 notifications */
if (syslog_format2) {
DEBUGMSGTL(( "snmptrapd", "syslog_format v1 = '%s'\n", syslog_format2));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
syslog_format2, pdu, transport);
} else {
DEBUGMSGTL(( "snmptrapd", "v2/3 format\n"));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
SYSLOG_V23_NOTIFICATION_FORMAT,
pdu, transport);
}
}
}
snmp_log(LOG_WARNING, "%s%s", rbuf, (trunc?" [TRUNCATED]\n":""));
free(rbuf);
return NETSNMPTRAPD_HANDLER_OK;
}
#define PRINT_V23_NOTIFICATION_FORMAT "%.4y-%.2m-%.2l %.2h:%.2j:%.2k %B [%b]:\n%v\n"
/*
* Trap handler for logging to a file
*/
int print_handler( netsnmp_pdu *pdu,
netsnmp_transport *transport,
netsnmp_trapd_handler *handler)
{
u_char *rbuf = NULL;
size_t r_len = 64, o_len = 0;
int trunc = 0;
extern int dropauth;
DEBUGMSGTL(( "snmptrapd", "print_handler\n"));
/*
* Don't bother logging authentication failures
* XXX - can we handle this via suitable handler entries instead?
*/
if (pdu->trap_type == SNMP_TRAP_AUTHFAIL && dropauth)
return NETSNMPTRAPD_HANDLER_OK;
if ((rbuf = (u_char *) calloc(r_len, 1)) == NULL) {
snmp_log(LOG_ERR, "couldn't display trap -- malloc failed\n");
return NETSNMPTRAPD_HANDLER_FAIL; /* Failed but keep going */
}
/*
* If there's a format string registered for this trap, then use it.
*/
if (handler && handler->format) {
DEBUGMSGTL(( "snmptrapd", "format = '%s'\n", handler->format));
if (*handler->format) {
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
handler->format, pdu, transport);
} else {
free(rbuf);
return NETSNMPTRAPD_HANDLER_OK; /* A 0-length format string means don't log */
}
/*
* Otherwise (i.e. a NULL handler format string),
* use a standard output format setting
* either configurable, or hardwired
*
* XXX - v1 traps use a different routine for hardwired output
* Do we actually need this separate v1 routine?
* Or would a suitable format string be sufficient?
*/
} else {
if ( pdu->command == SNMP_MSG_TRAP ) {
if (print_format1) {
DEBUGMSGTL(( "snmptrapd", "print_format v1 = '%s'\n", print_format1));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
print_format1, pdu, transport);
} else {
DEBUGMSGTL(( "snmptrapd", "v1 format\n"));
trunc = !realloc_format_plain_trap(&rbuf, &r_len, &o_len, 1,
pdu, transport);
}
} else {
if (print_format2) {
DEBUGMSGTL(( "snmptrapd", "print_format v2 = '%s'\n", print_format2));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
print_format2, pdu, transport);
} else {
DEBUGMSGTL(( "snmptrapd", "v2/3 format\n"));
trunc = !realloc_format_trap(&rbuf, &r_len, &o_len, 1,
PRINT_V23_NOTIFICATION_FORMAT,
pdu, transport);
}
}
}
snmp_log(LOG_INFO, "%s%s", rbuf, (trunc?" [TRUNCATED]\n":""));
free(rbuf);
return NETSNMPTRAPD_HANDLER_OK;
}
/*
* Trap handler for invoking a suitable script
*/
void
send_handler_data(FILE * file, struct hostent *host,
netsnmp_pdu *pdu, netsnmp_transport *transport)
{
netsnmp_variable_list tmpvar, *vars;
static oid trapoids[] = { 1, 3, 6, 1, 6, 3, 1, 1, 5, 0 };
static oid snmpsysuptime[] = { 1, 3, 6, 1, 2, 1, 1, 3, 0 };
static oid snmptrapoid[] = { 1, 3, 6, 1, 6, 3, 1, 1, 4, 1, 0 };
static oid snmptrapent[] = { 1, 3, 6, 1, 6, 3, 1, 1, 4, 3, 0 };
static oid snmptrapaddr[] = { 1, 3, 6, 1, 6, 3, 18, 1, 3, 0 };
static oid snmptrapcom[] = { 1, 3, 6, 1, 6, 3, 18, 1, 4, 0 };
oid enttrapoid[MAX_OID_LEN];
int enttraplen = pdu->enterprise_length;
char *tstr = NULL;
if (transport != NULL && transport->f_fmtaddr != NULL) {
tstr = transport->f_fmtaddr(transport, pdu->transport_data,
pdu->transport_data_length);
fprintf(file, "%s\n%s\n", host ? host->h_name : tstr, tstr);
free(tstr);
} else {
fprintf(file, "%s\n<UNKNOWN>\n", host ? host->h_name : "<UNKNOWN>");
}
if (pdu->command == SNMP_MSG_TRAP) {
/*
* convert a v1 trap to a v2 variable binding list:
* The uptime and trapOID go first in the list.
*/
tmpvar.val.integer = (long *) &pdu->time;
tmpvar.val_len = sizeof(pdu->time);
tmpvar.type = ASN_TIMETICKS;
fprint_variable(file, snmpsysuptime,
sizeof(snmpsysuptime) / sizeof(oid), &tmpvar);
tmpvar.type = ASN_OBJECT_ID;
if (pdu->trap_type == SNMP_TRAP_ENTERPRISESPECIFIC) {
memcpy(enttrapoid, pdu->enterprise, sizeof(oid) * enttraplen);
if (enttrapoid[enttraplen - 1] != 0)
enttrapoid[enttraplen++] = 0;
enttrapoid[enttraplen++] = pdu->specific_type;
tmpvar.val.objid = enttrapoid;
tmpvar.val_len = enttraplen * sizeof(oid);
} else {
trapoids[9] = pdu->trap_type + 1;
tmpvar.val.objid = trapoids;
tmpvar.val_len = 10 * sizeof(oid);
}
fprint_variable(file, snmptrapoid,
sizeof(snmptrapoid) / sizeof(oid), &tmpvar);
}
/*
* do the variables in the pdu
*/
for (vars = pdu->variables; vars; vars = vars->next_variable) {
fprint_variable(file, vars->name, vars->name_length, vars);
}
if (pdu->command == SNMP_MSG_TRAP) {
/*
* convert a v1 trap to a v2 variable binding list:
* The enterprise goes last.
*/
tmpvar.val.string = pdu->agent_addr;
tmpvar.val_len = 4;
tmpvar.type = ASN_IPADDRESS;
fprint_variable(file, snmptrapaddr,
sizeof(snmptrapaddr) / sizeof(oid), &tmpvar);
tmpvar.val.string = pdu->community;
tmpvar.val_len = pdu->community_len;
tmpvar.type = ASN_OCTET_STR;
fprint_variable(file, snmptrapcom,
sizeof(snmptrapcom) / sizeof(oid), &tmpvar);
tmpvar.val.objid = pdu->enterprise;
tmpvar.val_len = pdu->enterprise_length * sizeof(oid);
tmpvar.type = ASN_OBJECT_ID;
fprint_variable(file, snmptrapent,
sizeof(snmptrapent) / sizeof(oid), &tmpvar);
}
}
void
do_external(char *cmd, struct hostent *host,
netsnmp_pdu *pdu, netsnmp_transport *transport)
{
FILE *file;
int oldquick, result;
DEBUGMSGTL(("snmptrapd", "Running: %s\n", cmd));
oldquick = snmp_get_quick_print();
snmp_set_quick_print(1);
if (cmd) {
#ifndef WIN32
int fd[2];
int pid;
if (pipe(fd)) {
snmp_log_perror("pipe");
}
if ((pid = fork()) == 0) {
/*
* child
*/
close(0);
if (dup(fd[0]) != 0) {
snmp_log_perror("dup");
}
close(fd[1]);
close(fd[0]);
system(cmd);
exit(0);
} else if (pid > 0) {
file = fdopen(fd[1], "w");
send_handler_data(file, host, pdu, transport);
fclose(file);
close(fd[0]);
close(fd[1]);
if (waitpid(pid, &result, 0) < 0) {
snmp_log_perror("waitpid");
}
} else {
snmp_log_perror("fork");
}
#else
char command_buf[128];
char file_buf[L_tmpnam];
tmpnam(file_buf);
file = fopen(file_buf, "w");
if (!file) {
fprintf(stderr, "fopen: %s: %s\n", file_buf, strerror(errno));
} else {
send_handler_data(file, host, pdu, transport);
fclose(file);
snprintf(command_buf, sizeof(command_buf),
"%s < %s", cmd, file_buf);
command_buf[ sizeof(command_buf)-1 ] = 0;
result = system(command_buf);
if (result == -1)
fprintf(stderr, "system: %s: %s\n", command_buf,
strerror(errno));
else if (result)
fprintf(stderr, "system: %s: %d\n", command_buf, result);
remove(file_buf);
}
#endif /* WIN32 */
}
snmp_set_quick_print(oldquick);
}
#define EXECUTE_FORMAT "%B\n%b\n%V\n%v\n"
int command_handler( netsnmp_pdu *pdu,
netsnmp_transport *transport,
netsnmp_trapd_handler *handler)
{
u_char *rbuf = NULL;
size_t r_len = 64, o_len = 0;
int oldquick;
DEBUGMSGTL(( "snmptrapd", "command_handler\n"));
DEBUGMSGTL(( "snmptrapd", "token = '%s'\n", handler->token));
if (handler && handler->token && *handler->token) {
netsnmp_pdu *v2_pdu = NULL;
if (pdu->command == SNMP_MSG_TRAP)
v2_pdu = convert_v1pdu_to_v2(pdu);
else
v2_pdu = pdu;
oldquick = snmp_get_quick_print();
snmp_set_quick_print(1);
/*
* Format the trap and pass this string to the external command
*/
if ((rbuf = (u_char *) calloc(r_len, 1)) == NULL) {
snmp_log(LOG_ERR, "couldn't display trap -- malloc failed\n");
return NETSNMPTRAPD_HANDLER_FAIL; /* Failed but keep going */
}
/*
* If there's a format string registered for this trap, then use it.
* Otherwise use the standard execution format setting.
*/
if (handler && handler->format && *handler->format) {
DEBUGMSGTL(( "snmptrapd", "format = '%s'\n", handler->format));
realloc_format_trap(&rbuf, &r_len, &o_len, 1,
handler->format,
v2_pdu, transport);
} else {
DEBUGMSGTL(( "snmptrapd", "execute format\n"));
realloc_format_trap(&rbuf, &r_len, &o_len, 1,
EXECUTE_FORMAT,
v2_pdu, transport);
}
/*
* and pass this formatted string to the command specified
*/
run_exec_command(handler->token, rbuf, NULL, 0); /* Not interested in output */
snmp_set_quick_print(oldquick);
if (pdu->command == SNMP_MSG_TRAP)
snmp_free_pdu(v2_pdu);
free(rbuf);
}
return NETSNMPTRAPD_HANDLER_OK;
}
/*
* Trap handler for doing something with "event" traps
* (not entirely clear what this is about ???)
*/
/* XXX - in snmptrapd.c */
void event_input(netsnmp_variable_list * vp);
int event_handler( netsnmp_pdu *pdu,
netsnmp_transport *transport,
netsnmp_trapd_handler *handler)
{
DEBUGMSGTL(( "snmptrapd", "event_handler\n"));
event_input(pdu->variables);
return NETSNMPTRAPD_HANDLER_OK;
}
/*
* Trap handler for forwarding to another destination
*/
int forward_handler( netsnmp_pdu *pdu,
netsnmp_transport *transport,
netsnmp_trapd_handler *handler)
{
netsnmp_session session, *ss;
netsnmp_pdu *pdu2;
char buf[BUFSIZ], *cp;
DEBUGMSGTL(( "snmptrapd", "forward_handler (%s)\n", handler->token));
snmp_sess_init( &session );
if (strchr( handler->token, ':') == NULL) {
snprintf( buf, BUFSIZ, "%s:%d", handler->token, SNMP_TRAP_PORT);
cp = buf;
} else {
cp = handler->token;
}
session.peername = cp;
session.version = pdu->version;
ss = snmp_open( &session );
pdu2 = snmp_clone_pdu(pdu);
if (pdu2->transport_data) {
free(pdu2->transport_data);
pdu2->transport_data = NULL;
pdu2->transport_data_length = 0;
}
if (!snmp_send( ss, pdu2 )) {
snmp_sess_perror("Forward failed", ss);
snmp_free_pdu(pdu2);
}
snmp_close( ss );
return NETSNMPTRAPD_HANDLER_OK;
}
/*-----------------------------
*
* Main driving code, to process an incoming trap
*
*-----------------------------*/
int
snmp_input(int op, netsnmp_session *session,
int reqid, netsnmp_pdu *pdu, void *magic)
{
oid stdTrapOidRoot[] = { 1, 3, 6, 1, 6, 3, 1, 1, 5 };
oid snmpTrapOid[] = { 1, 3, 6, 1, 6, 3, 1, 1, 4, 1, 0 };
oid trapOid[MAX_OID_LEN+2] = {0};
int trapOidLen;
netsnmp_variable_list *vars;
netsnmp_trapd_handler *traph;
netsnmp_transport *transport = (netsnmp_transport *) magic;
int ret;
extern netsnmp_trapd_handler *netsnmp_auth_global_traphandlers;
extern netsnmp_trapd_handler *netsnmp_pre_global_traphandlers;
extern netsnmp_trapd_handler *netsnmp_post_global_traphandlers;
switch (op) {
case NETSNMP_CALLBACK_OP_RECEIVED_MESSAGE:
/*
* Determine the OID that identifies the trap being handled
*/
DEBUGMSGTL(("snmptrapd", "input: %x\n", pdu->command));
switch (pdu->command) {
case SNMP_MSG_TRAP:
/*
* Convert v1 traps into a v2-style trap OID
* (following RFC 2576)
*/
if (pdu->trap_type == SNMP_TRAP_ENTERPRISESPECIFIC) {
trapOidLen = pdu->enterprise_length;
memcpy(trapOid, pdu->enterprise, sizeof(oid) * trapOidLen);
if (trapOid[trapOidLen - 1] != 0) {
trapOid[trapOidLen++] = 0;
}
trapOid[trapOidLen++] = pdu->specific_type;
} else {
memcpy(trapOid, stdTrapOidRoot, sizeof(stdTrapOidRoot));
trapOidLen = OID_LENGTH(stdTrapOidRoot); /* 9 */
trapOid[trapOidLen++] = pdu->trap_type+1;
}
break;
case SNMP_MSG_TRAP2:
case SNMP_MSG_INFORM:
/*
* v2c/v3 notifications *should* have snmpTrapOID as the
* second varbind, so we can go straight there.
* But check, just to make sure
*/
vars = pdu->variables;
if (vars)
vars = vars->next_variable;
if (!vars || snmp_oid_compare(vars->name, vars->name_length,
snmpTrapOid, OID_LENGTH(snmpTrapOid))) {
/*
* Didn't find it!
* Let's look through the full list....
*/
for ( vars = pdu->variables; vars; vars=vars->next_variable) {
if (!snmp_oid_compare(vars->name, vars->name_length,
snmpTrapOid, OID_LENGTH(snmpTrapOid)))
break;
}
if (!vars) {
/*
* Still can't find it! Give up.
*/
snmp_log(LOG_ERR, "Cannot find TrapOID in TRAP2 PDU\n");
return 1; /* ??? */
}
}
memcpy(trapOid, vars->val.objid, vars->val_len);
trapOidLen = vars->val_len /sizeof(oid);
break;
default:
/* SHOULDN'T HAPPEN! */
return 1; /* ??? */
}
DEBUGMSGTL(( "snmptrapd", "Trap OID: "));
DEBUGMSGOID(("snmptrapd", trapOid, trapOidLen));
DEBUGMSG(( "snmptrapd", "\n"));
/*
* OK - We've found the Trap OID used to identify this trap.
* Call each of the various lists of handlers:
* a) authentication-related handlers,
* b) other handlers to be applied to all traps
* (*before* trap-specific handlers)
* c) the handler(s) specific to this trap
t * d) any other global handlers
*
* In each case, a particular trap handler can abort further
* processing - either just for that particular list,
* or for the trap completely.
*
* This is particularly designed for authentication-related
* handlers, but can also be used elsewhere.
*
* OK - Enough waffling, let's get to work.....
*/
/*
* a) authentication handlers
*/
traph = netsnmp_auth_global_traphandlers;
while (traph) {
ret = (*(traph->handler))(pdu, transport, traph);
if (ret == NETSNMPTRAPD_HANDLER_FINISH)
return 1;
if (ret == NETSNMPTRAPD_HANDLER_BREAK)
break;
traph = traph->nexth;
}
/*
* b) pre-specific global handlers
*/
traph = netsnmp_pre_global_traphandlers;
while (traph) {
ret = (*(traph->handler))(pdu, transport, traph);
if (ret == NETSNMPTRAPD_HANDLER_FINISH)
return 1;
if (ret == NETSNMPTRAPD_HANDLER_BREAK)
break;
traph = traph->nexth;
}
/*
* c) trap-specific handlers
*/
traph = netsnmp_get_traphandler(trapOid, trapOidLen);
while (traph) {
ret = (*(traph->handler))(pdu, transport, traph);
if (ret == NETSNMPTRAPD_HANDLER_FINISH)
return 1;
if (ret == NETSNMPTRAPD_HANDLER_BREAK)
break;
traph = traph->nexth;
}
/*
* d) other global handlers
*/
traph = netsnmp_post_global_traphandlers;
while (traph) {
ret = (*(traph->handler))(pdu, transport, traph);
if (ret == NETSNMPTRAPD_HANDLER_FINISH)
return 1;
if (ret == NETSNMPTRAPD_HANDLER_BREAK)
break;
traph = traph->nexth;
}
if (pdu->command == SNMP_MSG_INFORM) {
netsnmp_pdu *reply = snmp_clone_pdu(pdu);
if (!reply) {
snmp_log(LOG_ERR, "couldn't clone PDU for INFORM response\n");
} else {
reply->command = SNMP_MSG_RESPONSE;
reply->errstat = 0;
reply->errindex = 0;
if (!snmp_send(session, reply)) {
snmp_sess_perror("snmptrapd: Couldn't respond to inform pdu",
session);
snmp_free_pdu(reply);
}
}
}
break;
case NETSNMP_CALLBACK_OP_TIMED_OUT:
snmp_log(LOG_ERR, "Timeout: This shouldn't happen!\n");
break;
default:
snmp_log(LOG_ERR, "Unknown operation (%d): This shouldn't happen!\n", op);
break;
}
return 0;
}