mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-24 09:51:54 +00:00
460 lines
13 KiB
C
460 lines
13 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 "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[0] = 1;
|
|
oid->ids[1] = 15; // BGP4-MIB::bgp
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_BGP);
|
|
|
|
oid->n_subid = base_size + 3;
|
|
oid->ids[2] = 1; // BGP4-MIB::bgpVersion
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_VERSION);
|
|
|
|
oid->ids[2] = 2; // BGP4-MIB::bgpLocalAs
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LOCAL_AS);
|
|
|
|
oid->ids[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;
|
|
oid->ids[2] = 3;
|
|
oid->ids[3] = 1; // BGP4-MIB::bgpPeerEntry
|
|
snmp_oid_dump(oid);
|
|
SNMP_EXPECTED(snmp_bgp_state(oid), BGP_INTERNAL_PEER_ENTRY);
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_PEER_ENTRY);
|
|
|
|
oid->n_subid = base_size + 5;
|
|
oid->ids[2] = 3;
|
|
oid->ids[3] = 1;
|
|
oid->ids[4] = 1; // BGP4-MIB::bgpPeerIdentifier
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_IDENTIFIER);
|
|
|
|
oid->ids[4] = 2; // BGP4-MIB::bgpPeerState
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_STATE);
|
|
|
|
oid->ids[4] = 3; // BGP4-MIB::bgpPeerAdminStatus
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_ADMIN_STATUS);
|
|
|
|
oid->ids[4] = 4; // BGP4-MIB::bgpPeerNegotiatedVersion
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_NEGOTIATED_VERSION);
|
|
|
|
oid->ids[4] = 5; // BGP4-MIB::bgpPeerLocalAddr
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LOCAL_ADDR);
|
|
|
|
oid->ids[4] = 6; // BGP4-MIB::bgpPeerLocalPort
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LOCAL_PORT);
|
|
|
|
oid->ids[4] = 7; // BGP4-MIB::bgpPeerRemoteAddr
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_REMOTE_ADDR);
|
|
|
|
oid->ids[4] = 8; // BGP4-MIB::bgpPeerRemotePort
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_REMOTE_PORT);
|
|
|
|
oid->ids[4] = 9; // BGP4-MIB::bgpPeerRemoteAs
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_REMOTE_AS);
|
|
|
|
oid->ids[4] = 10; // BGP4-MIB::bgpPeerInUpdates
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_RX_UPDATES);
|
|
|
|
oid->ids[4] = 11; // BGP4-MIB::bgpPeerOutUpdates
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_TX_UPDATES);
|
|
|
|
oid->ids[4] = 12; // BGP4-MIB::bgpPeerInTotalMessages
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_RX_MESSAGES);
|
|
|
|
oid->ids[4] = 13; // BGP4-MIB::bgpPeerOutTotalMessages
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_TX_MESSAGES);
|
|
|
|
oid->ids[4] = 14; // BGP4-MIB::bgpPeerLastError
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_LAST_ERROR);
|
|
|
|
oid->ids[4] = 15; // BGP4-MIB::bgpPeerFsmEstablishedTransitions
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_FSM_TRANSITIONS);
|
|
|
|
oid->ids[4] = 16; // BGP4-MIB::bgpPeerFsmEstablishedTime
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_FSM_ESTABLISHED_TIME);
|
|
|
|
oid->ids[4] = 17; // BGP4-MIB::bgpPeerConnectionRetryInterval
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_RETRY_INTERVAL);
|
|
|
|
oid->ids[4] = 18; // BGP4-MIB::bgpPeerHoldTime
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_HOLD_TIME);
|
|
|
|
oid->ids[4] = 19; // BGP4-MIB::bgpPeerKeepAlive
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_KEEPALIVE);
|
|
|
|
oid->ids[4] = 20; // BGP4-MIB::bgpPeerHoldTimeConfigured
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_HOLD_TIME_CONFIGURED);
|
|
|
|
oid->ids[4] = 21; // BGP4-MIB::bgpPeerKeepAliveConfigured
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_KEEPALIVE_CONFIGURED);
|
|
|
|
oid->ids[4] = 22; // BGP4-MIB::bgpPeerMinASOriginationInterval
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_ORIGINATION_INTERVAL);
|
|
|
|
oid->ids[4] = 23; // BGP4-MIB::bgpPeerMinRouteAdvertisementInverval
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_MIN_ROUTE_ADVERTISEMENT);
|
|
|
|
oid->ids[4] = 24; // BGP4-MIB::bgpPeerInUpdateElapsedTime
|
|
bt_assert(snmp_bgp_state(oid) == BGP_INTERNAL_IN_UPDATE_ELAPSED_TIME);
|
|
}
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
t_s_prefixize(void)
|
|
{
|
|
struct oid *nulled = NULL;
|
|
|
|
struct snmp_proto snmp_proto;
|
|
|
|
test_fill(&snmp_proto);
|
|
|
|
bt_debug("before seg fault\n");
|
|
|
|
if (snmp_is_oid_empty(NULL))
|
|
bt_debug("null oid is empty");
|
|
else
|
|
bt_debug("null oid is not empty");
|
|
|
|
bt_debug("main cause\n");
|
|
struct oid *tmp = snmp_prefixize(&snmp_proto, nulled, BYTE_ORD);
|
|
bt_debug("after snmp_prefixize() call\n");
|
|
bt_assert( NULL == tmp );
|
|
|
|
bt_debug("after assert\n");
|
|
struct oid *blank = mb_allocz(&root_pool, sizeof(struct oid));
|
|
|
|
/* here the byte order should not matter */
|
|
bt_assert(snmp_is_oid_empty(snmp_prefixize(&snmp_proto, blank, 1 - BYTE_ORD)) == 1);
|
|
|
|
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(memcmp(snmp_prefixize(&snmp_proto, prefixed, BYTE_ORD), prefixed, snmp_oid_size(prefixed)) == 0);
|
|
|
|
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(memcmp(snmp_prefixize(&snmp_proto, to_prefix, BYTE_ORD), prefixed, snmp_oid_size(prefixed)) == 0);
|
|
|
|
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_prefixize(&snmp_proto, unprefixable, BYTE_ORD) == 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_prefixize(&snmp_proto, unprefixable2, BYTE_ORD) == 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);
|
|
|
|
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(l1, 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);
|
|
|
|
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);
|
|
|
|
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(l1, 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, 4);
|
|
|
|
/* 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, 4);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
bt_init(argc, argv);
|
|
|
|
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();
|
|
}
|