0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-03 15:41:54 +00:00
bird/proto/snmp/snmp_test.c
2023-07-26 14:34:55 +02:00

527 lines
14 KiB
C

/*
* BIRD -- Simple Network Management Protocol (SNMP) Unit tests
*
* (c) 2022 Vojtech Vilimek <vojtech.vilimek@nic.cz>
* (c) 2022 CZ.NIC z.s.p.o
*
* Can be freely distributed and used under the terms of the GNU GPL.
*/
#include "test/birdtest.h"
#include "test/bt-utils.h"
#include "bgp_mib.h"
#include "subagent.h"
#include "snmp.h"
#include "snmp_utils.h"
#define SNMP_EXPECTED(actual, expected) \
bt_debug("%s expected: %3u actual: %3u\n", \
#expected, expected, actual);
#ifdef CPU_BIG_ENDIAN
#define BYTE_ORD 1
#else
#define BYTE_ORD 0
#endif
#define OID_ALLOCATE(size) mb_alloc(&root_pool, sizeof(struct oid) + (size) * sizeof (u32))
#define OID_INIT(oid, n_subid_, prefix_, include_, arr_) \
(oid)->n_subid = (n_subid_); \
(oid)->prefix = (prefix_); \
(oid)->include = (include_); \
memcpy((oid)->ids, (arr_), sizeof(arr_)); \
void
test_fill(struct snmp_proto *p)
{
((struct proto *) p)->pool = &root_pool;
}
static void
test_oid(struct oid *oid, uint base_size)
{
/* tests all states one by one */
oid->n_subid = base_size + 2;
oid->ids[base_size + 0] = 1;
oid->ids[base_size + 1] = 15; // BGP4-MIB::bgp
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_BGP);
oid->n_subid = base_size + 3;
oid->ids[base_size + 2] = 1; // BGP4-MIB::bgpVersion
snmp_oid_dump(oid);
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_VERSION);
oid->ids[base_size + 2] = 2; // BGP4-MIB::bgpLocalAs
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LOCAL_AS);
oid->ids[base_size + 2] = 3; // BGP4-MIB::bgpPeerTable
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_PEER_TABLE);
bt_debug("testing BGP4-MIB::bgpPeerEntry\n");
oid->n_subid = base_size + 4;
bt_debug("arith\n");
oid->ids[base_size + 2] = 3;
oid->ids[base_size + 3] = 1; // BGP4-MIB::bgpPeerEntry
bt_debug("dumpping\n");
bt_debug("after dump, assertion\n");
// SNMP_EXPECTED(snmp_bgp_state(oid), BGP_INTERNAL_PEER_ENTRY);
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_PEER_ENTRY);
bt_debug("finish\n");
oid->n_subid = base_size + 5;
oid->ids[base_size + 2] = 3;
oid->ids[base_size + 3] = 1;
oid->ids[base_size + 4] = 1; // BGP4-MIB::bgpPeerIdentifier
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_IDENTIFIER);
oid->ids[base_size + 4] = 2; // BGP4-MIB::bgpPeerState
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_STATE);
oid->ids[base_size + 4] = 3; // BGP4-MIB::bgpPeerAdminStatus
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_ADMIN_STATUS);
bt_debug(" 1/4\n");
oid->ids[base_size + 4] = 4; // BGP4-MIB::bgpPeerNegotiatedVersion
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_NEGOTIATED_VERSION);
oid->ids[base_size + 4] = 5; // BGP4-MIB::bgpPeerLocalAddr
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LOCAL_ADDR);
oid->ids[base_size + 4] = 6; // BGP4-MIB::bgpPeerLocalPort
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LOCAL_PORT);
oid->ids[base_size + 4] = 7; // BGP4-MIB::bgpPeerRemoteAddr
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_REMOTE_ADDR);
oid->ids[base_size + 4] = 8; // BGP4-MIB::bgpPeerRemotePort
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_REMOTE_PORT);
oid->ids[base_size + 4] = 9; // BGP4-MIB::bgpPeerRemoteAs
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_REMOTE_AS);
oid->ids[base_size + 4] = 10; // BGP4-MIB::bgpPeerInUpdates
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_RX_UPDATES);
bt_debug(" 1/2 \n");
oid->ids[base_size + 4] = 11; // BGP4-MIB::bgpPeerOutUpdates
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_TX_UPDATES);
oid->ids[base_size + 4] = 12; // BGP4-MIB::bgpPeerInTotalMessages
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_RX_MESSAGES);
oid->ids[base_size + 4] = 13; // BGP4-MIB::bgpPeerOutTotalMessages
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_TX_MESSAGES);
oid->ids[base_size + 4] = 14; // BGP4-MIB::bgpPeerLastError
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LAST_ERROR);
oid->ids[base_size + 4] = 15; // BGP4-MIB::bgpPeerFsmEstablishedTransitions
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_FSM_TRANSITIONS);
oid->ids[base_size + 4] = 16; // BGP4-MIB::bgpPeerFsmEstablishedTime
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_FSM_ESTABLISHED_TIME);
oid->ids[base_size + 4] = 17; // BGP4-MIB::bgpPeerConnectionRetryInterval
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_RETRY_INTERVAL);
bt_debug( " 3/4\n");
oid->ids[base_size + 4] = 18; // BGP4-MIB::bgpPeerHoldTime
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_HOLD_TIME);
oid->ids[base_size + 4] = 19; // BGP4-MIB::bgpPeerKeepAlive
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_KEEPALIVE);
oid->ids[base_size + 4] = 20; // BGP4-MIB::bgpPeerHoldTimeConfigured
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_HOLD_TIME_CONFIGURED);
oid->ids[base_size + 4] = 21; // BGP4-MIB::bgpPeerKeepAliveConfigured
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_KEEPALIVE_CONFIGURED);
oid->ids[base_size + 4] = 22; // BGP4-MIB::bgpPeerMinASOriginationInterval
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_ORIGINATION_INTERVAL);
oid->ids[base_size + 4] = 23; // BGP4-MIB::bgpPeerMinRouteAdvertisementInverval
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_MIN_ROUTE_ADVERTISEMENT);
oid->ids[base_size + 4] = 24; // BGP4-MIB::bgpPeerInUpdateElapsedTime
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_IN_UPDATE_ELAPSED_TIME);
bt_debug("testing BGP4-MIB::bgpPeerEntry end\n");
}
static int
t_s_is_oid_empty(void)
{
bt_assert(snmp_is_oid_empty(NULL) == 0);
struct oid *blank = mb_alloc(&root_pool, sizeof(struct oid));
blank->n_subid = 0;
blank->prefix = 0;
blank->include = 0;
bt_assert(snmp_is_oid_empty(blank) == 1);
mb_free(blank); blank = NULL;
struct oid *prefixed = mb_alloc(&root_pool, sizeof(struct oid) + 3 * sizeof(u32));
prefixed->n_subid = 3;
prefixed->prefix = 100;
prefixed->include = 1;
u32 prefixed_arr[] = { ~((u32) 0), 0, 256 };
memcpy(&prefixed->ids, prefixed_arr, sizeof(prefixed_arr) /
sizeof(prefixed_arr[0]));
bt_assert(snmp_is_oid_empty(prefixed) == 0);
mb_free(prefixed); prefixed = NULL;
struct oid *to_prefix = mb_alloc(&root_pool, sizeof(struct oid) + 8 * sizeof(u32));
to_prefix->n_subid = 8;
to_prefix->prefix = 0;
to_prefix->include = 1;
u32 to_prefix_arr[] = {1, 3, 6, 1, 100, ~((u32) 0), 0, 256 };
memcpy(&to_prefix->n_subid, to_prefix_arr, sizeof(to_prefix_arr) /
sizeof(to_prefix_arr[0]));
bt_assert(snmp_is_oid_empty(to_prefix) == 0);
mb_free(to_prefix); to_prefix = NULL;
struct oid *unprefixable = mb_alloc(&root_pool, sizeof(struct oid) + 2 * sizeof(u32));
unprefixable->n_subid = 2;
unprefixable->prefix = 0;
unprefixable->include = 0;
u32 unpref[] = { 65535, 4 };
memcpy(&unprefixable->ids, unpref, sizeof(unpref) / sizeof(unpref[0]));
bt_assert(snmp_is_oid_empty(unprefixable) == 0);
mb_free(unprefixable); unprefixable = NULL;
struct oid *unprefixable2 = mb_alloc(&root_pool, sizeof(struct oid) + 8 * sizeof(u32));
unprefixable2->n_subid = 8;
unprefixable2->prefix = 0;
unprefixable2->include = 1;
u32 unpref2[] = { 1, 3, 6, 2, 1, 2, 15, 6 };
memcpy(&unprefixable2->ids, unpref2, sizeof(unpref2) / sizeof(unpref2[0]));
bt_assert(snmp_is_oid_empty(unprefixable2) == 0);
mb_free(unprefixable2); unprefixable2 = NULL;
return 1;
}
static int
t_s_prefixize(void)
{
//struct oid *nulled = NULL;
struct snmp_proto snmp_proto;
test_fill(&snmp_proto);
//struct oid *result = snmp_prefixize(&snmp_proto, nulled, BYTE_ORD);
//bt_assert(NULL == result);
//result != NULL ? mb_free(result) : NULL;
struct oid *result;
struct oid *blank = mb_allocz(&root_pool, sizeof(struct oid));
/* here the byte order should not matter */
result = snmp_prefixize(&snmp_proto, blank, 1 - BYTE_ORD);
bt_assert(snmp_is_oid_empty(result) == 1);
mb_free(result); result = NULL;
mb_free(blank); blank = NULL;
struct oid *prefixed = mb_alloc(&root_pool, sizeof(struct oid) + 3 * sizeof(u32));
prefixed->n_subid = 3;
prefixed->prefix = 100;
prefixed->include = 1;
u32 prefixed_arr[] = { ~((u32) 0), 0, 256 };
memcpy(&prefixed->ids, prefixed_arr, sizeof(prefixed_arr));
/* struct oid */result = snmp_prefixize(&snmp_proto, prefixed, BYTE_ORD);
bt_assert(memcmp(result, prefixed, snmp_oid_size(prefixed)) == 0);
mb_free(result); result = NULL;
//mb_free(prefixed); prefixed = NULL;
struct oid *to_prefix = mb_alloc(&root_pool, sizeof(struct oid) + 8 * sizeof(u32));
to_prefix->n_subid = 8;
to_prefix->prefix = 0;
to_prefix->include = 1;
u32 to_prefix_arr[] = {1, 3, 6, 1, 100, ~((u32) 0), 0, 256 };
memcpy(to_prefix->ids, to_prefix_arr, sizeof(to_prefix_arr));
result = snmp_prefixize(&snmp_proto, to_prefix, BYTE_ORD);
bt_assert(memcmp(result, prefixed, snmp_oid_size(prefixed)) == 0);
mb_free(result); result = NULL;
mb_free(to_prefix); to_prefix = NULL;
struct oid *unprefixable = mb_alloc(&root_pool, sizeof(struct oid) + 2 * sizeof(u32));
unprefixable->n_subid = 2;
unprefixable->prefix = 0;
unprefixable->include = 0;
u32 unpref[] = { 65535, 4 };
memcpy(&unprefixable->ids, unpref, sizeof(unpref) / sizeof(unpref[0]));
result = snmp_prefixize(&snmp_proto, unprefixable, BYTE_ORD);
bt_assert(result == NULL);
result != NULL ? mb_free(result) : NULL;
struct oid *unprefixable2 = mb_alloc(&root_pool, sizeof(struct oid) + 8 * sizeof(u32));
unprefixable2->n_subid = 8;
unprefixable2->prefix = 0;
unprefixable2->include = 1;
u32 unpref2[] = { 1, 3, 6, 2, 1, 2, 15, 6 };
memcpy(&unprefixable2->ids, unpref2, sizeof(unpref2) / sizeof(unpref2[0]));
result = snmp_prefixize(&snmp_proto, unprefixable2, BYTE_ORD);
bt_assert(result == NULL);
result != NULL ? mb_free(result) : NULL;
return 1;
}
static int
t_oid_compare(void)
{
/* same length, no prefix */
struct oid *l1 = OID_ALLOCATE(5);
{
u32 arr[] = { 1, 2, 3, 4, 5 };
OID_INIT(l1, 5, 0, 1, arr);
}
struct oid *r1 = OID_ALLOCATE(5);
{
u32 arr[] = { 1, 2, 3, 4, 6 };
OID_INIT(r1, 5, 0, 0, arr);
}
bt_assert(snmp_oid_compare(l1, r1) == -1);
bt_assert(snmp_oid_compare(r1, l1) == 1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
/* same results for prefixed oids */
l1->prefix = 1;
r1->prefix = 1;
bt_assert(snmp_oid_compare(l1, r1) == -1);
bt_assert(snmp_oid_compare(r1, l1) == 1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
/* different prefix -- has higher priority */
l1->prefix = 8;
r1->prefix = 4;
bt_assert(snmp_oid_compare(l1, r1) == 1);
bt_assert(snmp_oid_compare(r1, l1) == -1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
mb_free(l1);
mb_free(r1);
/* different length, no prefix */
l1 = OID_ALLOCATE(4);
{
u32 arr[] = { 1, 2, 3, 4 };
OID_INIT(l1, 4, 0, 0, arr);
}
r1 = OID_ALLOCATE(5);
{
u32 arr[] = { 1, 2, 3, 4, 1 };
OID_INIT(r1, 5, 0, 1, arr);
}
bt_assert(snmp_oid_compare(l1, r1) == -1);
bt_assert(snmp_oid_compare(r1, l1) == 1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
/* same results for prefixed oids */
l1->prefix = 3;
r1->prefix = 3;
bt_assert(snmp_oid_compare(l1, r1) == -1);
bt_assert(snmp_oid_compare(r1, l1) == 1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
/* different prefix -- has higher priority */
l1->prefix = 17;
r1->prefix = 14;
bt_assert(snmp_oid_compare(l1, r1) == 1);
bt_assert(snmp_oid_compare(r1, l1) == -1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
mb_free(l1);
mb_free(r1);
/* inverse order different length, no prefix */
l1 = OID_ALLOCATE(4);
{
u32 arr[] = { 1, 2, 3, 5 };
OID_INIT(l1, 4, 0, 0, arr);
}
r1 = OID_ALLOCATE(5);
{
u32 arr[] = { 1, 2, 3, 4, 1 };
OID_INIT(r1, 5, 0, 0, arr);
}
bt_assert(snmp_oid_compare(l1, r1) == 1);
bt_assert(snmp_oid_compare(r1, l1) == -1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
/* same results for prefixed oids */
l1->prefix = 254;
r1->prefix = 254;
bt_assert(snmp_oid_compare(l1, r1) == 1);
bt_assert(snmp_oid_compare(r1, l1) == -1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
/* different prefix -- has higher priority */
l1->prefix = 127;
r1->prefix = 35;
bt_assert(snmp_oid_compare(l1, r1) == 1);
bt_assert(snmp_oid_compare(r1, l1) == -1);
mb_free(l1);
mb_free(r1);
/* ==== MIXED PREFIXED / NON PREFIXED OID compare ==== */
/* same length, mixed */
l1 = OID_ALLOCATE(6); /* OID .1.2.17.3.21.4 */
{
u32 arr[] = { 1, 2, 17, 3, 21, 4 };
OID_INIT(l1, 6, 0, 1, arr);
}
r1 = OID_ALLOCATE(1); /* OID .1.3.6.1.5.3 */
{
u32 arr[] = { 3 };
OID_INIT(r1, 1, 5, 1, arr);
}
bt_assert(snmp_oid_compare(l1, r1) == -1);
bt_assert(snmp_oid_compare(r1, l1) == 1);
bt_assert(snmp_oid_compare(l1, l1) == 0);
bt_assert(snmp_oid_compare(r1, r1) == 0);
mb_free(l1);
mb_free(r1);
return 1;
}
static int
t_s_bgp_state(void)
{
struct oid *oid = mb_alloc(&root_pool, sizeof(struct oid) + 10 * sizeof(u32));
/* oid header */
oid->n_subid = 0;
oid->prefix = 2;
oid->include = 0;
oid->pad = 0;
/* test all states with expected oid length */
bt_debug("testing precise oids\n");
test_oid(oid, 0);
for (int i = 0; i < 10; i++)
oid->ids[i] = (u32) bt_random();
/* if this subid is too high it does not match the test case
* in general test_oid() func
*/
oid->ids[2] = 0;
/* test all states with garbage ip */
bt_debug("testing oids with random ip index\n");
test_oid(oid, 0);
/* test all states with invalid ip */
bt_debug("testing oids with invalid ip index\n");
/* zero the states that overlap */
oid->ids[2] = 0;
oid->ids[3] = 0;
oid->ids[4] = 0;
oid->ids[5] = 0;
oid->ids[6] = 257;
oid->ids[7] = 127;
oid->ids[8] = 0xFFFF;
test_oid(oid, 0);
mb_free(oid);
return 1;
}
int main(int argc, char **argv)
{
bt_init(argc, argv);
bt_bird_init();
bt_test_suite(t_s_bgp_state, "Function snmp_bgp_state()");
bt_test_suite(t_s_is_oid_empty, "Function snmp_is_oid_empty()");
bt_test_suite(t_s_prefixize, "Function snmp_prefixize()");
bt_test_suite(t_oid_compare, "Function snmp_oid_compare()");
return bt_exit_value();
}