2008-12-27 01:07:07 +00:00
|
|
|
/* (C) 2008 by Jan Luebbe <jluebbe@debian.org>
|
2016-04-01 18:21:03 +00:00
|
|
|
* (C) 2009-2016 by Holger Hans Peter Freyther <zecke@selfish.org>
|
2014-03-08 17:48:49 +00:00
|
|
|
* (C) 2014 by Alexander Chemeris <Alexander.Chemeris@fairwaves.co>
|
2008-12-27 01:07:07 +00:00
|
|
|
* All Rights Reserved
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
2011-01-01 14:25:50 +00:00
|
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
2008-12-27 01:07:07 +00:00
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
2011-01-01 14:25:50 +00:00
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2008-12-27 01:07:07 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-09-04 13:04:35 +00:00
|
|
|
#include <osmocom/msc/debug.h>
|
|
|
|
#include <osmocom/msc/db.h>
|
|
|
|
#include <osmocom/msc/gsm_subscriber.h>
|
|
|
|
#include <osmocom/msc/gsm_04_11.h>
|
|
|
|
#include <osmocom/msc/mgcp.h>
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2012-01-06 13:27:08 +00:00
|
|
|
#include <osmocom/core/application.h>
|
|
|
|
|
2008-12-27 00:45:41 +00:00
|
|
|
#include <stdio.h>
|
2008-12-27 15:59:25 +00:00
|
|
|
#include <string.h>
|
2009-05-21 07:23:02 +00:00
|
|
|
#include <stdlib.h>
|
2016-06-30 08:25:49 +00:00
|
|
|
#include <stdbool.h>
|
2014-03-08 17:48:49 +00:00
|
|
|
#include <inttypes.h>
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2012-01-06 13:27:08 +00:00
|
|
|
static struct gsm_network dummy_net;
|
2014-12-03 08:28:24 +00:00
|
|
|
static struct gsm_subscriber_group dummy_sgrp;
|
2012-01-06 13:27:08 +00:00
|
|
|
|
|
|
|
#define SUBSCR_PUT(sub) \
|
2014-12-03 08:28:24 +00:00
|
|
|
sub->group = &dummy_sgrp; \
|
2012-01-06 13:27:08 +00:00
|
|
|
subscr_put(sub);
|
|
|
|
|
2009-01-01 18:02:05 +00:00
|
|
|
#define COMPARE(original, copy) \
|
|
|
|
if (original->id != copy->id) \
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("Ids do not match in %s:%d %llu %llu\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->id, copy->id); \
|
|
|
|
if (original->lac != copy->lac) \
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("LAC do not match in %s:%d %d %d\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->lac, copy->lac); \
|
|
|
|
if (original->authorized != copy->authorized) \
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("Authorize do not match in %s:%d %d %d\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->authorized, \
|
|
|
|
copy->authorized); \
|
|
|
|
if (strcmp(original->imsi, copy->imsi) != 0) \
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("IMSIs do not match in %s:%d '%s' '%s'\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->imsi, copy->imsi); \
|
2009-08-19 10:53:57 +00:00
|
|
|
if (original->tmsi != copy->tmsi) \
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("TMSIs do not match in %s:%d '%u' '%u'\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->tmsi, copy->tmsi); \
|
|
|
|
if (strcmp(original->name, copy->name) != 0) \
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("names do not match in %s:%d '%s' '%s'\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->name, copy->name); \
|
|
|
|
if (strcmp(original->extension, copy->extension) != 0) \
|
2013-10-04 00:42:24 +00:00
|
|
|
printf("Extensions do not match in %s:%d '%s' '%s'\n", \
|
2009-01-01 18:02:05 +00:00
|
|
|
__FUNCTION__, __LINE__, original->extension, copy->extension); \
|
|
|
|
|
2014-03-07 18:08:03 +00:00
|
|
|
/*
|
|
|
|
* Create/Store a SMS and then try to load it.
|
|
|
|
*/
|
|
|
|
static void test_sms(void)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
struct gsm_sms *sms;
|
|
|
|
struct gsm_subscriber *subscr;
|
|
|
|
subscr = db_get_subscriber(GSM_SUBSCRIBER_IMSI, "9993245423445");
|
|
|
|
OSMO_ASSERT(subscr);
|
2014-12-03 08:28:24 +00:00
|
|
|
subscr->group = &dummy_sgrp;
|
2014-03-07 18:08:03 +00:00
|
|
|
|
|
|
|
sms = sms_alloc();
|
|
|
|
sms->receiver = subscr_get(subscr);
|
|
|
|
|
|
|
|
sms->src.ton = 0x23;
|
|
|
|
sms->src.npi = 0x24;
|
|
|
|
memcpy(sms->src.addr, "1234", strlen("1234") + 1);
|
|
|
|
|
|
|
|
sms->dst.ton = 0x32;
|
|
|
|
sms->dst.npi = 0x42;
|
|
|
|
memcpy(sms->dst.addr, subscr->extension, sizeof(subscr->extension));
|
|
|
|
|
|
|
|
memcpy(sms->text, "Text123", strlen("Text123") + 1);
|
|
|
|
memcpy(sms->user_data, "UserData123", strlen("UserData123") + 1);
|
|
|
|
sms->user_data_len = strlen("UserData123");
|
|
|
|
|
|
|
|
/* random values */
|
|
|
|
sms->reply_path_req = 1;
|
|
|
|
sms->status_rep_req = 2;
|
|
|
|
sms->ud_hdr_ind = 3;
|
|
|
|
sms->protocol_id = 4;
|
|
|
|
sms->data_coding_scheme = 5;
|
|
|
|
|
|
|
|
rc = db_sms_store(sms);
|
|
|
|
sms_free(sms);
|
|
|
|
OSMO_ASSERT(rc == 0);
|
|
|
|
|
|
|
|
/* now query */
|
|
|
|
sms = db_sms_get_unsent_for_subscr(subscr);
|
|
|
|
OSMO_ASSERT(sms);
|
|
|
|
OSMO_ASSERT(sms->receiver == subscr);
|
|
|
|
OSMO_ASSERT(sms->reply_path_req == 1);
|
|
|
|
OSMO_ASSERT(sms->status_rep_req == 2);
|
|
|
|
OSMO_ASSERT(sms->ud_hdr_ind == 3);
|
|
|
|
OSMO_ASSERT(sms->protocol_id == 4);
|
|
|
|
OSMO_ASSERT(sms->data_coding_scheme == 5);
|
|
|
|
OSMO_ASSERT(sms->src.ton == 0x23);
|
|
|
|
OSMO_ASSERT(sms->src.npi == 0x24);
|
|
|
|
OSMO_ASSERT(sms->dst.ton == 0x32);
|
|
|
|
OSMO_ASSERT(sms->dst.npi == 0x42);
|
|
|
|
OSMO_ASSERT(strcmp((char *) sms->text, "Text123") == 0);
|
|
|
|
OSMO_ASSERT(sms->user_data_len == strlen("UserData123"));
|
|
|
|
OSMO_ASSERT(strcmp((char *) sms->user_data, "UserData123") == 0);
|
|
|
|
|
2014-12-25 16:27:15 +00:00
|
|
|
/* Mark the SMS as delivered */
|
|
|
|
db_sms_mark_delivered(sms);
|
|
|
|
sms_free(sms);
|
|
|
|
|
|
|
|
sms = db_sms_get_unsent_for_subscr(subscr);
|
|
|
|
OSMO_ASSERT(!sms);
|
|
|
|
|
2014-03-07 18:08:03 +00:00
|
|
|
subscr_put(subscr);
|
|
|
|
}
|
|
|
|
|
2014-03-08 15:41:37 +00:00
|
|
|
static void test_sms_migrate(void)
|
|
|
|
{
|
|
|
|
struct gsm_subscriber *rcv_subscr;
|
|
|
|
struct gsm_sms *sms;
|
|
|
|
static const uint8_t user_data_1[] = {
|
|
|
|
0x41, 0xf1, 0xd8, 0x05, 0x22, 0x96, 0xcd, 0x2e,
|
|
|
|
0x90, 0xf1, 0xfd, 0x06, 0x00 };
|
|
|
|
static const uint8_t user_data_2[] = {
|
|
|
|
0x41, 0xf1, 0xd8, 0x05, 0x22, 0x96, 0xcd, 0x2e,
|
|
|
|
0xd0, 0xf1, 0xfd, 0x06, 0x00 };
|
|
|
|
|
|
|
|
rcv_subscr = db_get_subscriber(GSM_SUBSCRIBER_IMSI, "901010000001111");
|
2014-12-03 08:28:24 +00:00
|
|
|
rcv_subscr->group = &dummy_sgrp;
|
2014-03-08 15:41:37 +00:00
|
|
|
|
|
|
|
sms = db_sms_get(&dummy_net, 1);
|
|
|
|
OSMO_ASSERT(sms->id == 1);
|
|
|
|
OSMO_ASSERT(sms->receiver == rcv_subscr);
|
|
|
|
OSMO_ASSERT(strcmp(sms->text, "Abc. Def. Foo") == 0);
|
|
|
|
OSMO_ASSERT(sms->user_data_len == ARRAY_SIZE(user_data_1));
|
|
|
|
OSMO_ASSERT(memcmp(sms->user_data, user_data_1, ARRAY_SIZE(user_data_1)) == 0);
|
|
|
|
sms_free(sms);
|
|
|
|
|
|
|
|
sms = db_sms_get(&dummy_net, 2);
|
|
|
|
OSMO_ASSERT(sms->id == 2);
|
|
|
|
OSMO_ASSERT(sms->receiver == rcv_subscr);
|
|
|
|
OSMO_ASSERT(strcmp(sms->text, "Abc. Def. Goo") == 0);
|
|
|
|
OSMO_ASSERT(sms->user_data_len == ARRAY_SIZE(user_data_2));
|
|
|
|
OSMO_ASSERT(memcmp(sms->user_data, user_data_2, ARRAY_SIZE(user_data_2)) == 0);
|
|
|
|
sms_free(sms);
|
|
|
|
|
|
|
|
subscr_put(rcv_subscr);
|
|
|
|
}
|
|
|
|
|
2016-06-30 08:25:49 +00:00
|
|
|
static void test_subs(const char *imsi, char *imei1, char *imei2, bool make_ext)
|
2012-01-06 13:27:08 +00:00
|
|
|
{
|
2016-05-11 11:05:13 +00:00
|
|
|
struct gsm_subscriber *alice = NULL, *alice_db;
|
2014-03-08 17:48:49 +00:00
|
|
|
char scratch_str[256];
|
|
|
|
|
2016-06-30 08:25:49 +00:00
|
|
|
alice = db_create_subscriber(imsi, GSM_MIN_EXTEN, GSM_MAX_EXTEN,
|
|
|
|
make_ext);
|
2016-05-11 11:05:13 +00:00
|
|
|
db_subscriber_assoc_imei(alice, imei1);
|
|
|
|
if (imei2)
|
|
|
|
db_subscriber_assoc_imei(alice, imei2);
|
|
|
|
db_subscriber_alloc_tmsi(alice);
|
|
|
|
alice->lac=42;
|
|
|
|
db_sync_subscriber(alice);
|
|
|
|
/* Get by TMSI */
|
|
|
|
snprintf(scratch_str, sizeof(scratch_str), "%"PRIu32, alice->tmsi);
|
|
|
|
alice_db = db_get_subscriber(GSM_SUBSCRIBER_TMSI, scratch_str);
|
|
|
|
COMPARE(alice, alice_db);
|
|
|
|
SUBSCR_PUT(alice_db);
|
|
|
|
/* Get by IMSI */
|
2016-06-30 08:25:49 +00:00
|
|
|
alice_db = db_get_subscriber(GSM_SUBSCRIBER_IMSI, imsi);
|
2016-05-11 11:05:13 +00:00
|
|
|
COMPARE(alice, alice_db);
|
|
|
|
SUBSCR_PUT(alice_db);
|
|
|
|
/* Get by id */
|
|
|
|
snprintf(scratch_str, sizeof(scratch_str), "%llu", alice->id);
|
|
|
|
alice_db = db_get_subscriber(GSM_SUBSCRIBER_ID, scratch_str);
|
|
|
|
COMPARE(alice, alice_db);
|
|
|
|
SUBSCR_PUT(alice_db);
|
|
|
|
/* Get by extension */
|
|
|
|
alice_db = db_get_subscriber(GSM_SUBSCRIBER_EXTENSION, alice->extension);
|
2016-06-30 08:25:49 +00:00
|
|
|
if (alice_db) {
|
|
|
|
if (!make_ext)
|
|
|
|
printf("FAIL: bogus extension created for IMSI %s\n",
|
|
|
|
imsi);
|
|
|
|
COMPARE(alice, alice_db);
|
|
|
|
SUBSCR_PUT(alice_db);
|
|
|
|
} else if (make_ext)
|
|
|
|
printf("FAIL: no subscriber extension for IMSI %s\n", imsi);
|
2016-05-11 11:05:13 +00:00
|
|
|
SUBSCR_PUT(alice);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("Testing subscriber database code.\n");
|
|
|
|
osmo_init_logging(&log_info);
|
2014-05-01 05:53:42 +00:00
|
|
|
log_set_print_filename(osmo_stderr_target, 0);
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2014-12-03 08:28:24 +00:00
|
|
|
dummy_net.subscr_group = &dummy_sgrp;
|
|
|
|
dummy_sgrp.net = &dummy_net;
|
|
|
|
|
2009-06-06 13:54:20 +00:00
|
|
|
if (db_init("hlr.sqlite3")) {
|
2008-12-27 15:59:25 +00:00
|
|
|
printf("DB: Failed to init database. Please check the option settings.\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf("DB: Database initialized.\n");
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2008-12-27 15:59:25 +00:00
|
|
|
if (db_prepare()) {
|
|
|
|
printf("DB: Failed to prepare database.\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf("DB: Database prepared.\n");
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2008-12-27 15:59:25 +00:00
|
|
|
struct gsm_subscriber *alice = NULL;
|
2009-01-01 18:02:05 +00:00
|
|
|
struct gsm_subscriber *alice_db;
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2009-01-01 18:02:05 +00:00
|
|
|
char *alice_imsi = "3243245432345";
|
2016-06-30 08:25:49 +00:00
|
|
|
alice = db_create_subscriber(alice_imsi, GSM_MIN_EXTEN, GSM_MAX_EXTEN,
|
|
|
|
true);
|
2009-01-01 18:02:05 +00:00
|
|
|
db_sync_subscriber(alice);
|
2013-10-04 06:35:11 +00:00
|
|
|
alice_db = db_get_subscriber(GSM_SUBSCRIBER_IMSI, alice->imsi);
|
2009-01-01 18:02:05 +00:00
|
|
|
COMPARE(alice, alice_db);
|
2012-01-06 13:27:08 +00:00
|
|
|
SUBSCR_PUT(alice_db);
|
|
|
|
SUBSCR_PUT(alice);
|
2008-12-27 00:45:41 +00:00
|
|
|
|
2016-06-30 08:25:49 +00:00
|
|
|
test_subs("3693245423445", "1234567890", NULL, true);
|
|
|
|
test_subs("9993245423445", "1234567890", "6543560920", true);
|
|
|
|
test_subs("3123122223445", "1234567890", NULL, false);
|
|
|
|
test_subs("9123121223445", "1234567890", "6543560920", false);
|
2008-12-27 15:59:25 +00:00
|
|
|
|
2016-04-01 18:21:03 +00:00
|
|
|
/* create it again and see it fails */
|
2016-06-30 08:25:49 +00:00
|
|
|
alice = db_create_subscriber(alice_imsi, GSM_MIN_EXTEN, GSM_MAX_EXTEN,
|
|
|
|
true);
|
2016-04-01 18:21:03 +00:00
|
|
|
OSMO_ASSERT(!alice);
|
|
|
|
|
2014-03-07 18:08:03 +00:00
|
|
|
test_sms();
|
2014-03-08 15:41:37 +00:00
|
|
|
test_sms_migrate();
|
2014-03-07 18:08:03 +00:00
|
|
|
|
2008-12-27 15:59:25 +00:00
|
|
|
db_fini();
|
|
|
|
|
2012-01-06 13:27:08 +00:00
|
|
|
printf("Done\n");
|
2008-12-27 15:59:25 +00:00
|
|
|
return 0;
|
2008-12-27 00:45:41 +00:00
|
|
|
}
|
2009-04-12 05:37:07 +00:00
|
|
|
|
|
|
|
/* stubs */
|
2010-12-29 10:07:22 +00:00
|
|
|
void vty_out() {}
|
2016-06-19 16:06:02 +00:00
|
|
|
void vlr_subscr_disconnected() {}
|
|
|
|
void vlr_subscr_rx_tmsi_reall_compl() {}
|
|
|
|
void vlr_subscr_rx_id_resp() {}
|
|
|
|
void vlr_subscr_rx_auth_resp() {}
|
|
|
|
void vlr_loc_update() {}
|
|
|
|
void vlr_proc_acc_req() {}
|
|
|
|
void vlr_init() {}
|
2016-05-20 19:59:55 +00:00
|
|
|
unsigned int mgcpgw_client_next_endpoint(struct mgcpgw_client *client)
|
|
|
|
{ return 0; }
|
|
|
|
struct msgb *mgcp_msg_crcx(struct mgcpgw_client *mgcp,
|
|
|
|
uint16_t rtp_endpoint, unsigned int call_id,
|
|
|
|
enum mgcp_connection_mode mode)
|
|
|
|
{ return NULL; }
|
|
|
|
struct msgb *mgcp_msg_mdcx(struct mgcpgw_client *mgcp,
|
|
|
|
uint16_t rtp_endpoint, const char *rtp_conn_addr,
|
|
|
|
uint16_t rtp_port, enum mgcp_connection_mode mode)
|
|
|
|
{ return NULL; }
|
|
|
|
int mgcpgw_client_tx(struct mgcpgw_client *mgcp, struct msgb *msg,
|
|
|
|
mgcp_response_cb_t response_cb, void *priv)
|
|
|
|
{ return -EINVAL; }
|
|
|
|
const char *mgcpgw_client_remote_addr_str(struct mgcpgw_client *mgcp)
|
|
|
|
{ return "0.0.0.0"; }
|
|
|
|
uint32_t mgcpgw_client_remote_addr_n(struct mgcpgw_client *mgcp)
|
|
|
|
{ return 0; }
|
|
|
|
int mgcp_response_parse_params(struct mgcp_response *r)
|
|
|
|
{ return -EINVAL; }
|
|
|
|
struct RANAP_Cause;
|
|
|
|
int iu_tx_release(struct ue_conn_ctx *ctx, const struct RANAP_Cause *cause)
|
|
|
|
{ return 0; }
|