blob: 7bb6910de0ae59641662a902e88780762db4d48a [file] [log] [blame]
Harald Welte4526da92020-03-05 23:08:10 +01001module PGW_Tests {
2
Pau Espin Pedrole343a882022-05-18 18:51:11 +02003import from TCCEncoding_Functions all;
4
Harald Welte4526da92020-03-05 23:08:10 +01005import from General_Types all;
6import from Osmocom_Types all;
7import from Native_Functions all;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +02008import from Misc_Helpers all;
Harald Welte4526da92020-03-05 23:08:10 +01009
10import from GTPv2_Types all;
11import from GTPv2_Templates all;
12import from GTPv2_Emulation all;
13
14import from UECUPS_Types all;
15
16import from DNS_Helpers all;
17
Harald Weltef4001512020-04-26 21:48:34 +020018
19import from DIAMETER_Types all;
20import from DIAMETER_Templates all;
21import from DIAMETER_Emulation all;
22
23
Harald Welte4526da92020-03-05 23:08:10 +010024modulepar {
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060025 charstring mp_pgw_hostname := "127.0.0.4";
Harald Welte4526da92020-03-05 23:08:10 +010026 charstring mp_local_hostname_c := "127.0.0.1";
27 charstring mp_local_hostname_u := "127.0.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020028
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +060029 charstring mp_run_prog_log_path := "/tmp";
Harald Welte4526da92020-03-05 23:08:10 +010030 charstring mp_run_prog_as_user := "laforge";
31 charstring mp_ping_hostname := "10.45.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020032
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060033 charstring mp_pcrf_local_ip := "127.0.0.9";
Harald Weltef4001512020-04-26 21:48:34 +020034 integer mp_pcrf_local_port := 3868;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020035
36 charstring mp_ocs_local_ip := "127.0.0.9";
37 integer mp_ocs_local_port := 3869;
Pau Espin Pedrole343a882022-05-18 18:51:11 +020038
39 charstring mp_diam_realm := "localdomain";
Harald Welte4526da92020-03-05 23:08:10 +010040}
41
42/* main component, we typically have one per testcase */
43type component PGW_Test_CT {
44 var GTPv2_Emulation_CT vc_GTP2;
45 port GTP2EM_PT TEID0;
Harald Weltef4001512020-04-26 21:48:34 +020046
47 /* emulated PCRF */
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020048 var DIAMETER_Emulation_CT vc_Gx;
49 port DIAMETER_PT Gx_UNIT;
50 port DIAMETEREM_PROC_PT Gx_PROC;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020051 /* emulated OCS */
52 var DIAMETER_Emulation_CT vc_Gy;
53 port DIAMETER_PT Gy_UNIT;
54 port DIAMETEREM_PROC_PT Gy_PROC;
Pau Espin Pedrol08880f12022-04-07 19:05:15 +020055 /* global test case guard timer (actual timeout value is set in f_init()) */
56 timer T_guard;
57}
58
59/* global altstep for global guard timer; */
60altstep as_Tguard() runs on PGW_Test_CT {
61 [] T_guard.timeout {
62 setverdict(fail, "Timeout of T_guard");
63 mtc.stop;
64 }
Harald Welte4526da92020-03-05 23:08:10 +010065}
66
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020067type component DIAMETER_ConnHdlr_CT extends DIAMETER_ConnHdlr {
68 port DIAMETER_Conn_PT DIAMETER_CLIENT;
69}
70
71function f_diam_connhldr_ct_main(hexstring imsi) runs on DIAMETER_ConnHdlr_CT {
72 var PDU_DIAMETER msg;
73
74 if (DIAMETER_PROC.checkstate("Connected")) {
75 f_diameter_expect(imsi);
76 }
77
78 while (true) {
79 alt {
80 [] DIAMETER_CLIENT.receive(PDU_DIAMETER:?) -> value msg {
81 DIAMETER.send(msg);
82 }
83 [] DIAMETER.receive(PDU_DIAMETER:?) -> value msg {
84 DIAMETER_CLIENT.send(msg);
85 }
86 }
87 }
88}
89
90
Harald Welte4526da92020-03-05 23:08:10 +010091/* per-session component; we typically have 1..N per testcase */
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020092type component PGW_Session_CT extends GTP2_ConnHdlr {
Harald Welte4526da92020-03-05 23:08:10 +010093 var SessionPars g_pars;
94
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020095 port DIAMETER_Conn_PT Gx;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020096 port DIAMETER_Conn_PT Gy;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020097
Harald Welte4526da92020-03-05 23:08:10 +010098 /* TEI (Data) local side */
99 var OCT4 g_teid;
100 /* TEI (Control) local side */
101 var OCT4 g_teic;
102 /* TEI (Data) remote side */
103 var OCT4 g_teid_remote;
104 /* TEI (Control) remote side */
105 var OCT4 g_teic_remote;
106 /* GTP-U IPv4 address remote sie */
107 var OCT4 g_gtpu4_remote;
108 var OCT16 g_gtpu6_remote;
109
110 /* Address allocation */
111 var OCT4 g_ip4_addr;
112 var OCT16 g_ip6_addr;
113 var integer g_ip6_plen;
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200114
115 /* Store last received Gy message */
116 var PDU_DIAMETER g_rx_gy;
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100117
118 /* number of programs started, used as identifier. */
119 var integer g_start_prog_count := 0;
Harald Welte4526da92020-03-05 23:08:10 +0100120}
121
122/* configuration data for a given Session */
123type record SessionPars {
124 hexstring imsi,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200125 hexstring msisdn optional,
Harald Welte4526da92020-03-05 23:08:10 +0100126 // serving network
127 integer rat_type,
128 // flags?
129 charstring apn,
130 /* Apn subscribed or non-subscribed */
131 boolean selection_mode,
132 BIT3 pdn_type,
133 /* PAA */
134 /* Max APN Restriction */
135 /* APN-AMBR */
136 octetstring pco optional,
137 octetstring epco optional,
138 /* Bearer Contexts to be created */
139
140 charstring tun_dev_name,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200141 charstring tun_netns_name optional,
142
143 /* In seconds. 0 => disabled, !0 => grant over CC-Time period */
144 integer gy_validity_time
Harald Welte4526da92020-03-05 23:08:10 +0100145}
146
147template (value) SessionPars
148t_SessionPars(hexstring imsi, charstring tundev, integer rat_type := 6, charstring apn := "internet",
149 boolean selection_mode := false, BIT3 pdn_type := '001'B) := {
150 imsi := imsi,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200151 msisdn := '1234'H,
Harald Welte4526da92020-03-05 23:08:10 +0100152 rat_type := rat_type,
153 apn := apn,
154 selection_mode := selection_mode,
155 pdn_type := pdn_type,
156 pco := omit,
157 epco := omit,
158 tun_dev_name := tundev,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200159 tun_netns_name := tundev,
160 gy_validity_time := 0
Harald Welte4526da92020-03-05 23:08:10 +0100161}
162
163type record BearerConfig {
164 integer eps_bearer_id
165}
166
167type function void_fn() runs on PGW_Session_CT;
168
Harald Weltef4001512020-04-26 21:48:34 +0200169friend function DiameterForwardUnitdataCallback(PDU_DIAMETER msg)
170runs on DIAMETER_Emulation_CT return template PDU_DIAMETER {
171 DIAMETER_UNIT.send(msg);
172 return omit;
173}
174
175friend function f_init_diameter(charstring id) runs on PGW_Test_CT {
176 var DIAMETEROps ops := {
177 create_cb := refers(DIAMETER_Emulation.ExpectedCreateCallback),
Vadim Yanitskiyb46f01e2021-12-06 03:23:13 +0300178 unitdata_cb := refers(DiameterForwardUnitdataCallback),
179 raw := false /* handler mode (IMSI based routing) */
Harald Weltef4001512020-04-26 21:48:34 +0200180 };
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200181 var DIAMETER_conn_parameters pars;
182
183 /* Gx setup: */
184 pars := {
Harald Weltef4001512020-04-26 21:48:34 +0200185 remote_ip := mp_pgw_hostname,
186 remote_sctp_port := -1,
187 local_ip := mp_pcrf_local_ip,
188 local_sctp_port := mp_pcrf_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200189 origin_host := "pcrf." & mp_diam_realm,
190 origin_realm := mp_diam_realm,
Pau Espin Pedrol33b47492022-03-08 17:43:01 +0100191 auth_app_id := omit,
Harald Weltef4001512020-04-26 21:48:34 +0200192 vendor_app_id := c_DIAMETER_3GPP_Gx_AID
193 };
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200194 vc_Gx := DIAMETER_Emulation_CT.create(id);
195 map(vc_Gx:DIAMETER, system:DIAMETER_CODEC_PT);
196 connect(vc_Gx:DIAMETER_UNIT, self:Gx_UNIT);
197 connect(vc_Gx:DIAMETER_PROC, self:Gx_PROC);
198 vc_Gx.start(DIAMETER_Emulation.main(ops, pars, id));
Harald Weltef4001512020-04-26 21:48:34 +0200199
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200200 /* Gy setup: */
201 pars := {
202 remote_ip := mp_pgw_hostname,
203 remote_sctp_port := -1,
204 local_ip := mp_ocs_local_ip,
205 local_sctp_port := mp_ocs_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200206 origin_host := "ocs." & mp_diam_realm,
207 origin_realm := mp_diam_realm,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200208 auth_app_id := c_DIAMETER_CREDIT_CONTROL_AID,
209 vendor_app_id := omit
210 };
211 vc_Gy := DIAMETER_Emulation_CT.create(id);
212 map(vc_Gy:DIAMETER, system:DIAMETER_CODEC_PT);
213 connect(vc_Gy:DIAMETER_UNIT, self:Gy_UNIT);
214 connect(vc_Gy:DIAMETER_PROC, self:Gy_PROC);
215 vc_Gy.start(DIAMETER_Emulation.main(ops, pars, id));
216
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200217 f_diameter_wait_capability(Gx_UNIT);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200218 f_diameter_wait_capability(Gy_UNIT);
Pau Espin Pedrolbb5f45f2022-04-11 12:36:55 +0200219 /* Give some time for our emulation to get out of SUSPECT list of SUT (3 watchdong ping-pongs):
220 * RFC6733 sec 5.1
221 * RFC3539 sec 3.4.1 [5]
222 * https://github.com/freeDiameter/freeDiameter/blob/master/libfdcore/p_psm.c#L49
223 */
224 f_sleep(1.0);
Harald Weltef4001512020-04-26 21:48:34 +0200225}
226
Pau Espin Pedrol08880f12022-04-07 19:05:15 +0200227private function f_init(float guard_timeout := 60.0) runs on PGW_Test_CT {
228 T_guard.start(guard_timeout);
229 activate(as_Tguard());
230
Harald Welte4526da92020-03-05 23:08:10 +0100231 var Gtp2EmulationCfg cfg := {
232 gtpc_bind_ip := mp_local_hostname_c,
233 gtpc_bind_port := GTP2C_PORT,
234 gtpc_remote_ip := mp_pgw_hostname,
235 gtpc_remote_port := GTP2C_PORT,
236 sgw_role := true,
237 use_gtpu_daemon := true
238 };
239
240 vc_GTP2 := GTPv2_Emulation_CT.create("GTP2_EM");
241 map(vc_GTP2:GTP2C, system:GTP2C);
242 connect(vc_GTP2:TEID0, self:TEID0);
243 vc_GTP2.start(GTPv2_Emulation.main(cfg));
Harald Weltef4001512020-04-26 21:48:34 +0200244
245 if (mp_pcrf_local_ip != "") {
246 f_init_diameter(testcasename());
247 }
Harald Welte4526da92020-03-05 23:08:10 +0100248}
249
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200250function f_start_handler(void_fn fn, template (omit) SessionPars pars_tmpl := omit)
Harald Welte4526da92020-03-05 23:08:10 +0100251runs on PGW_Test_CT return PGW_Session_CT {
252 var charstring id := testcasename();
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200253 var DIAMETER_ConnHdlr_CT vc_conn_gx, vc_conn_gy;
Harald Welte4526da92020-03-05 23:08:10 +0100254 var PGW_Session_CT vc_conn;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200255 var SessionPars pars;
256
257 if (isvalue(pars_tmpl)) {
258 pars := valueof(pars_tmpl);
259 } else {
260 /*TODO: set default values */
261 }
262
Harald Welte4526da92020-03-05 23:08:10 +0100263 vc_conn := PGW_Session_CT.create(id);
264 connect(vc_conn:GTP2, vc_GTP2:CLIENT);
265 connect(vc_conn:GTP2_PROC, vc_GTP2:CLIENT_PROC);
Harald Weltef4001512020-04-26 21:48:34 +0200266
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200267 if (isbound(vc_Gx)) {
268 vc_conn_gx := DIAMETER_ConnHdlr_CT.create(id);
269 connect(vc_conn_gx:DIAMETER, vc_Gx:DIAMETER_CLIENT);
270 connect(vc_conn_gx:DIAMETER_PROC, vc_Gx:DIAMETER_PROC);
271 connect(vc_conn:Gx, vc_conn_gx:DIAMETER_CLIENT);
272 vc_conn_gx.start(f_diam_connhldr_ct_main(pars.imsi));
Harald Weltef4001512020-04-26 21:48:34 +0200273 }
274
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200275 if (isbound(vc_Gy)) {
276 vc_conn_gy := DIAMETER_ConnHdlr_CT.create(id);
277 connect(vc_conn_gy:DIAMETER, vc_Gy:DIAMETER_CLIENT);
278 connect(vc_conn_gy:DIAMETER_PROC, vc_Gy:DIAMETER_PROC);
279 connect(vc_conn:Gy, vc_conn_gy:DIAMETER_CLIENT);
280 vc_conn_gy.start(f_diam_connhldr_ct_main(pars.imsi));
281 }
282
Harald Welte4526da92020-03-05 23:08:10 +0100283 vc_conn.start(f_handler_init(fn, pars));
284 return vc_conn;
285}
286
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200287private function f_handler_init(void_fn fn, SessionPars pars)
Harald Welte4526da92020-03-05 23:08:10 +0100288runs on PGW_Session_CT {
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200289 g_pars := valueof(pars);
Harald Welte4526da92020-03-05 23:08:10 +0100290 fn.apply();
291}
292
Harald Weltefe595e42020-04-21 22:56:47 +0200293private function f_concat_pad(integer tot_len, hexstring prefix, integer suffix) return hexstring {
294 var integer suffix_len := tot_len - lengthof(prefix);
295 var charstring suffix_ch := int2str(suffix);
296 var integer pad_len := suffix_len - lengthof(suffix_ch);
297
298 return prefix & int2hex(0, pad_len) & str2hex(suffix_ch);
299}
300
301function f_gen_imei(integer suffix) return hexstring {
302 return f_concat_pad(14, '49999'H, suffix);
303}
304
305function f_gen_imsi(integer suffix) return hexstring {
306 return f_concat_pad(15, '26242'H, suffix);
307}
308
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200309private altstep as_DIA_Gx_CCR(DCC_NONE_CC_Request_Type req_type) runs on PGW_Session_CT {
Harald Weltef4001512020-04-26 21:48:34 +0200310 var PDU_DIAMETER rx_dia;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200311 [] Gx.receive(tr_DIA_Gx_CCR(req_type := req_type)) -> value rx_dia {
Harald Weltef4001512020-04-26 21:48:34 +0200312 var template (omit) AVP avp;
313 var octetstring sess_id;
314 var AVP_Unsigned32 req_num;
315
316 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
317 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
318
319 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_DCC_NONE_CC_Request_Number);
320 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
321
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200322 Gx.send(ts_DIA_Gx_CCA(rx_dia.hop_by_hop_id, rx_dia.end_to_end_id, sess_id,
Harald Weltef4001512020-04-26 21:48:34 +0200323 req_type, req_num));
324 }
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200325 [] Gx.receive(PDU_DIAMETER:?) -> value rx_dia {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200326 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
327 log2str("Received unexpected DIAMETER ", rx_dia));
328 }
329}
330
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200331function f_tr_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
332runs on PGW_Session_CT return template (present) PDU_DIAMETER
333{
334 var template (present) PDU_DIAMETER tpl;
335 var charstring smf_origin_host := "smf." & mp_diam_realm;
336 var template (present) octetstring imsi := ?;
337 var template (present) octetstring msisdn := ?;
338 var template (present) octetstring rat_type := ?;
339 var template (present) OCT4 charging_char := ?;
340 var template (present) OCT1 nsapi := ?;
341 imsi := char2oct(f_dec_TBCD(imsi_hex2oct(g_pars.imsi)));
342 //msisdn := char2oct(f_dec_TBCD(substr(ctx_val.msisdn, 1, lengthof(ctx_val.msisdn) -1)));
343 rat_type := int2oct(g_pars.rat_type, 1);
344 charging_char := char2oct(oct2str('0000'O)); // f_create_session() uses hardcoded chg_car := '0000'O
345 nsapi := '01'O; // f_create_session() uses hardcoded bearer_id := 1
346 select (req_type) {
347 case (INITIAL_REQUEST) {
348 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
349 avps := superset(
350 tr_AVP_SessionId,
351 tr_AVP_OriginHost(smf_origin_host),
352 tr_AVP_OriginRealm(mp_diam_realm),
353 tr_AVP_DestinationRealm(mp_diam_realm),
354 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
355 tr_AVP_ServiceContextId,
356 tr_AVP_CcReqType(req_type),
357 tr_AVP_CcReqNum(?),
358 tr_AVP_EventTimestamp(?),
359 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
360 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
361 tr_AVP_RequestedAction(DIRECT_DEBITING),
362 tr_AVP_3GPP_AoCRequestType,
363 tr_AVP_MultipleServicesIndicator,
364 tr_AVP_Multiple_Services_Credit_Control(content := superset(
365 tr_AVP_Requested_Service_Unit,
366 tr_AVP_Used_Service_Unit,
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200367 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200368 tr_AVP_GI_3GPP_RatType(rat_type)
369 )),
370 tr_AVP_3GPP_ServiceInformation(content := superset(
371 tr_AVP_3GPP_PSInformation(content := superset(
372 tr_AVP_3GPP_ChargingId,
373 tr_AVP_3GPP_PDPType((IPv4,IPv6,IPv4v6)),
374 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
375 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
376 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
377 tr_AVP_3GPP_CalledStationId,
378 tr_AVP_3GPP_SelectionMode,
379 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
380 tr_AVP_3GPP_SGSNMCCMNC,
381 tr_AVP_3GPP_NSAPI(nsapi),
382 /*We don't yet send MS_Tz in CreateSessionReq:
383 tr_AVP_3GPP_MS_TimeZone,*/
384 tr_AVP_3GPP_ULI/*,
385 We don't yet send IMEI in CreateSessionReq:
386 tr_AVP_UserEquipmentInfo({
387 tr_AVP_UserEquipmentInfoType(IMEISV),
388 tr_AVP_UserEquipmentInfoValue(imeisv)
389 })*/
390 ))
391 ))
392 ));
393 }
394 case (UPDATE_REQUEST) {
395 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
396 avps := superset(
397 tr_AVP_SessionId,
398 tr_AVP_OriginHost(smf_origin_host),
399 tr_AVP_OriginRealm(mp_diam_realm),
400 tr_AVP_DestinationRealm(mp_diam_realm),
401 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
402 tr_AVP_ServiceContextId,
403 tr_AVP_CcReqType(req_type),
404 tr_AVP_CcReqNum(?),
405 tr_AVP_DestinationHost(?),
406 tr_AVP_EventTimestamp(?),
407 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
408 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
409 tr_AVP_RequestedAction(DIRECT_DEBITING),
410 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200411 tr_AVP_Multiple_Services_Credit_Control(content := superset(
412 tr_AVP_Requested_Service_Unit,
413 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200414 /* tr_AVP_3GPP_Reporting_Reason, can be sometimes inside UsedServiceUnit */
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200415 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200416 tr_AVP_GI_3GPP_RatType(rat_type)
417 )),
418 tr_AVP_3GPP_ServiceInformation(content := superset(
419 tr_AVP_3GPP_PSInformation(content := superset(
420 tr_AVP_3GPP_ChargingId,
421 /* tr_AVP_3GPP_PDPType, Only in INIT */
422 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
423 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
424 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
425 tr_AVP_3GPP_CalledStationId,
426 tr_AVP_3GPP_SelectionMode,
427 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
428 tr_AVP_3GPP_SGSNMCCMNC,
429 tr_AVP_3GPP_NSAPI(nsapi),
430 /*We don't yet send MS_Tz in CreateSessionReq:
431 tr_AVP_3GPP_MS_TimeZone,*/
432 tr_AVP_3GPP_ULI/*,
433 We don't yet send IMEI in CreateSessionReq:
434 tr_AVP_UserEquipmentInfo({
435 tr_AVP_UserEquipmentInfoType(IMEISV),
436 tr_AVP_UserEquipmentInfoValue(imeisv)
437 })*/
438 ))
439 ))
440 ));
441 }
442 case (TERMINATION_REQUEST) {
443 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
444 avps := superset(
445 tr_AVP_SessionId,
446 tr_AVP_OriginHost(smf_origin_host),
447 tr_AVP_OriginRealm(mp_diam_realm),
448 tr_AVP_DestinationRealm(mp_diam_realm),
449 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
450 tr_AVP_ServiceContextId,
451 tr_AVP_CcReqType(req_type),
452 tr_AVP_CcReqNum(?),
453 tr_AVP_DestinationHost(?),
454 tr_AVP_EventTimestamp(?),
455 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
456 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
457 tr_AVP_TerminationCause(?),
458 tr_AVP_RequestedAction(DIRECT_DEBITING),
459 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200460 tr_AVP_Multiple_Services_Credit_Control(content := superset(
461 /* tr_AVP_Requested_Service_Unit, Only in INIT and UPDATE */
462 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200463 tr_AVP_3GPP_Reporting_Reason(FINAL),
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200464 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200465 tr_AVP_GI_3GPP_RatType(rat_type)
466 )),
467 tr_AVP_3GPP_ServiceInformation(content := superset(
468 tr_AVP_3GPP_PSInformation(content := superset(
469 tr_AVP_3GPP_ChargingId,
470 /* tr_AVP_3GPP_PDPType, Only in INIT */
471 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
472 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
473 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
474 tr_AVP_3GPP_CalledStationId,
475 tr_AVP_3GPP_SelectionMode,
476 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
477 tr_AVP_3GPP_SGSNMCCMNC,
478 tr_AVP_3GPP_NSAPI(nsapi),
479 /*We don't yet send MS_Tz in CreateSessionReq:
480 tr_AVP_3GPP_MS_TimeZone,*/
481 tr_AVP_3GPP_ULI/*,
482 We don't yet send IMEI in CreateSessionReq:
483 tr_AVP_UserEquipmentInfo({
484 tr_AVP_UserEquipmentInfoType(IMEISV),
485 tr_AVP_UserEquipmentInfoValue(imeisv)
486 })*/
487 ))
488 ))
489 ));
490 }
491 }
492 return tpl;
493}
494
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200495private altstep as_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
496runs on PGW_Session_CT {
497 [] Gy.receive(f_tr_DIA_Gy_CCR(req_type := req_type)) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200498 var template (value) PDU_DIAMETER tx_dia;
499 var template (omit) AVP avp;
500 var octetstring sess_id;
501 var AVP_Unsigned32 req_num;
502
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200503 avp := f_DIAMETER_get_avp(g_rx_gy, c_AVP_Code_BASE_NONE_Session_Id);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200504 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
505
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200506 avp := f_DIAMETER_get_avp(g_rx_gy, c_AVP_Code_DCC_NONE_CC_Request_Number);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200507 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
508 if (g_pars.gy_validity_time > 0) {
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200509 tx_dia := ts_DIA_Gy_CCA_ValidityTime(g_rx_gy.hop_by_hop_id, g_rx_gy.end_to_end_id, sess_id,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200510 req_type, req_num, g_pars.gy_validity_time);
511 } else {
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200512 tx_dia := ts_DIA_Gy_CCA(g_rx_gy.hop_by_hop_id, g_rx_gy.end_to_end_id, sess_id,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200513 req_type, req_num);
514 }
515 Gy.send(tx_dia);
516 }
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200517 [] Gy.receive(PDU_DIAMETER:?) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200518 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200519 log2str("Received unexpected DIAMETER Gy", g_rx_gy));
Harald Weltef4001512020-04-26 21:48:34 +0200520 }
521}
522
Harald Welte4526da92020-03-05 23:08:10 +0100523
524/* find TEID of given interface type (and optionally instance) */
525private function f_find_teid(FullyQualifiedTEID_List list,
526 template (present) integer if_type,
527 template (present) BIT4 instance := ?)
528return template (omit) FullyQualifiedTEID
529{
530 var integer i;
531 for (i := 0; i < lengthof(list); i := i+1) {
532 if (match(list[i].interfaceType, if_type) and
533 match(list[i].instance, instance)) {
534 return list[i];
535 }
536 }
537 return omit;
538}
539
540/* process one to-be-created bearer context */
541private function process_bctx_create(BearerContextGrouped bctx) runs on PGW_Session_CT
542{
543 /* FIXME: EPS Bearer ID */
544 /* FIXME: Cause */
545
546 /* find F-TEID of the P-GW U side */
547 var FullyQualifiedTEID rx_fteid;
548 rx_fteid := valueof(f_find_teid(bctx.bearerContextIEs.fullyQualifiedTEID, 5, '0010'B));
549 g_teid_remote := rx_fteid.tEID_GRE_Key;
550 if (rx_fteid.v4_Flag == '1'B) {
551 g_gtpu4_remote := rx_fteid.iPv4_Address;
552 }
553 if (rx_fteid.v6_Flag == '1'B) {
554 g_gtpu6_remote := rx_fteid.iPv6_Address;
555 }
556
557 var UECUPS_CreateTun uecups_create := {
558 tx_teid := oct2int(g_teid_remote),
559 rx_teid := oct2int(g_teid),
560 user_addr_type := IPV4,
561 user_addr := '00000000'O,
562 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
563 remote_gtp_ep := valueof(ts_UECUPS_SockAddr(g_gtpu4_remote)),
564 tun_dev_name := g_pars.tun_dev_name,
565 tun_netns_name := g_pars.tun_netns_name
566 };
567
568 /* create tunnel in daemon */
569 if (isbound(g_ip4_addr)) {
570 uecups_create.user_addr := g_ip4_addr;
571 f_gtp2_create_tunnel(uecups_create);
572 }
573 if (isbound(g_ip6_addr)) {
574 uecups_create.user_addr_type := IPV6;
575 uecups_create.user_addr := g_ip6_addr;
576 f_gtp2_create_tunnel(uecups_create);
577 }
578}
579
580/* create a session on the PGW */
581private function f_create_session() runs on PGW_Session_CT {
582 var PDU_GTPCv2 rx;
583
584 /* allocate + register TEID-C on local side */
585 g_teic := f_gtp2_allocate_teid();
586 g_teid := g_teic;
587
588 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
589 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_teic, 0,
590 f_inet_addr(mp_local_hostname_c), omit);
591 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPU, g_teid, 2,
592 f_inet_addr(mp_local_hostname_u), omit);
593 var template (value) PDU_GTPCv2 g2c :=
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200594 ts_GTP2C_CreateSessionReq(imsi := g_pars.imsi, msisdn := g_pars.msisdn, rat_type := 6,
Harald Welte4526da92020-03-05 23:08:10 +0100595 sender_fteid := fteid_c_ie,
596 apn := f_enc_dns_hostname(g_pars.apn),
597 pdn_type := g_pars.pdn_type, teid_list := { fteid_u_ie },
598 chg_car := '0000'O, bearer_id := 1);
599 /* open5gs up to 1.2.3 won't accept it without ULI, despite not mandatory */
600 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
601 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
602 g2c.gtpcv2_pdu.createSessionRequest.userLocationInfo := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
Vadim Yanitskiybada3c92022-01-20 18:59:07 +0600603 g2c.gtpcv2_pdu.createSessionRequest.servingNetwork := ts_GTP2C_ServingNetwork('001'H, '01F'H);
Harald Welte4526da92020-03-05 23:08:10 +0100604
605 GTP2.send(g2c);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200606 if (Gx.checkstate("Connected")) {
607 as_DIA_Gx_CCR(INITIAL_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200608 }
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200609 if (Gy.checkstate("Connected")) {
610 as_DIA_Gy_CCR(INITIAL_REQUEST);
611 }
Harald Welte4526da92020-03-05 23:08:10 +0100612 alt {
613 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_teic, cause:='10'O)) -> value rx {
614 /* extract TEIDs */
615 var CreateSessionResponse resp := rx.gtpcv2_pdu.createSessionResponse;
616 g_teic_remote := resp.fullyQualifiedTEID[0].tEID_GRE_Key;
617
618 /* extract allocated address[es] */
619 var PDN_Address_and_Prefix paa := resp.pDN_AddressAllocation.pDN_Address_and_Prefix;
620 if (ischosen(paa.iPv4_Address)) {
621 g_ip4_addr := paa.iPv4_Address;
622 } else if (ischosen(paa.iPv6_Address)) {
623 g_ip6_addr := paa.iPv6_Address.iPv6_Address;
624 g_ip6_plen := paa.iPv6_Address.prefixLength;
625 } else if (ischosen(paa.iPv4_IPv6)) {
626 g_ip4_addr := paa.iPv4_IPv6.iPv4_Address;
627 g_ip6_addr := paa.iPv4_IPv6.iPv6_Address;
628 g_ip6_plen := paa.iPv4_IPv6.prefixLength;
629 }
630 var integer i;
631 for (i := 0; i < lengthof(resp.bearerContextGrouped); i := i+1) {
632 var BearerContextGrouped bctx := resp.bearerContextGrouped[i];
633 select (bctx.instance) {
634 case ('0000'B) { // created
635 process_bctx_create(bctx);
636 }
637 case ('0001'B) { // removed
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100638 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
639 "We don't expect removed bearer contexts yet");
Harald Welte4526da92020-03-05 23:08:10 +0100640 }
641 }
642 }
643 }
644 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_teic, cause:=?)) -> value rx {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100645 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
646 log2str("Unexpected CreateSessionResp(cause=",
647 rx.gtpcv2_pdu.createSessionResponse.cause.causeValue, ")"));
Harald Welte4526da92020-03-05 23:08:10 +0100648 }
649 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100650 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
651 "Unexpected GTPv2 while waiting for CreateSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100652 }
653 }
654
655}
656
657/* delete the session from the PGW */
658private function f_delete_session(template (omit) OCT1 tx_cause := omit,
659 template (present) OCT4 exp_teid,
Pau Espin Pedrol5b0327b2022-04-11 13:02:36 +0200660 template (present) OCT1 exp_cause,
661 boolean expect_diameter := true) runs on PGW_Session_CT {
Harald Welte4526da92020-03-05 23:08:10 +0100662 var template (value) FullyQualifiedTEID fteid_c_ie
663 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_teic, 0,
664 f_inet_addr(mp_local_hostname_c), omit);
665 var template PDU_GTPCv2 g2c :=
666 ts_GTP2C_DeleteSessionReq(d_teid := g_teic_remote, cause := tx_cause,
667 sender_fteid := fteid_c_ie,
668 teid_list := {}, bearer_id := 1);
669
670 GTP2.send(g2c);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200671 if (Gx.checkstate("Connected") and expect_diameter) {
672 as_DIA_Gx_CCR(TERMINATION_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200673 }
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200674 if (Gy.checkstate("Connected") and expect_diameter) {
675 as_DIA_Gy_CCR(TERMINATION_REQUEST);
676 }
Harald Welte4526da92020-03-05 23:08:10 +0100677 alt {
678 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(d_teid := exp_teid, cause := exp_cause)) {
679 setverdict(pass);
680 }
681 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(?, ?)) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100682 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
683 "Unexpected DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100684 }
685 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100686 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
687 "Unexpected GTPv2 while waiting for DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100688 }
689 }
690
691 /* destroy tunnel in daemon */
692 if (isbound(g_teid)) {
693 var UECUPS_DestroyTun uecups_destroy := {
694 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
695 rx_teid := oct2int(g_teid)
696 };
697 /* FIXME: what about IPv4/IPv6 differentiation? */
698 f_gtp2_destroy_tunnel(uecups_destroy);
699 }
700}
701
702/* start a program on the user plane side; return its PID */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600703private function f_start_prog(charstring command, boolean redirect_output := true)
704runs on PGW_Session_CT return integer
Harald Welte4526da92020-03-05 23:08:10 +0100705{
706 var UECUPS_StartProgram sprog := {
707 command := command,
708 environment := {},
709 run_as_user := mp_run_prog_as_user,
710 tun_netns_name := g_pars.tun_netns_name
711 };
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100712 g_start_prog_count := g_start_prog_count + 1;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600713
714 /* Redirect stdout/stderr to the user-specified location */
715 if (redirect_output) {
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100716 var charstring id := testcasename() & "-" & hex2str(g_pars.imsi) & "-" & int2str(g_start_prog_count);
717 var charstring prefix := mp_run_prog_log_path & "/" & id;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600718 sprog.command := sprog.command & " 1>>" & prefix & ".prog.stdout";
719 sprog.command := sprog.command & " 2>>" & prefix & ".prog.stderr";
720 }
721
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600722 log("Starting a program: ", command);
Harald Welte4526da92020-03-05 23:08:10 +0100723 var UECUPS_StartProgramRes res := f_gtp2_start_program(sprog);
724 if (res.result != OK) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100725 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
726 log2str("Unable to start program '", command, "'"));
Harald Welte4526da92020-03-05 23:08:10 +0100727 }
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100728 log("Started program '", command, "' with PID ", res.pid);
Harald Welte4526da92020-03-05 23:08:10 +0100729 return res.pid;
730}
731
732/* wait for termination of a given PID with specified exit_code */
733private function f_wait_term(integer pid, template (present) integer exit_code := 0,
734 float tout := 10.0) runs on PGW_Session_CT
735{
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600736 var UECUPS_ProgramTermInd pti;
Harald Welte4526da92020-03-05 23:08:10 +0100737 timer T := tout;
738
739 T.start;
740 alt {
741 [] GTP2.receive(UECUPS_ProgramTermInd:{pid := pid, exit_code := exit_code}) {
742 setverdict(pass);
743 }
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600744 [] GTP2.receive(UECUPS_ProgramTermInd:?) -> value pti {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100745 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
746 log2str("Received unexpected ProgramTermInd := ", pti));
Harald Welte4526da92020-03-05 23:08:10 +0100747 }
748 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100749 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100750 log2str("timeout (", tout, " seconds) waiting for user-plane program PID ", pid, " termination"));
Harald Welte4526da92020-03-05 23:08:10 +0100751 }
752 }
753}
754
755/* execute a program and wait for result */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600756private function f_start_prog_wait(charstring command,
757 template integer exit_code := 0,
758 float tout := 10.0,
759 boolean redirect_output := true)
760runs on PGW_Session_CT
Harald Welte4526da92020-03-05 23:08:10 +0100761{
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600762 var integer pid := f_start_prog(command, redirect_output);
Harald Welte4526da92020-03-05 23:08:10 +0100763 f_wait_term(pid, exit_code, tout);
764}
765
766/* execute ping command and wait for result */
767private function f_ping4(charstring host, integer interval := 1, integer count := 10) runs on PGW_Session_CT
768{
769 var charstring ping :="ping -c " & int2str(count) & " -i " & int2str(interval);
Pau Espin Pedrol55980c42023-02-28 12:40:52 +0100770
771 if (not isbound(g_ip4_addr)) {
772 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "f_ping4(): g_ip4_addr is unset!");
773 }
Harald Welte4526da92020-03-05 23:08:10 +0100774 ping := ping & " -I " & f_inet_ntoa(g_ip4_addr);
775 ping := ping & " " & host;
Harald Welte8cfdc7c2020-04-21 22:48:34 +0200776 f_start_prog_wait(ping, tout := int2float(5 + interval*count));
Harald Welte4526da92020-03-05 23:08:10 +0100777}
778
779
780
781
782/* send echo request; expect response */
783testcase TC_tx_echo() runs on PGW_Test_CT {
784 timer T := 5.0;
785
786 f_init();
787
788 TEID0.send(ts_GTP2C_EchoReq(0));
789 T.start;
790 alt {
791 [] TEID0.receive(tr_GTP2C_EchoResp) {
792 setverdict(pass);
793 }
794 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100795 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "timeout waiting for Echo Response");
Harald Welte4526da92020-03-05 23:08:10 +0100796 }
797 }
798}
799
800/* create a session, expect it to succeed */
801private function f_TC_createSession() runs on PGW_Session_CT {
802 f_create_session();
803 setverdict(pass);
804}
805testcase TC_createSession() runs on PGW_Test_CT {
806 var PGW_Session_CT vc_conn;
807 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun22"));
808 f_init();
809 vc_conn := f_start_handler(refers(f_TC_createSession), pars);
810 vc_conn.done;
811}
812
813/* create a session, then execute a ping command on the user plane */
814private function f_TC_createSession_ping4() runs on PGW_Session_CT {
815 f_create_session();
816 f_ping4(mp_ping_hostname);
817 setverdict(pass);
818}
819testcase TC_createSession_ping4() runs on PGW_Test_CT {
820 var PGW_Session_CT vc_conn;
821 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
822 f_init();
823 vc_conn := f_start_handler(refers(f_TC_createSession_ping4), pars);
824 vc_conn.done;
825}
Harald Weltefe595e42020-04-21 22:56:47 +0200826testcase TC_createSession_ping4_256() runs on PGW_Test_CT {
827 var PGW_Session_CT vc_conn[256];
828 var integer i;
829
830 f_init();
831
832 for (i := 0; i < sizeof(vc_conn); i:=i+1) {
833 var charstring tundev := "ping" & int2str(i);
834 var SessionPars pars := valueof(t_SessionPars(f_gen_imsi(i), tundev));
835 vc_conn[i] := f_start_handler(refers(f_TC_createSession_ping4), pars);
836 }
837
838 for (i := 0; i < lengthof(vc_conn); i:=i+1) {
839 vc_conn[i].done;
840 }
841}
842
Harald Welte4526da92020-03-05 23:08:10 +0100843
844/* create a session, then delete it again */
845private function f_TC_createSession_deleteSession() runs on PGW_Session_CT {
846 f_create_session();
847 f_delete_session(omit, g_teic, '10'O);
848 setverdict(pass);
849}
850testcase TC_createSession_deleteSession() runs on PGW_Test_CT {
851 var PGW_Session_CT vc_conn;
852 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
853 f_init();
854 vc_conn := f_start_handler(refers(f_TC_createSession_deleteSession), pars);
855 vc_conn.done;
856}
857
858/* send a DeleteSessionReq for an unknown/invalid TEID */
859private function f_TC_deleteSession_unknown() runs on PGW_Session_CT {
860 g_teic := f_gtp2_allocate_teid();
861 g_teic_remote := f_rnd_octstring(4);
Pau Espin Pedrol5b0327b2022-04-11 13:02:36 +0200862 f_delete_session(omit, '00000000'O, '40'O /* Context Unknown */, false);
Harald Welte4526da92020-03-05 23:08:10 +0100863 setverdict(pass);
864}
865testcase TC_deleteSession_unknown() runs on PGW_Test_CT {
866 var PGW_Session_CT vc_conn;
867 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
868 f_init();
869 vc_conn := f_start_handler(refers(f_TC_deleteSession_unknown), pars);
870 vc_conn.done;
871}
872
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200873/* Test charging over Gy interface. */
874private function f_TC_gy_charging_cc_time() runs on PGW_Session_CT {
875 var default d;
Harald Welte4526da92020-03-05 23:08:10 +0100876
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200877 f_create_session();
Harald Welte4526da92020-03-05 23:08:10 +0100878
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200879 /* We should receive an update even if no traffic is sent: */
880 as_DIA_Gy_CCR(UPDATE_REQUEST);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200881 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200882
883 d := activate(as_DIA_Gy_CCR(UPDATE_REQUEST));
884 f_ping4(mp_ping_hostname);
885 /* Let the CCA reach the GGSN */
886 f_sleep(0.5);
887 deactivate(d);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200888 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), (28..1000), (28..1000));
889
890 as_DIA_Gy_CCR(UPDATE_REQUEST);
891 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), ?, ?);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200892
893 f_delete_session(omit, g_teic, '10'O);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200894 f_validate_gy_cc_report(g_rx_gy, FINAL, (0..1), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200895 setverdict(pass);
896}
897testcase TC_gy_charging_cc_time() runs on PGW_Test_CT {
898 var PGW_Session_CT vc_conn;
899 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
900 pars.gy_validity_time := 3; /* Grant access for 3 seconds, needs to be re-validated afterwards */
901 f_init();
902 vc_conn := f_start_handler(refers(f_TC_gy_charging_cc_time), pars);
903 vc_conn.done;
904}
Harald Welte4526da92020-03-05 23:08:10 +0100905
906control {
907 execute( TC_tx_echo() );
908 execute( TC_createSession() );
909 execute( TC_createSession_ping4() );
Harald Weltefe595e42020-04-21 22:56:47 +0200910 execute( TC_createSession_ping4_256() );
Harald Welte4526da92020-03-05 23:08:10 +0100911 execute( TC_createSession_deleteSession() );
912 execute( TC_deleteSession_unknown() );
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200913 execute( TC_gy_charging_cc_time() );
Harald Welte4526da92020-03-05 23:08:10 +0100914}
915
916
917}