blob: a5503a71843e3700ff0986230b80067944de76b1 [file] [log] [blame]
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +02001module HSS_Tests {
2
3import from General_Types all;
4import from Osmocom_Types all;
5import from Native_Functions all;
6import from Misc_Helpers all;
7
8import from DIAMETER_Types all;
9import from DIAMETER_Templates all;
Pau Espin Pedrol117a94f2023-12-21 16:10:12 +010010import from DIAMETER_ts29_272_Templates all;
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +020011import from DIAMETER_Emulation all;
12
13type record of hexstring SubscriberConfigs;
14
15modulepar {
16 charstring mp_hss_hostname := "127.0.0.4";
17 integer mp_hss_port := 3868;
18 charstring mp_diam_local_hostname := "127.0.0.1";
19 integer mp_diam_local_port := 3868;
20 charstring mp_diam_orig_realm := "localdomain";
21 charstring mp_diam_orig_host := "mme.localdomain";
22 charstring mp_diam_dest_realm := "localdomain";
23 charstring mp_diam_dest_host := "hss.localdomain";
24 SubscriberConfigs subscribers := {
25 /* Existing subscriber, ULA returns SERVICE_GRANTED */
Pau Espin Pedrold34720f2023-10-11 17:49:39 +020026 '001010000000000'H,
27 '001010000000001'H
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +020028 };
29}
30
31/* main component, we typically have one per testcase */
32type component MTC_CT {
33
34 /* emulated MME/SGSN */
35 var DIAMETER_Emulation_CT vc_S6a;
36 port DIAMETER_PT S6a_UNIT;
37 port DIAMETEREM_PROC_PT S6a_PROC;
38 /* global test case guard timer (actual timeout value is set in f_init()) */
39 timer T_guard;
40}
41
42/* global altstep for global guard timer; */
43altstep as_Tguard() runs on MTC_CT {
44 [] T_guard.timeout {
45 setverdict(fail, "Timeout of T_guard");
46 mtc.stop;
47 }
48}
49
50type component DIAMETER_ConnHdlr_CT extends DIAMETER_ConnHdlr {
51 port DIAMETER_Conn_PT DIAMETER_CLIENT;
52 port DIAMETEREM_PROC_PT DIAMETER_PROC_CLIENT;
53}
54
55function f_diam_connhldr_ct_main(hexstring imsi) runs on DIAMETER_ConnHdlr_CT {
56 var DIAMETER_ConnHdlr vc_conn_unused;
57 var PDU_DIAMETER msg;
58 var UINT32 ete_id;
59
60 f_diameter_expect_imsi(imsi);
61
62 while (true) {
63 alt {
64 [] DIAMETER_CLIENT.receive(PDU_DIAMETER:?) -> value msg {
65 DIAMETER.send(msg);
66 }
67 [] DIAMETER.receive(PDU_DIAMETER:?) -> value msg {
68 DIAMETER_CLIENT.send(msg);
69 }
70 [] DIAMETER_PROC_CLIENT.getcall(DIAMETEREM_register_eteid:{?,?}) -> param(ete_id, vc_conn_unused) {
71 DIAMETER_PROC.call(DIAMETEREM_register_eteid:{ete_id, self}) {
72 [] DIAMETER_PROC.getreply(DIAMETEREM_register_eteid:{?,?}) {};
73 }
74 DIAMETER_PROC_CLIENT.reply(DIAMETEREM_register_eteid:{ete_id, vc_conn_unused});
75 }
76 }
77 }
78}
79
80/* per-session component; we typically have 1..N per testcase */
81type component Cli_Session_CT {
82 var SessionPars g_pars;
83
84 port DIAMETER_Conn_PT S6a;
85 port DIAMETEREM_PROC_PT S6a_PROC;
86}
87function f_diam_connhldr_expect_eteid(UINT32 ete_id) runs on Cli_Session_CT {
88 S6a_PROC.call(DIAMETEREM_register_eteid:{ete_id, null}) {
89 [] S6a_PROC.getreply(DIAMETEREM_register_eteid:{?,?}) {};
90 }
91}
92
93/* configuration data for a given Session */
94type record SessionPars {
95 hexstring imsi,
96 uint32_t s6a_next_hbh_id,
97 uint32_t s6a_next_ete_id
98}
99
100template (value) SessionPars
101t_SessionPars(hexstring imsi, uint32_t s6a_next_hbh_id := 1000, uint32_t s6a_next_ete_id := 22220) := {
102 imsi := imsi,
103 s6a_next_hbh_id := s6a_next_hbh_id,
104 s6a_next_ete_id := s6a_next_ete_id
105}
106
107type function void_fn() runs on Cli_Session_CT;
108
109friend function DiameterForwardUnitdataCallback(PDU_DIAMETER msg)
110runs on DIAMETER_Emulation_CT return template PDU_DIAMETER {
111 DIAMETER_UNIT.send(msg);
112 return omit;
113}
114
115friend function f_init_diameter(charstring id) runs on MTC_CT {
116 var DIAMETEROps ops := {
117 create_cb := refers(DIAMETER_Emulation.ExpectedCreateCallback),
118 unitdata_cb := refers(DiameterForwardUnitdataCallback),
119 raw := false /* handler mode (IMSI based routing) */
120 };
121 var DIAMETER_conn_parameters pars;
122
123 /* S6a setup: */
124 pars := {
125 remote_ip := mp_hss_hostname,
126 remote_sctp_port := mp_hss_port,
127 local_ip := mp_diam_local_hostname,
128 local_sctp_port := mp_diam_local_port,
129 origin_host := mp_diam_orig_host,
130 origin_realm := mp_diam_orig_realm,
131 auth_app_id := omit,
132 vendor_app_id := c_DIAMETER_3GPP_S6_AID
133 };
134 vc_S6a := DIAMETER_Emulation_CT.create(id);
135 map(vc_S6a:DIAMETER, system:DIAMETER_CODEC_PT);
136 connect(vc_S6a:DIAMETER_UNIT, self:S6a_UNIT);
137 connect(vc_S6a:DIAMETER_PROC, self:S6a_PROC);
138 vc_S6a.start(DIAMETER_Emulation.main(ops, pars, id));
139
140 f_diameter_wait_capability(S6a_UNIT);
141 /* Give some time for our emulation to get out of SUSPECT list of SUT (3 watchdong ping-pongs):
142 * RFC6733 sec 5.1
143 * RFC3539 sec 3.4.1 [5]
144 * https://github.com/freeDiameter/freeDiameter/blob/master/libfdcore/p_psm.c#L49
145 */
146 f_sleep(1.0);
147}
148
149private function f_init(float guard_timeout := 60.0) runs on MTC_CT {
150 T_guard.start(guard_timeout);
151 activate(as_Tguard());
152 f_init_diameter(testcasename());
153}
154
155function f_start_handler(void_fn fn, template (omit) SessionPars pars_tmpl := omit)
156runs on MTC_CT return Cli_Session_CT {
157 var charstring id := testcasename();
158 var DIAMETER_ConnHdlr_CT vc_conn_s6a;
159 var Cli_Session_CT vc_conn;
160 var SessionPars pars;
161
162 if (isvalue(pars_tmpl)) {
163 pars := valueof(pars_tmpl);
164 } else {
165 /*TODO: set default values */
166 }
167
168 vc_conn := Cli_Session_CT.create(id);
169
170 vc_conn_s6a := DIAMETER_ConnHdlr_CT.create(id);
171 connect(vc_conn_s6a:DIAMETER, vc_S6a:DIAMETER_CLIENT);
172 connect(vc_conn_s6a:DIAMETER_PROC, vc_S6a:DIAMETER_PROC);
173 connect(vc_conn:S6a, vc_conn_s6a:DIAMETER_CLIENT);
174 connect(vc_conn:S6a_PROC, vc_conn_s6a:DIAMETER_PROC_CLIENT);
175 vc_conn_s6a.start(f_diam_connhldr_ct_main(pars.imsi));
176
177 vc_conn.start(f_handler_init(fn, pars));
178 return vc_conn;
179}
180
181private function f_handler_init(void_fn fn, SessionPars pars)
182runs on Cli_Session_CT {
183 g_pars := valueof(pars);
184 fn.apply();
185}
186
187/* ULR + ULA against HSS */
Pau Espin Pedrold34720f2023-10-11 17:49:39 +0200188private function f_dia_ulr_ula(template (present) AVP_list ula_sub_data) runs on Cli_Session_CT {
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200189 var octetstring sess_id := char2oct("foobar");
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200190 var PDU_DIAMETER rx_dia;
191 var UINT32 hbh_id := int2oct(g_pars.s6a_next_hbh_id, 4);
192 var UINT32 ete_id := int2oct(g_pars.s6a_next_ete_id, 4);
193
194 /* Unlike ULR, ULA contains no IMSI. Register ete_id in DIAMETER_Emulation,
195 * so AIA is forwarded back to us in DIAMETER port instead of MTC_CT.DIAMETER_UNIT.
196 */
197 f_diam_connhldr_expect_eteid(ete_id);
198
199 /* TODO: change this into a ts_DIA_ULR */
200 S6a.send(ts_DIA_ULR(g_pars.imsi, '111F11'O, sess_id,
201 mp_diam_orig_host, mp_diam_orig_realm,
202 mp_diam_dest_realm, hbh_id, ete_id));
203 g_pars.s6a_next_hbh_id := g_pars.s6a_next_hbh_id + 1;
204 g_pars.s6a_next_ete_id := g_pars.s6a_next_ete_id + 1;
205
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200206 alt {
Pau Espin Pedrold34720f2023-10-11 17:49:39 +0200207 [] S6a.receive(tr_DIA_ULA(ula_sub_data, sess_id, ?, ?, hbh_id, ete_id)) -> value rx_dia {
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200208 setverdict(pass);
209 }
210 [] S6a.receive(PDU_DIAMETER:?) -> value rx_dia {
211 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
212 log2str("Received unexpected DIAMETER ", rx_dia));
213 }
214 }
215}
216
217/* create a session, expect it to succeed */
218private function f_TC_ulr_ula() runs on Cli_Session_CT {
Pau Espin Pedrold34720f2023-10-11 17:49:39 +0200219 var template (present) AVP_list sub_data := superset(
220 tr_AVP_3GPP_SubscriberStatus(SERVICE_GRANTED),
221 tr_AVP_3GPP_SubscrRauTauTmr(?),
222 tr_AVP_3GPP_AMBR(?, ?),
223 tr_AVP_3GPP_ApnConfigProfile(superset(
224 tr_AVP_3GPP_ContextId(?),
225 tr_AVP_3GPP_AllApnConfigsIncl,
226 tr_AVP_3GPP_ApnConfig(?, ?, ?)
227 ))
228 );
229
230 f_dia_ulr_ula(sub_data);
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200231 setverdict(pass);
232}
233testcase TC_ulr_ula() runs on MTC_CT {
234 var Cli_Session_CT vc_conn;
235 var SessionPars pars := valueof(t_SessionPars(subscribers[0]));
236 f_init();
237 vc_conn := f_start_handler(refers(f_TC_ulr_ula), pars);
238 vc_conn.done;
239}
240
Pau Espin Pedrold34720f2023-10-11 17:49:39 +0200241/* Same as TC_ulr_ula, but done on a subscriber configured with
242Subscriber-Status=1 (OPERATOR_DETERMINED_BARRING) and
243Operator-Determined-Barring=7. */
244private function f_TC_ulr_ula_subscr_op_det_barring_7() runs on Cli_Session_CT {
245 var template (present) AVP_list sub_data := superset(
246 tr_AVP_3GPP_SubscriberStatus(OPERATOR_DETERMINED_BARRING),
247 tr_AVP_3GPP_OperatorDeterminedBarring(7),
248 tr_AVP_3GPP_SubscrRauTauTmr(?),
249 tr_AVP_3GPP_AMBR(?, ?),
250 tr_AVP_3GPP_ApnConfigProfile(superset(
251 tr_AVP_3GPP_ContextId(?),
252 tr_AVP_3GPP_AllApnConfigsIncl,
253 tr_AVP_3GPP_ApnConfig(?, ?, ?)
254 ))
255 );
256
257 f_dia_ulr_ula(sub_data);
258 setverdict(pass);
259}
260testcase TC_ulr_ula_subscr_op_det_barring_7() runs on MTC_CT {
261 var Cli_Session_CT vc_conn;
262 var SessionPars pars := valueof(t_SessionPars(subscribers[1]));
263 f_init();
264 vc_conn := f_start_handler(refers(f_TC_ulr_ula_subscr_op_det_barring_7), pars);
265 vc_conn.done;
266}
267
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200268
269control {
270 execute( TC_ulr_ula() );
Pau Espin Pedrold34720f2023-10-11 17:49:39 +0200271 execute( TC_ulr_ula_subscr_op_det_barring_7() );
Pau Espin Pedrolad4179b2023-10-10 19:12:26 +0200272}
273
274
275}