/* * ***** BEGIN LICENSE BLOCK ***** Copyright (C) 2022 Olof Hagsand and Kristofer Hallin This file is part of CLIXON. 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 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. Alternatively, the contents of this file may be used under the terms of the GNU General Public License Version 3 or later (the "GPL"), in which case the provisions of the GPL are applicable instead of those above. If you wish to allow use of your version of this file only under the terms of the GPL, and not to allow others to use your version of this file under the terms of Apache License version 2, indicate your decision by deleting the provisions above and replace them with the notice and other provisions required by the GPL. If you do not delete the provisions above, a recipient may use your version of this file under the terms of any one of the Apache License version 2 or the GPL. ***** END LICENSE BLOCK ***** */ #ifdef HAVE_CONFIG_H #include "clixon_config.h" /* generated by config & autoconf */ #endif #include #include #include #include #include #include #include /* net-snmp */ #include #include #include /* cligen */ #include /* clicon */ #include /* Command line options to be passed to getopt(3) */ #define SNMP_OPTS "hD:f:l:o:" #if 1 // XXX hardcoded from https://github.com/net-snmp/net-snmp/blob/master/agent/mibgroup/testhandler.c Netsnmp_Node_Handler my_test_handler; Netsnmp_Node_Handler my_test_table_handler; Netsnmp_Node_Handler my_data_table_handler; Netsnmp_Node_Handler my_test_instance_handler; static oid my_test_oid[4] = { 1, 2, 3, 4 }; static oid my_table_oid[4] = { 1, 2, 3, 5 }; static oid my_instance_oid[5] = { 1, 2, 3, 6, 1 }; static oid my_data_table_oid[4] = { 1, 2, 3, 7 }; static oid my_data_ulong_instance[4] = { 1, 2, 3, 9 }; u_long my_ulong = 42; void init_testhandler(void) { /* * we're registering at .1.2.3.4 */ netsnmp_handler_registration *my_test; netsnmp_table_registration_info *table_info; u_long ind1; netsnmp_table_data *table; netsnmp_table_row *row; clicon_debug(1, "%s", __FUNCTION__); /* * basic handler test */ netsnmp_register_handler(netsnmp_create_handler_registration ("myTest", my_test_handler, my_test_oid, 4, HANDLER_CAN_RONLY)); /* * instance handler test */ netsnmp_register_instance(netsnmp_create_handler_registration ("myInstance", my_test_instance_handler, my_instance_oid, 5, HANDLER_CAN_RWRITE)); netsnmp_register_ulong_instance("myulong", my_data_ulong_instance, 4, &my_ulong, NULL); /* * table helper test */ my_test = netsnmp_create_handler_registration("myTable", my_test_table_handler, my_table_oid, 4, HANDLER_CAN_RONLY); if (!my_test) return; table_info = SNMP_MALLOC_TYPEDEF(netsnmp_table_registration_info); if (table_info == NULL) return; netsnmp_table_helper_add_indexes(table_info, ASN_INTEGER, ASN_INTEGER, 0); table_info->min_column = 3; table_info->max_column = 3; netsnmp_register_table(my_test, table_info); /* * data table helper test */ /* * we'll construct a simple table here with two indexes: an * integer and a string (why not). It'll contain only one * column so the data pointer is merely the data in that * column. */ table = netsnmp_create_table_data("data_table_test"); netsnmp_table_data_add_index(table, ASN_INTEGER); netsnmp_table_data_add_index(table, ASN_OCTET_STR); /* * 1 partridge in a pear tree */ row = netsnmp_create_table_data_row(); ind1 = 1; netsnmp_table_row_add_index(row, ASN_INTEGER, &ind1, sizeof(ind1)); netsnmp_table_row_add_index(row, ASN_OCTET_STR, "partridge", strlen("partridge")); row->data = NETSNMP_REMOVE_CONST(void *, "pear tree"); netsnmp_table_data_add_row(table, row); /* * 2 turtle doves */ row = netsnmp_create_table_data_row(); ind1 = 2; netsnmp_table_row_add_index(row, ASN_INTEGER, &ind1, sizeof(ind1)); netsnmp_table_row_add_index(row, ASN_OCTET_STR, "turtle", strlen("turtle")); row->data = NETSNMP_REMOVE_CONST(void *, "doves"); netsnmp_table_data_add_row(table, row); /* * we're going to register it as a normal table too, so we get the * automatically parsed column and index information */ table_info = SNMP_MALLOC_TYPEDEF(netsnmp_table_registration_info); if (table_info == NULL) return; netsnmp_table_helper_add_indexes(table_info, ASN_INTEGER, ASN_OCTET_STR, 0); table_info->min_column = 3; table_info->max_column = 3; netsnmp_register_read_only_table_data( netsnmp_create_handler_registration("12days", my_data_table_handler, my_data_table_oid, 4, HANDLER_CAN_RONLY), table, table_info); } int my_test_handler(netsnmp_mib_handler *handler, netsnmp_handler_registration *reginfo, netsnmp_agent_request_info *reqinfo, netsnmp_request_info *requests) { oid myoid1[] = { 1, 2, 3, 4, 5, 6 }; static u_long accesses = 0; clicon_debug(1, "%s", __FUNCTION__); /* * loop through requests */ while (requests) { netsnmp_variable_list *var = requests->requestvb; DEBUGMSGTL(("testhandler", " oid:")); DEBUGMSGOID(("testhandler", var->name, var->name_length)); DEBUGMSG(("testhandler", "\n")); switch (reqinfo->mode) { case MODE_GET: if (netsnmp_oid_equals(var->name, var->name_length, myoid1, 6) == 0) { snmp_set_var_typed_value(var, ASN_INTEGER, (u_char *) & accesses, sizeof(accesses)); return SNMP_ERR_NOERROR; } break; case MODE_GETNEXT: if (snmp_oid_compare(var->name, var->name_length, myoid1, 6) < 0) { snmp_set_var_objid(var, myoid1, 6); snmp_set_var_typed_value(var, ASN_INTEGER, (u_char *) & accesses, sizeof(accesses)); return SNMP_ERR_NOERROR; } break; default: netsnmp_set_request_error(reqinfo, requests, SNMP_ERR_GENERR); break; } requests = requests->next; } return SNMP_ERR_NOERROR; } /* * functionally this is a simply a multiplication table for 12x12 */ #define MAX_COLONE 12 #define MAX_COLTWO 12 #define RESULT_COLUMN 3 int my_test_table_handler(netsnmp_mib_handler *handler, netsnmp_handler_registration *reginfo, netsnmp_agent_request_info *reqinfo, netsnmp_request_info *requests) { netsnmp_table_registration_info *handler_reg_info = (netsnmp_table_registration_info *) handler->prev->myvoid; netsnmp_table_request_info *table_info; u_long result; int x, y; DEBUGMSGTL(("testhandler", "Got request:\n")); while (requests) { netsnmp_variable_list *var = requests->requestvb; if (requests->processed != 0) continue; DEBUGMSGTL(("testhandler_table", "Got request:\n")); DEBUGMSGTL(("testhandler_table", " oid:")); DEBUGMSGOID(("testhandler_table", var->name, var->name_length)); DEBUGMSG(("testhandler_table", "\n")); table_info = netsnmp_extract_table_info(requests); if (table_info == NULL) { requests = requests->next; continue; } switch (reqinfo->mode) { case MODE_GETNEXT: /* * beyond our search range? */ if (table_info->colnum > RESULT_COLUMN) break; /* * below our minimum column? */ if (table_info->colnum < RESULT_COLUMN || /* * or no index specified */ table_info->indexes->val.integer == NULL) { table_info->colnum = RESULT_COLUMN; x = 0; y = 0; } else { x = *(table_info->indexes->val.integer); y = *(table_info->indexes->next_variable->val.integer); } if (table_info->number_indexes == handler_reg_info->number_indexes) { y++; /* GETNEXT is basically just y+1 for this table */ if (y > MAX_COLTWO) { /* (with wrapping) */ y = 0; x++; } } if (x <= MAX_COLONE) { result = x * y; *(table_info->indexes->val.integer) = x; *(table_info->indexes->next_variable->val.integer) = y; netsnmp_table_build_result(reginfo, requests, table_info, ASN_INTEGER, (u_char *) & result, sizeof(result)); } break; case MODE_GET: if (var->type == ASN_NULL) { /* valid request if ASN_NULL */ /* * is it the right column? */ if (table_info->colnum == RESULT_COLUMN && /* * and within the max boundries? */ *(table_info->indexes->val.integer) <= MAX_COLONE && *(table_info->indexes->next_variable->val.integer) <= MAX_COLTWO) { /* * then, the result is column1 * column2 */ result = *(table_info->indexes->val.integer) * *(table_info->indexes->next_variable->val.integer); snmp_set_var_typed_value(var, ASN_INTEGER, (u_char *) & result, sizeof(result)); } } break; } requests = requests->next; } return SNMP_ERR_NOERROR; } #define TESTHANDLER_SET_NAME "my_test" int my_test_instance_handler(netsnmp_mib_handler *handler, netsnmp_handler_registration *reginfo, netsnmp_agent_request_info *reqinfo, netsnmp_request_info *requests) { static u_long accesses = 42; u_long *accesses_cache = NULL; clicon_debug(1, "%s", __FUNCTION__); switch (reqinfo->mode) { case MODE_GET: snmp_set_var_typed_value(requests->requestvb, ASN_UNSIGNED, (u_char *) & accesses, sizeof(accesses)); break; #ifndef NETSNMP_NO_WRITE_SUPPORT case MODE_SET_RESERVE1: if (requests->requestvb->type != ASN_UNSIGNED) netsnmp_set_request_error(reqinfo, requests, SNMP_ERR_WRONGTYPE); break; case MODE_SET_RESERVE2: /* * store old info for undo later */ accesses_cache = netsnmp_memdup(&accesses, sizeof(accesses)); if (accesses_cache == NULL) { netsnmp_set_request_error(reqinfo, requests, SNMP_ERR_RESOURCEUNAVAILABLE); return SNMP_ERR_NOERROR; } netsnmp_request_add_list_data(requests, netsnmp_create_data_list (TESTHANDLER_SET_NAME, accesses_cache, free)); break; case MODE_SET_ACTION: /* * update current */ accesses = *(requests->requestvb->val.integer); DEBUGMSGTL(("testhandler", "updated accesses -> %lu\n", accesses)); break; case MODE_SET_UNDO: accesses = *((u_long *) netsnmp_request_get_list_data(requests, TESTHANDLER_SET_NAME)); break; case MODE_SET_COMMIT: case MODE_SET_FREE: /* * nothing to do */ break; #endif /* NETSNMP_NO_WRITE_SUPPORT */ } return SNMP_ERR_NOERROR; } int my_data_table_handler(netsnmp_mib_handler *handler, netsnmp_handler_registration *reginfo, netsnmp_agent_request_info *reqinfo, netsnmp_request_info *requests) { char *column3; netsnmp_table_request_info *table_info; netsnmp_table_row *row; clicon_debug(1, "%s", __FUNCTION__); while (requests) { if (requests->processed) { requests = requests->next; continue; } /* * extract our stored data and table info */ row = netsnmp_extract_table_row(requests); table_info = netsnmp_extract_table_info(requests); if (!table_info || !row || !row->data) continue; column3 = (char *) row->data; /* * there's only one column, we don't need to check if it's right */ netsnmp_table_data_build_result(reginfo, reqinfo, requests, row, table_info->colnum, ASN_OCTET_STR, (u_char*)column3, strlen(column3)); requests = requests->next; } return SNMP_ERR_NOERROR; } #endif /*! Signal terminates process * Just set exit flag for proper exit in event loop */ static void clixon_snmp_sig_term(int arg) { clicon_log(LOG_NOTICE, "%s: %s: pid: %u Signal %d", __PROGRAM__, __FUNCTION__, getpid(), arg); /* This should ensure no more accepts or incoming packets are processed because next time eventloop * is entered, it will terminate. * However there may be a case of sockets closing rather abruptly for clients */ clixon_exit_set(1); } /*! Callback for single socket * This is a workaround for netsnmps API usiing fdset:s, instead an fdset is created before calling * the snmp api * @param[in] s Read socket * @param[in] arg Clixon handle */ static int clixon_snmp_input_cb(int s, void *arg) { int retval = -1; fd_set readfds; // clicon_handle h = (clicon_handle)arg; clicon_debug(1, "%s", __FUNCTION__); FD_ZERO(&readfds); FD_SET(s, &readfds); snmp_read(&readfds); retval = 0; // done: return retval; } /*! Get which sockets are used from SNMP API, the register single sockets into clixon event system * * This is a workaround for netsnmps API usiing fdset:s, instead an fdset is created before calling * the snmp api * if you use select(), see snmp_select_info() in snmp_api(3) * snmp_select_info(int *numfds, fd_set *fdset, struct timeval *timeout, int *block) * @see clixon_snmp_input_cb */ static int clixon_snmp_fdset_register(clicon_handle h) { int retval = -1; int numfds = 0; fd_set readfds; struct timeval timeout = { LONG_MAX, 0 }; int block = 0; int nr; int i; FD_ZERO(&readfds); if ((nr = snmp_sess_select_info(NULL, &numfds, &readfds, &timeout, &block)) < 0){ clicon_err(OE_SNMP, errno, "snmp_select_error"); goto done; } for (i=0; i\tDebug level\n" "\t-f \tConfiguration file (mandatory)\n" "\t-l (e|o|s|f) Log on std(e)rr, std(o)ut, (s)yslog(default), (f)ile\n" "\t-o \"