blob: d26dc892b7074bca7e2df5633a130883587c1786 [file] [log] [blame]
Harald Welte03c0e562017-12-09 02:55:12 +01001module Osmocom_CTRL_Functions {
Harald Welte35bb7162018-01-03 21:07:52 +01002
3/* Definition of helper functions for the Osmocom CTRL interface.
4 *
5 * As opposed to many other parts of the Osmocom TTCN-3 code base, this module
6 * implements blocking functions, instead of asynchronous functions. The
7 * rationale for this is simple: One normally wants to inquire a value or set
8 * a value and not continue the main program until that operation is complete.
Pau Espin Pedrol76ba5412019-06-10 11:00:33 +02009 *
Harald Welte35bb7162018-01-03 21:07:52 +010010 * CTRL is a machine-type protocol on how external programs can interact with
11 * an Osmocom program in a structured way. It is intended for programmatic
12 * access (by other software), as opposed to the VTY interface intended for
13 * human consumption.
14 *
15 * (C) 2017 by Harald Welte <laforge@gnumonks.org>
16 * All rights reserved.
17 *
18 * Released under the terms of GNU General Public License, Version 2 or
19 * (at your option) any later version.
20 */
21
22
Harald Welte03c0e562017-12-09 02:55:12 +010023 import from Osmocom_CTRL_Types all;
24 import from IPA_Emulation all;
25
26 private function f_gen_rand_id() return CtrlId {
27 return int2str(float2int(rnd()*999999999.0));
28 }
29
30 /* perform a given GET Operation */
31 function f_ctrl_get(IPA_CTRL_PT pt, CtrlVariable variable) return CtrlValue {
32 timer T := 2.0;
33 var CtrlMessage rx;
34 var CtrlId id := f_gen_rand_id();
35 pt.send(ts_CtrlMsgGet(id, variable));
36 T.start;
37 alt {
Harald Weltefdfa0462017-12-10 18:09:40 +010038 [] pt.receive(tr_CtrlMsgGetRepl(id, variable)) -> value rx {
39 }
Harald Welte95a47812017-12-09 14:19:03 +010040 [] pt.receive(tr_CtrlMsgTrap) { repeat; }
Harald Welte03c0e562017-12-09 02:55:12 +010041 [] pt.receive(tr_CtrlMsgError) -> value rx {
42 setverdict(fail, "Error in CTRL GET ", variable, ": ", rx.err.reason);
Daniel Willmanne4ff5372018-07-05 17:35:03 +020043 mtc.stop;
Harald Welte03c0e562017-12-09 02:55:12 +010044 }
45 [] T.timeout {
46 setverdict(fail, "Timeout waiting for CTRL GET REPLY ", variable);
Daniel Willmanne4ff5372018-07-05 17:35:03 +020047 mtc.stop;
Harald Welte03c0e562017-12-09 02:55:12 +010048 }
49 }
50 return rx.resp.val;
51 }
52
53 /* perform a given SET Operation */
54 function f_ctrl_set(IPA_CTRL_PT pt, CtrlVariable variable, CtrlValue val) {
55 timer T := 2.0;
56 var CtrlMessage rx;
57 var CtrlId id := f_gen_rand_id();
58 pt.send(ts_CtrlMsgSet(id, variable, val));
59 T.start;
60 alt {
61 [] pt.receive(tr_CtrlMsgSetRepl(id, variable, val)) { }
Harald Welte95a47812017-12-09 14:19:03 +010062 [] pt.receive(tr_CtrlMsgTrap) { repeat; }
Harald Welte03c0e562017-12-09 02:55:12 +010063 [] pt.receive(tr_CtrlMsgError) -> value rx {
64 setverdict(fail, "Error in CTRL GET ", variable, ": ", rx.err.reason);
Daniel Willmanne4ff5372018-07-05 17:35:03 +020065 mtc.stop;
Harald Welte03c0e562017-12-09 02:55:12 +010066 }
67 [] T.timeout {
68 setverdict(fail, "Timeout waiting for CTRL SET REPLY ", variable);
Daniel Willmanne4ff5372018-07-05 17:35:03 +020069 mtc.stop;
Harald Welte03c0e562017-12-09 02:55:12 +010070 }
71 }
72 }
73
Pau Espin Pedrol579cd7a2019-06-10 21:01:30 +020074 /* send a TRAP */
75 function f_ctrl_trap(IPA_CTRL_PT pt, CtrlVariable variable, CtrlValue val) {
76 pt.send(ts_CtrlMsgTrap(variable, val));
77 }
78
Harald Welte03c0e562017-12-09 02:55:12 +010079 /* Expect a matching TRAP */
80 function f_ctrl_exp_trap(IPA_CTRL_PT pt, template CtrlVariable variable,
Pau Espin Pedrol579cd7a2019-06-10 21:01:30 +020081 template CtrlValue val := ?, float timeout_val := 2.0)
82 return CtrlValue {
83 timer T := timeout_val;
Harald Welte03c0e562017-12-09 02:55:12 +010084 var CtrlMessage rx;
85 T.start;
86 alt {
Harald Weltefdfa0462017-12-10 18:09:40 +010087 [] pt.receive(tr_CtrlMsgTrap(variable, val)) -> value rx {
88 }
Harald Welte03c0e562017-12-09 02:55:12 +010089 [] T.timeout {
90 setverdict(fail, "Timeout waiting for TRAP ", variable);
Daniel Willmanne4ff5372018-07-05 17:35:03 +020091 mtc.stop;
Harald Welte03c0e562017-12-09 02:55:12 +010092 }
93 }
94 return rx.trap.val;
95 }
Harald Welte852a3842017-12-09 14:19:36 +010096
Pau Espin Pedrol579cd7a2019-06-10 21:01:30 +020097 /* Expect a matching SET, optionally answer */
98 function f_ctrl_exp_set(IPA_CTRL_PT pt, template CtrlVariable variable,
99 template CtrlValue val := ?,
100 template (omit) CtrlValue rsp := omit,
101 float timeout_val := 2.0)
102 return CtrlValue {
103 timer T := timeout_val;
104 var CtrlMessage rx;
105 T.start;
106 alt {
107 [] pt.receive(tr_CtrlMsgSet(?, variable, val)) -> value rx {
108 if (ispresent(rsp)) {
109 pt.send(ts_CtrlMsgSetRepl(rx.cmd.id, valueof(variable), valueof(rsp)));
110 }
111 }
112 [] T.timeout {
113 setverdict(fail, "Timeout waiting for SET ", variable);
114 mtc.stop;
115 }
116 }
117 return rx.cmd.val;
118 }
119
Harald Welte852a3842017-12-09 14:19:36 +0100120 /* Expect a matching GET result */
121 function f_ctrl_get_exp(IPA_CTRL_PT pt, CtrlVariable variable, template CtrlValue exp) {
122 var charstring ctrl_resp;
123 ctrl_resp := f_ctrl_get(pt, variable);
124 if (not match(ctrl_resp, exp)) {
125 setverdict(fail, "Unexpected " & variable & ":" & ctrl_resp);
Daniel Willmanne4ff5372018-07-05 17:35:03 +0200126 mtc.stop;
Harald Welte852a3842017-12-09 14:19:36 +0100127 }
128 }
129
Harald Welte3ec493f2017-12-09 16:25:29 +0100130 template charstring ts_ctrl_ratectr(CtrlVariable grp, integer instance, CtrlVariable name,
131 CtrlVariable kind := "abs") :=
132 "rate_ctr." & kind & "." & grp & "." & int2str(instance) & "." & name;
133
134 function f_ctrl_get_ratectr_abs(IPA_CTRL_PT pt, CtrlVariable grp, integer instance,
135 CtrlVariable name) return integer {
136 return str2int(f_ctrl_get(pt, valueof(ts_ctrl_ratectr(grp, instance, name))));
137 }
138
139 function f_ctrl_get_exp_ratectr_abs(IPA_CTRL_PT pt, CtrlVariable grp, integer instance,
140 CtrlVariable name, template integer exp) {
141 var charstring ctrl_resp;
142 var CtrlVariable variable := valueof(ts_ctrl_ratectr(grp, instance, name));
143 ctrl_resp := f_ctrl_get(pt, variable);
144 if (not match(str2int(ctrl_resp), exp)) {
145 setverdict(fail, variable & " value " & ctrl_resp & " didn't match ", exp);
Daniel Willmanne4ff5372018-07-05 17:35:03 +0200146 mtc.stop;
Harald Welte3ec493f2017-12-09 16:25:29 +0100147 }
148 }
149
Harald Welte852a3842017-12-09 14:19:36 +0100150
Neels Hofmeyr74180592020-06-20 21:40:52 +0200151 /* --- Retrieve and verify rate counter values in bulk ---
152 *
153 * BSC_Tests.ttcn shows a nice way to conveniently shorten the code needed to use these functions, see
154 * f_ctrs_msc_init() and f_ctrs_msc_expect().
155 *
156 * Here also a full usage example:
157 *
158 * const CounterNameVals my_counternames := {
159 * { "mscpool:subscr:new", 0 },
160 * { "mscpool:subscr:known", 0 },
161 * { "mscpool:subscr:attach_lost", 0 },
162 * };
163 *
164 * var CounterNameValsList my_counters := f_counter_name_vals_get_n(instance_name := "msc", instance_count := 3,
165 * counternames := my_counternames);
166 *
167 * // run some tests that increment rate counters in the program,
168 * // and increment expected counters accordingly:
Neels Hofmeyr1393d022020-08-29 01:17:33 +0000169 * f_counter_name_vals_list_add(my_counters, instance_nr := 1, "mscpool:subscr:new", 7);
170 * f_counter_name_vals_list_add(my_counters, instance_nr := 2, "mscpool:subscr:attach_lost", 3);
Neels Hofmeyr74180592020-06-20 21:40:52 +0200171 *
172 * // verify that the program reflects the expected counters:
173 * f_counter_name_vals_expect_n(instance_name := "msc", my_counters);
174 *
175 * // run some more tests...
Neels Hofmeyr1393d022020-08-29 01:17:33 +0000176 * f_counter_name_vals_list_add(my_counters, instance_nr := 0, "mscpool:subscr:known");
Neels Hofmeyr74180592020-06-20 21:40:52 +0200177 * // and verify again
178 * f_counter_name_vals_expect_n(instance_name := "msc", my_counters);
179 */
180
181 /* One counter value, e.g. { "name", 23 } */
182 type record CounterNameVal {
183 charstring name,
184 integer val
185 }
186
187 /* List of one instance's counters,
188 * e.g. { {"foo",23}, {"bar",42} }
189 */
190 type record of CounterNameVal CounterNameVals;
191
192 /* List of numerous instances' counters,
193 * e.g. { { {"foo",23}, {"bar",42} },
194 * { {"foo",23}, {"bar",42} } }
195 */
196 type record of CounterNameVals CounterNameValsList;
197
198 /* Retrieve one instance's rate counter values of the given names. */
199 function f_counter_name_vals_get(IPA_CTRL_PT pt, charstring instance_name, integer instance_nr,
200 CounterNameVals counternames)
201 return CounterNameVals {
202 var CounterNameVals vals;
203 for (var integer i := 0; i < lengthof(counternames); i := i + 1) {
204 vals[i] := {
205 name := counternames[i].name,
206 val := f_ctrl_get_ratectr_abs(pt, instance_name, instance_nr, counternames[i].name)
207 };
208 }
209 return vals;
210 }
211
212 /* Retrieve the first N instances' rate counter values of the given names */
213 function f_counter_name_vals_get_n(IPA_CTRL_PT pt, charstring instance_name := "msc",
214 integer instance_count, CounterNameVals counternames)
215 return CounterNameValsList {
216 var CounterNameValsList valslist;
217 for (var integer instance_nr := 0; instance_nr < instance_count; instance_nr := instance_nr + 1) {
218 valslist[instance_nr] := f_counter_name_vals_get(pt, instance_name, instance_nr, counternames);
219 }
220 log("retrieved rate counters: ", instance_name, ": ", valslist);
221 return valslist;
222 }
223
224 /* In a list of one instance's counters, increment a specifically named counter. */
Neels Hofmeyr9656e922020-06-30 01:27:01 +0200225 function f_counter_name_vals_add(inout CounterNameVals vals, charstring countername, integer val := 1)
226 {
Neels Hofmeyr74180592020-06-20 21:40:52 +0200227 for (var integer i := 0; i < lengthof(vals); i := i + 1) {
228 if (vals[i].name == countername) {
229 vals[i].val := vals[i].val + val;
Neels Hofmeyr9656e922020-06-30 01:27:01 +0200230 return;
Neels Hofmeyr74180592020-06-20 21:40:52 +0200231 }
232 }
233 /* name not found, append */
234 vals[lengthof(vals)] := {
235 name := countername,
236 val := val
237 }
Neels Hofmeyr74180592020-06-20 21:40:52 +0200238 }
239
240 /* In a list of several instances' counters, increment a specific instance's specifically named counter. */
Neels Hofmeyr9656e922020-06-30 01:27:01 +0200241 function f_counter_name_vals_list_add(inout CounterNameValsList vals, integer instance_nr,
Neels Hofmeyr74180592020-06-20 21:40:52 +0200242 charstring countername, integer val := 1)
Neels Hofmeyr9656e922020-06-30 01:27:01 +0200243 {
244 f_counter_name_vals_add(vals[instance_nr], countername, val);
Neels Hofmeyr74180592020-06-20 21:40:52 +0200245 }
246
247 /* For a specific instance, call f_counter_name_vals_get() and compare with expected counter values.
248 * Set the test verdict accordingly. */
249 function f_counter_name_vals_expect(IPA_CTRL_PT pt, charstring instance_name, integer instance_nr,
250 CounterNameVals vals) {
251 var CounterNameVals now := f_counter_name_vals_get(pt, instance_name, instance_nr, vals);
252 for (var integer i := 0; i < lengthof(vals); i := i + 1) {
253 if (now[i].name != vals[i].name) {
254 setverdict(fail, "Internal error");
255 }
256 if (now[i].val != vals[i].val) {
257 setverdict(fail, "Rate counter mismatch: ", instance_name, " ", instance_nr,
258 " ", vals[i].name, " is at ", now[i].val, " but expected ", vals[i].val);
259 }
260 }
261 setverdict(pass);
262 }
263
264 /* For N instances, call f_counter_name_vals_get() and compare with expected counter values.
265 * Set the test verdict accordingly. The number of instances is given by lengthof(valslist). */
266 function f_counter_name_vals_expect_n(IPA_CTRL_PT pt, charstring instance_name, CounterNameValsList valslist) {
267 for (var integer instance_nr := 0; instance_nr < lengthof(valslist); instance_nr := instance_nr + 1) {
268 f_counter_name_vals_expect(pt, instance_name, instance_nr, valslist[instance_nr]);
269 }
270 }
271
Harald Welte03c0e562017-12-09 02:55:12 +0100272}