blob: d45d52ebb2c53a982fe107db10ec77ffd4a19949 [file] [log] [blame]
Brian Waters13d96012017-12-08 16:53:31 -06001/*********************************************************************************************************
2* Software License Agreement (BSD License) *
3* Author: Sebastien Decugis <sdecugis@freediameter.net> *
4* *
5* Copyright (c) 2013, WIDE Project and NICT *
6* All rights reserved. *
7* *
8* Redistribution and use of this software in source and binary forms, with or without modification, are *
9* permitted provided that the following conditions are met: *
10* *
11* * Redistributions of source code must retain the above *
12* copyright notice, this list of conditions and the *
13* following disclaimer. *
14* *
15* * Redistributions in binary form must reproduce the above *
16* copyright notice, this list of conditions and the *
17* following disclaimer in the documentation and/or other *
18* materials provided with the distribution. *
19* *
20* * Neither the name of the WIDE Project or NICT nor the *
21* names of its contributors may be used to endorse or *
22* promote products derived from this software without *
23* specific prior written permission of WIDE Project and *
24* NICT. *
25* *
26* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED *
27* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A *
28* PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR *
29* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT *
30* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS *
31* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR *
32* TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF *
33* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *
34*********************************************************************************************************/
35
36#include "tests.h"
37
38/* Test for the dict_iterate_rules function */
39int iter_test(void * data, struct dict_rule_data * rule)
40{
41 struct dict_avp_data avpdata;
42 (*(int *)data)++;
43
44 CHECK( 0, fd_dict_getval ( rule->rule_avp, &avpdata ) );
45 TRACE_DEBUG(FULL, "rule #%d: avp '%s'", *(int *)data, avpdata.avp_name);
46 return 0;
47}
48
49/* Main test routine */
50int main(int argc, char *argv[])
51{
52 /* First, initialize the daemon modules */
53 INIT_FD();
54
55 /* Test creating and searching all types of objects */
56 {
57 struct dict_object * obj1 = NULL;
58 struct dict_object * obj2 = NULL;
59 struct dict_object * obj3 = NULL;
60
61 vendor_id_t vendor_id = 735671;
62 struct dict_vendor_data vendor1_data = { 735671, "Vendor test 1" };
63 struct dict_vendor_data vendor2_data = { 735672, "Vendor test 2" };
64 struct dict_application_data app1_data = { 735674, "Application test 1" };
65
66
67 /* Create two vendors */
68 CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_VENDOR, &vendor1_data , NULL, &obj1 ) );
69 CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_VENDOR, &vendor2_data , NULL, NULL ) );
70
71 /* Check we always retrieve the correct vendor object */
72 CHECK( 0, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_ID, &vendor_id, &obj2, ENOENT ) );
73 CHECK( obj1, obj2);
74 CHECK( 0, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_NAME, "Vendor test 1", &obj2, ENOENT ) );
75 CHECK( obj1, obj2);
76
77 /* Check the error conditions */
78 CHECK( 0, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_ID, &vendor_id, NULL, ENOENT ) );
79
80 vendor_id = 735673; /* Not defined */
81 CHECK( ENOENT, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_ID, &vendor_id, NULL, ENOENT ) );
82 CHECK( ENOENT, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_NAME, "Vendor test 3", NULL, ENOENT ) );
83 CHECK( ENOENT, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_ID, &vendor_id, &obj2, ENOENT ) );
84 CHECK( ENOENT, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_NAME, "Vendor test 3", &obj2, ENOENT ) );
85 CHECK( ENOTSUP, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_NAME, "Vendor test 3", &obj2, ENOTSUP ) );
86
87 /* Check the get_* functions */
88 CHECK( 0, fd_dict_getval ( obj1, &vendor1_data ) );
89 CHECK( 735671, vendor1_data.vendor_id );
90 CHECK( 0, strcmp(vendor1_data.vendor_name, "Vendor test 1") );
91 /* error conditions */
92 CHECK( EINVAL, fd_dict_getval ( (struct dict_object *)"not an object", &vendor1_data ) );
93
94 /* Create the application with vendor1 as parent */
95 CHECK( EINVAL, fd_dict_new ( fd_g_config->cnf_dict, DICT_APPLICATION, &app1_data , (struct dict_object *)"bad object", &obj2 ) );
96 CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_APPLICATION, &app1_data , obj1, &obj2 ) );
97
98 CHECK( 0, fd_dict_search ( fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_OF_APPLICATION, obj2, &obj3, ENOENT ) );
99 CHECK( obj1, obj3);
100
101 /* Creating and searching the other objects is already done in dictionary initialization */
102 }
103
104 /* Test creation of the "Example-AVP" grouped AVP from the RFC */
105 {
106 int nbr = 0;
107 struct dict_object * origin_host_avp = NULL;
108 struct dict_object * session_id_avp = NULL;
109 struct dict_object * example_avp_avp = NULL;
110 struct dict_rule_data rule_data = { NULL, RULE_REQUIRED, -1, -1 };
111 struct dict_avp_data example_avp_data = { 999999, 0, "Example-AVP", AVP_FLAG_VENDOR , 0, AVP_TYPE_GROUPED };
112
113 CHECK( 0, fd_dict_search ( fd_g_config->cnf_dict, DICT_AVP, AVP_BY_NAME, "Origin-Host", &origin_host_avp, ENOENT ) );
114 CHECK( 0, fd_dict_search ( fd_g_config->cnf_dict, DICT_AVP, AVP_BY_NAME, "Session-Id", &session_id_avp, ENOENT ) );
115
116 CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_AVP, &example_avp_data , NULL, &example_avp_avp ) );
117
118 rule_data.rule_avp = origin_host_avp;
119 rule_data.rule_min = 1;
120 rule_data.rule_max = 1;
121 CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_RULE, &rule_data, example_avp_avp, NULL ) );
122
123 rule_data.rule_avp = session_id_avp;
124 rule_data.rule_min = 1;
125 rule_data.rule_max = -1;
126 CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_RULE, &rule_data, example_avp_avp, NULL ) );
127
128 CHECK( 0, fd_dict_iterate_rules ( example_avp_avp, &nbr, iter_test) );
129 CHECK( 2, nbr );
130 }
131
132 /* Test list function */
133 {
134 struct fd_list * li = NULL;
135 struct fd_list * sentinel = NULL;
136 enum dict_object_type type;
137 struct dict_object * defvnd=NULL;
138 vendor_id_t vid = 0;
139 int first = 1;
140
141 CHECK( 0, fd_dict_getlistof(VENDOR_BY_ID, fd_g_config->cnf_dict, &sentinel));
142
143 for (li = sentinel; (li != sentinel) || (first != 0); li = li->next) {
144 first = 0;
145 CHECK(0, fd_dict_gettype(li->o, &type));
146 CHECK(DICT_VENDOR, type);
147#if 0
148 struct dict_vendor_data data;
149 CHECK( 0, fd_dict_getval(li->o, &data) );
150 printf("%d : %s\n", data.vendor_id, data.vendor_name);
151#endif
152 }
153
154 CHECK( 0, fd_dict_search(fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_ID, &vid, &defvnd, ENOENT) );
155
156 CHECK( 0, fd_dict_getlistof(AVP_BY_NAME, defvnd, &sentinel));
157 for (li = sentinel->next; li != sentinel; li = li->next) {
158 CHECK(0, fd_dict_gettype(li->o, &type));
159 CHECK(DICT_AVP, type);
160#if 0
161 struct dict_avp_data data;
162 CHECK( 0, fd_dict_getval(li->o, &data) );
163 printf("%d : %s\n", data.avp_code, data.avp_name);
164#endif
165 }
166 }
167
168 /* Test delete function */
169 {
170 struct fd_list * li = NULL;
171 struct fd_list * sentinel = NULL;
172 struct dict_object * obj=NULL;
173 vendor_id_t vid = 0;
174 int count = 0, cntbkp;
175
176 CHECK( 0, fd_dict_search(fd_g_config->cnf_dict, DICT_VENDOR, VENDOR_BY_ID, &vid, &obj, ENOENT) );
177
178 CHECK( EINVAL, fd_dict_delete(obj) );
179
180
181 CHECK( 0, fd_dict_getlistof(AVP_BY_NAME, obj, &sentinel));
182 obj = NULL;
183
184 for (li = sentinel->next; li != sentinel; li = li->next) {
185 struct dict_avp_data data;
186 CHECK( 0, fd_dict_getval(li->o, &data) );
187 count++;
188 if (data.avp_basetype != AVP_TYPE_GROUPED)
189 obj = li->o;
190 }
191
192 CHECK(1, obj ? 1 : 0 );
193#if 1
194 fd_log_debug("%s", fd_dict_dump_object(FD_DUMP_TEST_PARAMS, obj));
195#endif
196 CHECK( 0, fd_dict_delete(obj) );
197 cntbkp = count;
198 count = 0;
199 for (li = sentinel->next; li != sentinel; li = li->next) {
200 count++;
201 }
202 CHECK( 1, cntbkp - count );
203
204 }
205
206 LOG_D( "Dictionary at the end of %s: %s", __FILE__, fd_dict_dump(FD_DUMP_TEST_PARAMS, fd_g_config->cnf_dict) ?: "error");
207
208 /* That's all for the tests yet */
209 PASSTEST();
210}
211