blob: a0c1e79c3328eeba83798e8a038d4b72394fa3da [file] [log] [blame]
Jan Luebbefaaa49c2008-12-27 01:07:07 +00001/* (C) 2008 by Jan Luebbe <jluebbe@debian.org>
Holger Hans Peter Freyther2826df52016-04-01 20:21:03 +02002 * (C) 2009-2016 by Holger Hans Peter Freyther <zecke@selfish.org>
Alexander Chemeris82a18582014-03-08 18:48:49 +01003 * (C) 2014 by Alexander Chemeris <Alexander.Chemeris@fairwaves.co>
Jan Luebbefaaa49c2008-12-27 01:07:07 +00004 * All Rights Reserved
5 *
6 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +01007 * it under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation; either version 3 of the License, or
Jan Luebbefaaa49c2008-12-27 01:07:07 +00009 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010016 * You should have received a copy of the GNU Affero General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Jan Luebbefaaa49c2008-12-27 01:07:07 +000018 *
19 */
20
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010021#include <openbsc/debug.h>
Jan Luebbefaaa49c2008-12-27 01:07:07 +000022#include <openbsc/db.h>
Holger Hans Peter Freyther28dcbc52010-12-22 18:21:14 +010023#include <openbsc/gsm_subscriber.h>
Holger Hans Peter Freyther73bc51d2014-03-07 19:08:03 +010024#include <openbsc/gsm_04_11.h>
Neels Hofmeyr84da6b12016-05-20 21:59:55 +020025#include <openbsc/mgcp.h>
Jan Luebbe7398eb92008-12-27 00:45:41 +000026
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010027#include <osmocom/core/application.h>
28
Jan Luebbe7398eb92008-12-27 00:45:41 +000029#include <stdio.h>
Jan Luebbe5c15c852008-12-27 15:59:25 +000030#include <string.h>
Harald Welte12247c62009-05-21 07:23:02 +000031#include <stdlib.h>
Maxe6052c42016-06-30 10:25:49 +020032#include <stdbool.h>
Alexander Chemeris82a18582014-03-08 18:48:49 +010033#include <inttypes.h>
Jan Luebbe7398eb92008-12-27 00:45:41 +000034
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010035static struct gsm_network dummy_net;
Jacob Erlbeck1e30a282014-12-03 09:28:24 +010036static struct gsm_subscriber_group dummy_sgrp;
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010037
38#define SUBSCR_PUT(sub) \
Jacob Erlbeck1e30a282014-12-03 09:28:24 +010039 sub->group = &dummy_sgrp; \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010040 subscr_put(sub);
41
Holger Freyther12aa50d2009-01-01 18:02:05 +000042#define COMPARE(original, copy) \
43 if (original->id != copy->id) \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010044 printf("Ids do not match in %s:%d %llu %llu\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000045 __FUNCTION__, __LINE__, original->id, copy->id); \
46 if (original->lac != copy->lac) \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010047 printf("LAC do not match in %s:%d %d %d\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000048 __FUNCTION__, __LINE__, original->lac, copy->lac); \
49 if (original->authorized != copy->authorized) \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010050 printf("Authorize do not match in %s:%d %d %d\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000051 __FUNCTION__, __LINE__, original->authorized, \
52 copy->authorized); \
53 if (strcmp(original->imsi, copy->imsi) != 0) \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010054 printf("IMSIs do not match in %s:%d '%s' '%s'\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000055 __FUNCTION__, __LINE__, original->imsi, copy->imsi); \
Holger Hans Peter Freyther22230252009-08-19 12:53:57 +020056 if (original->tmsi != copy->tmsi) \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010057 printf("TMSIs do not match in %s:%d '%u' '%u'\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000058 __FUNCTION__, __LINE__, original->tmsi, copy->tmsi); \
59 if (strcmp(original->name, copy->name) != 0) \
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +010060 printf("names do not match in %s:%d '%s' '%s'\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000061 __FUNCTION__, __LINE__, original->name, copy->name); \
62 if (strcmp(original->extension, copy->extension) != 0) \
Alexander Chemeris86d46c52013-10-04 02:42:24 +020063 printf("Extensions do not match in %s:%d '%s' '%s'\n", \
Holger Freyther12aa50d2009-01-01 18:02:05 +000064 __FUNCTION__, __LINE__, original->extension, copy->extension); \
65
Holger Hans Peter Freyther73bc51d2014-03-07 19:08:03 +010066/*
67 * Create/Store a SMS and then try to load it.
68 */
69static void test_sms(void)
70{
71 int rc;
72 struct gsm_sms *sms;
73 struct gsm_subscriber *subscr;
74 subscr = db_get_subscriber(GSM_SUBSCRIBER_IMSI, "9993245423445");
75 OSMO_ASSERT(subscr);
Jacob Erlbeck1e30a282014-12-03 09:28:24 +010076 subscr->group = &dummy_sgrp;
Holger Hans Peter Freyther73bc51d2014-03-07 19:08:03 +010077
78 sms = sms_alloc();
79 sms->receiver = subscr_get(subscr);
80
81 sms->src.ton = 0x23;
82 sms->src.npi = 0x24;
83 memcpy(sms->src.addr, "1234", strlen("1234") + 1);
84
85 sms->dst.ton = 0x32;
86 sms->dst.npi = 0x42;
87 memcpy(sms->dst.addr, subscr->extension, sizeof(subscr->extension));
88
89 memcpy(sms->text, "Text123", strlen("Text123") + 1);
90 memcpy(sms->user_data, "UserData123", strlen("UserData123") + 1);
91 sms->user_data_len = strlen("UserData123");
92
93 /* random values */
94 sms->reply_path_req = 1;
95 sms->status_rep_req = 2;
96 sms->ud_hdr_ind = 3;
97 sms->protocol_id = 4;
98 sms->data_coding_scheme = 5;
99
100 rc = db_sms_store(sms);
101 sms_free(sms);
102 OSMO_ASSERT(rc == 0);
103
104 /* now query */
105 sms = db_sms_get_unsent_for_subscr(subscr);
106 OSMO_ASSERT(sms);
107 OSMO_ASSERT(sms->receiver == subscr);
108 OSMO_ASSERT(sms->reply_path_req == 1);
109 OSMO_ASSERT(sms->status_rep_req == 2);
110 OSMO_ASSERT(sms->ud_hdr_ind == 3);
111 OSMO_ASSERT(sms->protocol_id == 4);
112 OSMO_ASSERT(sms->data_coding_scheme == 5);
113 OSMO_ASSERT(sms->src.ton == 0x23);
114 OSMO_ASSERT(sms->src.npi == 0x24);
115 OSMO_ASSERT(sms->dst.ton == 0x32);
116 OSMO_ASSERT(sms->dst.npi == 0x42);
117 OSMO_ASSERT(strcmp((char *) sms->text, "Text123") == 0);
118 OSMO_ASSERT(sms->user_data_len == strlen("UserData123"));
119 OSMO_ASSERT(strcmp((char *) sms->user_data, "UserData123") == 0);
120
Holger Hans Peter Freytherbe8e7752014-12-25 17:27:15 +0100121 /* Mark the SMS as delivered */
122 db_sms_mark_delivered(sms);
123 sms_free(sms);
124
125 sms = db_sms_get_unsent_for_subscr(subscr);
126 OSMO_ASSERT(!sms);
127
Holger Hans Peter Freyther73bc51d2014-03-07 19:08:03 +0100128 subscr_put(subscr);
129}
130
Holger Hans Peter Freyther61144012014-03-08 16:41:37 +0100131static void test_sms_migrate(void)
132{
133 struct gsm_subscriber *rcv_subscr;
134 struct gsm_sms *sms;
135 static const uint8_t user_data_1[] = {
136 0x41, 0xf1, 0xd8, 0x05, 0x22, 0x96, 0xcd, 0x2e,
137 0x90, 0xf1, 0xfd, 0x06, 0x00 };
138 static const uint8_t user_data_2[] = {
139 0x41, 0xf1, 0xd8, 0x05, 0x22, 0x96, 0xcd, 0x2e,
140 0xd0, 0xf1, 0xfd, 0x06, 0x00 };
141
142 rcv_subscr = db_get_subscriber(GSM_SUBSCRIBER_IMSI, "901010000001111");
Jacob Erlbeck1e30a282014-12-03 09:28:24 +0100143 rcv_subscr->group = &dummy_sgrp;
Holger Hans Peter Freyther61144012014-03-08 16:41:37 +0100144
145 sms = db_sms_get(&dummy_net, 1);
146 OSMO_ASSERT(sms->id == 1);
147 OSMO_ASSERT(sms->receiver == rcv_subscr);
148 OSMO_ASSERT(strcmp(sms->text, "Abc. Def. Foo") == 0);
149 OSMO_ASSERT(sms->user_data_len == ARRAY_SIZE(user_data_1));
150 OSMO_ASSERT(memcmp(sms->user_data, user_data_1, ARRAY_SIZE(user_data_1)) == 0);
151 sms_free(sms);
152
153 sms = db_sms_get(&dummy_net, 2);
154 OSMO_ASSERT(sms->id == 2);
155 OSMO_ASSERT(sms->receiver == rcv_subscr);
156 OSMO_ASSERT(strcmp(sms->text, "Abc. Def. Goo") == 0);
157 OSMO_ASSERT(sms->user_data_len == ARRAY_SIZE(user_data_2));
158 OSMO_ASSERT(memcmp(sms->user_data, user_data_2, ARRAY_SIZE(user_data_2)) == 0);
159 sms_free(sms);
160
161 subscr_put(rcv_subscr);
162}
163
Maxe6052c42016-06-30 10:25:49 +0200164static void test_subs(const char *imsi, char *imei1, char *imei2, bool make_ext)
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +0100165{
Max9a7e25b2016-05-11 13:05:13 +0200166 struct gsm_subscriber *alice = NULL, *alice_db;
Alexander Chemeris82a18582014-03-08 18:48:49 +0100167 char scratch_str[256];
168
Maxe6052c42016-06-30 10:25:49 +0200169 alice = db_create_subscriber(imsi, GSM_MIN_EXTEN, GSM_MAX_EXTEN,
170 make_ext);
Max9a7e25b2016-05-11 13:05:13 +0200171 db_subscriber_assoc_imei(alice, imei1);
172 if (imei2)
173 db_subscriber_assoc_imei(alice, imei2);
174 db_subscriber_alloc_tmsi(alice);
175 alice->lac=42;
176 db_sync_subscriber(alice);
177 /* Get by TMSI */
178 snprintf(scratch_str, sizeof(scratch_str), "%"PRIu32, alice->tmsi);
179 alice_db = db_get_subscriber(GSM_SUBSCRIBER_TMSI, scratch_str);
180 COMPARE(alice, alice_db);
181 SUBSCR_PUT(alice_db);
182 /* Get by IMSI */
Maxe6052c42016-06-30 10:25:49 +0200183 alice_db = db_get_subscriber(GSM_SUBSCRIBER_IMSI, imsi);
Max9a7e25b2016-05-11 13:05:13 +0200184 COMPARE(alice, alice_db);
185 SUBSCR_PUT(alice_db);
186 /* Get by id */
187 snprintf(scratch_str, sizeof(scratch_str), "%llu", alice->id);
188 alice_db = db_get_subscriber(GSM_SUBSCRIBER_ID, scratch_str);
189 COMPARE(alice, alice_db);
190 SUBSCR_PUT(alice_db);
191 /* Get by extension */
192 alice_db = db_get_subscriber(GSM_SUBSCRIBER_EXTENSION, alice->extension);
Maxe6052c42016-06-30 10:25:49 +0200193 if (alice_db) {
194 if (!make_ext)
195 printf("FAIL: bogus extension created for IMSI %s\n",
196 imsi);
197 COMPARE(alice, alice_db);
198 SUBSCR_PUT(alice_db);
199 } else if (make_ext)
200 printf("FAIL: no subscriber extension for IMSI %s\n", imsi);
Max9a7e25b2016-05-11 13:05:13 +0200201 SUBSCR_PUT(alice);
202}
203
204int main()
205{
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +0100206 printf("Testing subscriber database code.\n");
207 osmo_init_logging(&log_info);
Holger Hans Peter Freyther3c9068f2014-05-01 07:53:42 +0200208 log_set_print_filename(osmo_stderr_target, 0);
Jan Luebbe7398eb92008-12-27 00:45:41 +0000209
Jacob Erlbeck1e30a282014-12-03 09:28:24 +0100210 dummy_net.subscr_group = &dummy_sgrp;
211 dummy_sgrp.net = &dummy_net;
212
Holger Freytherc7b86f92009-06-06 13:54:20 +0000213 if (db_init("hlr.sqlite3")) {
Jan Luebbe5c15c852008-12-27 15:59:25 +0000214 printf("DB: Failed to init database. Please check the option settings.\n");
215 return 1;
216 }
217 printf("DB: Database initialized.\n");
Jan Luebbe7398eb92008-12-27 00:45:41 +0000218
Jan Luebbe5c15c852008-12-27 15:59:25 +0000219 if (db_prepare()) {
220 printf("DB: Failed to prepare database.\n");
221 return 1;
222 }
223 printf("DB: Database prepared.\n");
Jan Luebbe7398eb92008-12-27 00:45:41 +0000224
Jan Luebbe5c15c852008-12-27 15:59:25 +0000225 struct gsm_subscriber *alice = NULL;
Holger Freyther12aa50d2009-01-01 18:02:05 +0000226 struct gsm_subscriber *alice_db;
Jan Luebbe7398eb92008-12-27 00:45:41 +0000227
Holger Freyther12aa50d2009-01-01 18:02:05 +0000228 char *alice_imsi = "3243245432345";
Maxe6052c42016-06-30 10:25:49 +0200229 alice = db_create_subscriber(alice_imsi, GSM_MIN_EXTEN, GSM_MAX_EXTEN,
230 true);
Holger Freyther12aa50d2009-01-01 18:02:05 +0000231 db_sync_subscriber(alice);
Holger Hans Peter Freyther7634ec12013-10-04 08:35:11 +0200232 alice_db = db_get_subscriber(GSM_SUBSCRIBER_IMSI, alice->imsi);
Holger Freyther12aa50d2009-01-01 18:02:05 +0000233 COMPARE(alice, alice_db);
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +0100234 SUBSCR_PUT(alice_db);
235 SUBSCR_PUT(alice);
Jan Luebbe7398eb92008-12-27 00:45:41 +0000236
Maxe6052c42016-06-30 10:25:49 +0200237 test_subs("3693245423445", "1234567890", NULL, true);
238 test_subs("9993245423445", "1234567890", "6543560920", true);
239 test_subs("3123122223445", "1234567890", NULL, false);
240 test_subs("9123121223445", "1234567890", "6543560920", false);
Jan Luebbe5c15c852008-12-27 15:59:25 +0000241
Holger Hans Peter Freyther2826df52016-04-01 20:21:03 +0200242 /* create it again and see it fails */
Maxe6052c42016-06-30 10:25:49 +0200243 alice = db_create_subscriber(alice_imsi, GSM_MIN_EXTEN, GSM_MAX_EXTEN,
244 true);
Holger Hans Peter Freyther2826df52016-04-01 20:21:03 +0200245 OSMO_ASSERT(!alice);
246
Holger Hans Peter Freyther73bc51d2014-03-07 19:08:03 +0100247 test_sms();
Holger Hans Peter Freyther61144012014-03-08 16:41:37 +0100248 test_sms_migrate();
Holger Hans Peter Freyther73bc51d2014-03-07 19:08:03 +0100249
Jan Luebbe5c15c852008-12-27 15:59:25 +0000250 db_fini();
251
Holger Hans Peter Freyther45222a72012-01-06 14:27:08 +0100252 printf("Done\n");
Jan Luebbe5c15c852008-12-27 15:59:25 +0000253 return 0;
Jan Luebbe7398eb92008-12-27 00:45:41 +0000254}
Holger Freytherbab9cd92009-04-12 05:37:07 +0000255
256/* stubs */
Holger Hans Peter Freyther763b42a2010-12-29 11:07:22 +0100257void vty_out() {}
Harald Welte2483f1b2016-06-19 18:06:02 +0200258void vlr_subscr_disconnected() {}
259void vlr_subscr_rx_tmsi_reall_compl() {}
260void vlr_subscr_rx_id_resp() {}
261void vlr_subscr_rx_auth_resp() {}
262void vlr_loc_update() {}
263void vlr_proc_acc_req() {}
264void vlr_init() {}
Neels Hofmeyr84da6b12016-05-20 21:59:55 +0200265unsigned int mgcpgw_client_next_endpoint(struct mgcpgw_client *client)
266{ return 0; }
267struct msgb *mgcp_msg_crcx(struct mgcpgw_client *mgcp,
268 uint16_t rtp_endpoint, unsigned int call_id,
269 enum mgcp_connection_mode mode)
270{ return NULL; }
271struct msgb *mgcp_msg_mdcx(struct mgcpgw_client *mgcp,
272 uint16_t rtp_endpoint, const char *rtp_conn_addr,
273 uint16_t rtp_port, enum mgcp_connection_mode mode)
274{ return NULL; }
275int mgcpgw_client_tx(struct mgcpgw_client *mgcp, struct msgb *msg,
276 mgcp_response_cb_t response_cb, void *priv)
277{ return -EINVAL; }
278const char *mgcpgw_client_remote_addr_str(struct mgcpgw_client *mgcp)
279{ return "0.0.0.0"; }
280uint32_t mgcpgw_client_remote_addr_n(struct mgcpgw_client *mgcp)
281{ return 0; }
282int mgcp_response_parse_params(struct mgcp_response *r)
283{ return -EINVAL; }
284struct RANAP_Cause;
285int iu_tx_release(struct ue_conn_ctx *ctx, const struct RANAP_Cause *cause)
286{ return 0; }