blob: 6706aa44c1072f7441844b62449d9f3720fc23e7 [file] [log] [blame]
Neels Hofmeyr7685a782017-01-30 23:30:26 +01001/* OsmoHLR VTY implementation */
2
3/* (C) 2016 sysmocom s.f.m.c. GmbH <info@sysmocom.de>
Neels Hofmeyr7685a782017-01-30 23:30:26 +01004 * Author: Neels Hofmeyr <nhofmeyr@sysmocom.de>
Harald Weltefa7ee332018-06-24 13:20:32 +02005 * (C) 2018 Harald Welte <laforge@gnumonks.org>
6 *
7 * All Rights Reserved
Neels Hofmeyr7685a782017-01-30 23:30:26 +01008 *
Harald Welte4956ae12018-06-15 22:04:28 +02009 * (C) 2018 Harald Welte <laforge@gnumonks.org>
10 *
11 * All Rights Reserved
12 *
Neels Hofmeyr7685a782017-01-30 23:30:26 +010013 * This program is free software; you can redistribute it and/or modify
14 * it under the terms of the GNU Affero General Public License as published by
15 * the Free Software Foundation; either version 3 of the License, or
16 * (at your option) any later version.
17 *
18 * This program is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU Affero General Public License for more details.
22 *
23 * You should have received a copy of the GNU Affero General Public License
24 * along with this program. If not, see <http://www.gnu.org/licenses/>.
25 *
26 */
27
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +020028#include <osmocom/core/talloc.h>
Neels Hofmeyr7685a782017-01-30 23:30:26 +010029#include <osmocom/vty/vty.h>
30#include <osmocom/vty/command.h>
31#include <osmocom/vty/logging.h>
Harald Welte7ee6e552018-02-14 00:52:05 +010032#include <osmocom/vty/misc.h>
Harald Weltefa7ee332018-06-24 13:20:32 +020033#include <osmocom/abis/ipa.h>
Neels Hofmeyr7685a782017-01-30 23:30:26 +010034
Harald Weltedab544e2018-07-29 16:14:48 +020035#include "hlr.h"
Neels Hofmeyr7685a782017-01-30 23:30:26 +010036#include "hlr_vty.h"
Neels Hofmeyr183e7002017-10-06 02:59:54 +020037#include "hlr_vty_subscr.h"
Vadim Yanitskiyd157a562018-12-01 00:03:39 +070038#include "hlr_ussd.h"
Harald Weltefa7ee332018-06-24 13:20:32 +020039#include "gsup_server.h"
Neels Hofmeyr7685a782017-01-30 23:30:26 +010040
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +020041struct cmd_node hlr_node = {
42 HLR_NODE,
43 "%s(config-hlr)# ",
44 1,
45};
46
47DEFUN(cfg_hlr,
48 cfg_hlr_cmd,
49 "hlr",
50 "Configure the HLR")
51{
52 vty->node = HLR_NODE;
53 return CMD_SUCCESS;
54}
55
56struct cmd_node gsup_node = {
57 GSUP_NODE,
58 "%s(config-hlr-gsup)# ",
59 1,
60};
61
62DEFUN(cfg_gsup,
63 cfg_gsup_cmd,
64 "gsup",
65 "Configure GSUP options")
66{
67 vty->node = GSUP_NODE;
68 return CMD_SUCCESS;
69}
70
71static int config_write_hlr(struct vty *vty)
72{
73 vty_out(vty, "hlr%s", VTY_NEWLINE);
74 return CMD_SUCCESS;
75}
76
77static int config_write_hlr_gsup(struct vty *vty)
78{
79 vty_out(vty, " gsup%s", VTY_NEWLINE);
80 if (g_hlr->gsup_bind_addr)
81 vty_out(vty, " bind ip %s%s", g_hlr->gsup_bind_addr, VTY_NEWLINE);
82 return CMD_SUCCESS;
83}
84
Harald Weltefa7ee332018-06-24 13:20:32 +020085static void show_one_conn(struct vty *vty, const struct osmo_gsup_conn *conn)
86{
87 const struct ipa_server_conn *isc = conn->conn;
88 char *name;
89 int rc;
90
91 rc = osmo_gsup_conn_ccm_get(conn, (uint8_t **) &name, IPAC_IDTAG_SERNR);
92 OSMO_ASSERT(rc);
93
94 vty_out(vty, " '%s' from %s:%5u, CS=%u, PS=%u, 3G_IND=%u%s",
95 name, isc->addr, isc->port, conn->supports_cs, conn->supports_ps, conn->auc_3g_ind,
96 VTY_NEWLINE);
97}
98
99DEFUN(show_gsup_conn, show_gsup_conn_cmd,
100 "show gsup-connections",
101 SHOW_STR "GSUP Connections from VLRs, SGSNs, EUSEs\n")
102{
103 struct osmo_gsup_server *gs = g_hlr->gs;
104 struct osmo_gsup_conn *conn;
105
106 llist_for_each_entry(conn, &gs->clients, list)
107 show_one_conn(vty, conn);
108
109 return CMD_SUCCESS;
110}
111
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200112DEFUN(cfg_hlr_gsup_bind_ip,
113 cfg_hlr_gsup_bind_ip_cmd,
114 "bind ip A.B.C.D",
115 "Listen/Bind related socket option\n"
116 IP_STR
117 "IPv4 Address to bind the GSUP interface to\n")
118{
119 if(g_hlr->gsup_bind_addr)
120 talloc_free(g_hlr->gsup_bind_addr);
121 g_hlr->gsup_bind_addr = talloc_strdup(g_hlr, argv[0]);
122
123 return CMD_SUCCESS;
124}
125
Harald Welte4956ae12018-06-15 22:04:28 +0200126/***********************************************************************
Harald Weltedab544e2018-07-29 16:14:48 +0200127 * USSD Entity
Harald Welte4956ae12018-06-15 22:04:28 +0200128 ***********************************************************************/
129
130#include "hlr_ussd.h"
131
Harald Weltedab544e2018-07-29 16:14:48 +0200132#define USSD_STR "USSD Configuration\n"
133#define UROUTE_STR "Routing Configuration\n"
134#define PREFIX_STR "Prefix-Matching Route\n" "USSD Prefix\n"
Harald Welte4956ae12018-06-15 22:04:28 +0200135
Harald Weltedab544e2018-07-29 16:14:48 +0200136#define INT_CHOICE "(own-msisdn|own-imsi)"
137#define INT_STR "Internal USSD Handler\n" \
138 "Respond with subscribers' own MSISDN\n" \
139 "Respond with subscribers' own IMSI\n"
140
141#define EXT_STR "External USSD Handler\n" \
142 "Name of External USSD Handler (IPA CCM ID)\n"
143
144DEFUN(cfg_ussd_route_pfx_int, cfg_ussd_route_pfx_int_cmd,
145 "ussd route prefix PREFIX internal " INT_CHOICE,
146 USSD_STR UROUTE_STR PREFIX_STR INT_STR)
147{
148 const struct hlr_iuse *iuse = iuse_find(argv[1]);
149 struct hlr_ussd_route *rt = ussd_route_find_prefix(g_hlr, argv[0]);
Harald Welte4956ae12018-06-15 22:04:28 +0200150 if (rt) {
151 vty_out(vty, "%% Cannot add [another?] route for prefix %s%s", argv[0], VTY_NEWLINE);
152 return CMD_WARNING;
153 }
Harald Weltedab544e2018-07-29 16:14:48 +0200154 ussd_route_prefix_alloc_int(g_hlr, argv[0], iuse);
Harald Welte4956ae12018-06-15 22:04:28 +0200155
156 return CMD_SUCCESS;
157}
158
Harald Weltedab544e2018-07-29 16:14:48 +0200159DEFUN(cfg_ussd_route_pfx_ext, cfg_ussd_route_pfx_ext_cmd,
160 "ussd route prefix PREFIX external EUSE",
161 USSD_STR UROUTE_STR PREFIX_STR EXT_STR)
Harald Welte4956ae12018-06-15 22:04:28 +0200162{
Harald Weltedab544e2018-07-29 16:14:48 +0200163 struct hlr_euse *euse = euse_find(g_hlr, argv[1]);
164 struct hlr_ussd_route *rt = ussd_route_find_prefix(g_hlr, argv[0]);
165 if (rt) {
166 vty_out(vty, "%% Cannot add [another?] route for prefix %s%s", argv[0], VTY_NEWLINE);
167 return CMD_WARNING;
168 }
169 if (!euse) {
170 vty_out(vty, "%% Cannot find euse '%s'%s", argv[1], VTY_NEWLINE);
171 return CMD_WARNING;
172 }
173 ussd_route_prefix_alloc_ext(g_hlr, argv[0], euse);
174
175 return CMD_SUCCESS;
176}
177
178DEFUN(cfg_ussd_no_route_pfx, cfg_ussd_no_route_pfx_cmd,
179 "no ussd route prefix PREFIX",
180 NO_STR USSD_STR UROUTE_STR PREFIX_STR)
181{
182 struct hlr_ussd_route *rt = ussd_route_find_prefix(g_hlr, argv[0]);
Harald Welte4956ae12018-06-15 22:04:28 +0200183 if (!rt) {
184 vty_out(vty, "%% Cannot find route for prefix %s%s", argv[0], VTY_NEWLINE);
185 return CMD_WARNING;
186 }
Harald Weltedab544e2018-07-29 16:14:48 +0200187 ussd_route_del(rt);
Harald Welte4956ae12018-06-15 22:04:28 +0200188
189 return CMD_SUCCESS;
190}
191
Harald Weltedab544e2018-07-29 16:14:48 +0200192DEFUN(cfg_ussd_defaultroute, cfg_ussd_defaultroute_cmd,
193 "ussd default-route external EUSE",
194 USSD_STR "Configure default-route for all USSD to unknown destinations\n"
195 EXT_STR)
Harald Welte4956ae12018-06-15 22:04:28 +0200196{
Vadim Yanitskiyb93c44f2018-08-02 23:37:51 +0700197 struct hlr_euse *euse;
198
199 euse = euse_find(g_hlr, argv[0]);
200 if (!euse) {
201 vty_out(vty, "%% Cannot find EUSE %s%s", argv[0], VTY_NEWLINE);
202 return CMD_WARNING;
203 }
Harald Welte4956ae12018-06-15 22:04:28 +0200204
205 if (g_hlr->euse_default != euse) {
206 vty_out(vty, "Switching default route from %s to %s%s",
Harald Welte55d32a12018-07-30 17:26:35 +0200207 g_hlr->euse_default ? g_hlr->euse_default->name : "<none>",
208 euse->name, VTY_NEWLINE);
Harald Welte4956ae12018-06-15 22:04:28 +0200209 g_hlr->euse_default = euse;
210 }
211
212 return CMD_SUCCESS;
213}
214
Harald Weltedab544e2018-07-29 16:14:48 +0200215DEFUN(cfg_ussd_no_defaultroute, cfg_ussd_no_defaultroute_cmd,
216 "no ussd default-route",
217 NO_STR USSD_STR "Remove the default-route for all USSD to unknown destinations\n")
Harald Welte4956ae12018-06-15 22:04:28 +0200218{
Harald Welte4956ae12018-06-15 22:04:28 +0200219 g_hlr->euse_default = NULL;
220
221 return CMD_SUCCESS;
222}
223
224struct cmd_node euse_node = {
225 EUSE_NODE,
226 "%s(config-hlr-euse)# ",
227 1,
228};
229
230DEFUN(cfg_euse, cfg_euse_cmd,
231 "euse NAME",
232 "Configure a particular External USSD Entity\n"
233 "Alphanumeric name of the External USSD Entity\n")
234{
235 struct hlr_euse *euse;
236 const char *id = argv[0];
237
238 euse = euse_find(g_hlr, id);
239 if (!euse) {
240 euse = euse_alloc(g_hlr, id);
241 if (!euse)
242 return CMD_WARNING;
243 }
244 vty->index = euse;
245 vty->index_sub = &euse->description;
246 vty->node = EUSE_NODE;
247
248 return CMD_SUCCESS;
249}
250
251DEFUN(cfg_no_euse, cfg_no_euse_cmd,
252 "no euse NAME",
253 NO_STR "Remove a particular External USSD Entity\n"
254 "Alphanumeric name of the External USSD Entity\n")
255{
256 struct hlr_euse *euse = euse_find(g_hlr, argv[0]);
257 if (!euse) {
258 vty_out(vty, "%% Cannot remove non-existant EUSE %s%s", argv[0], VTY_NEWLINE);
259 return CMD_WARNING;
260 }
261 if (g_hlr->euse_default == euse) {
262 vty_out(vty, "%% Cannot remove EUSE %s, it is the default route%s", argv[0], VTY_NEWLINE);
263 return CMD_WARNING;
264 }
265 euse_del(euse);
266 return CMD_SUCCESS;
267}
268
269static void dump_one_euse(struct vty *vty, struct hlr_euse *euse)
270{
Harald Welte4956ae12018-06-15 22:04:28 +0200271 vty_out(vty, " euse %s%s", euse->name, VTY_NEWLINE);
Harald Welte4956ae12018-06-15 22:04:28 +0200272}
273
274static int config_write_euse(struct vty *vty)
275{
276 struct hlr_euse *euse;
Harald Weltedab544e2018-07-29 16:14:48 +0200277 struct hlr_ussd_route *rt;
Harald Welte4956ae12018-06-15 22:04:28 +0200278
279 llist_for_each_entry(euse, &g_hlr->euse_list, list)
280 dump_one_euse(vty, euse);
281
Harald Weltedab544e2018-07-29 16:14:48 +0200282 llist_for_each_entry(rt, &g_hlr->ussd_routes, list) {
283 vty_out(vty, " ussd route prefix %s %s %s%s", rt->prefix,
284 rt->is_external ? "external" : "internal",
285 rt->is_external ? rt->u.euse->name : rt->u.iuse->name,
286 VTY_NEWLINE);
287 }
288
289 if (g_hlr->euse_default)
290 vty_out(vty, " ussd default-route external %s%s", g_hlr->euse_default->name, VTY_NEWLINE);
291
Vadim Yanitskiyd157a562018-12-01 00:03:39 +0700292 if (g_hlr->ncss_guard_timeout != NCSS_GUARD_TIMEOUT_DEFAULT)
293 vty_out(vty, " ncss-guard-timeout %i%s",
294 g_hlr->ncss_guard_timeout, VTY_NEWLINE);
295
Harald Welte4956ae12018-06-15 22:04:28 +0200296 return 0;
297}
298
Vadim Yanitskiyd157a562018-12-01 00:03:39 +0700299DEFUN(cfg_ncss_guard_timeout, cfg_ncss_guard_timeout_cmd,
300 "ncss-guard-timeout <0-255>",
301 "Set guard timer for NCSS (call independent SS) session activity\n"
302 "Guard timer value (sec.), or 0 to disable")
303{
304 g_hlr->ncss_guard_timeout = atoi(argv[0]);
305 return CMD_SUCCESS;
306}
307
Harald Welte4956ae12018-06-15 22:04:28 +0200308/***********************************************************************
309 * Common Code
310 ***********************************************************************/
311
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200312int hlr_vty_go_parent(struct vty *vty)
313{
314 switch (vty->node) {
315 case GSUP_NODE:
Harald Welte4956ae12018-06-15 22:04:28 +0200316 case EUSE_NODE:
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200317 vty->node = HLR_NODE;
318 vty->index = NULL;
Harald Welte4956ae12018-06-15 22:04:28 +0200319 vty->index_sub = NULL;
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200320 break;
321 default:
322 case HLR_NODE:
323 vty->node = CONFIG_NODE;
324 vty->index = NULL;
325 break;
326 case CONFIG_NODE:
327 vty->node = ENABLE_NODE;
328 vty->index = NULL;
329 break;
330 }
331
332 return vty->node;
333}
334
Neels Hofmeyr7685a782017-01-30 23:30:26 +0100335int hlr_vty_is_config_node(struct vty *vty, int node)
336{
337 switch (node) {
338 /* add items that are not config */
339 case CONFIG_NODE:
340 return 0;
341
342 default:
343 return 1;
344 }
345}
346
Harald Welted5807b82018-07-29 12:27:41 +0200347void hlr_vty_init(const struct log_info *cat)
Neels Hofmeyr7685a782017-01-30 23:30:26 +0100348{
349 logging_vty_add_cmds(cat);
Harald Welte7ee6e552018-02-14 00:52:05 +0100350 osmo_talloc_vty_add_cmds();
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200351
Harald Weltefa7ee332018-06-24 13:20:32 +0200352 install_element_ve(&show_gsup_conn_cmd);
353
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200354 install_element(CONFIG_NODE, &cfg_hlr_cmd);
355 install_node(&hlr_node, config_write_hlr);
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200356
357 install_element(HLR_NODE, &cfg_gsup_cmd);
358 install_node(&gsup_node, config_write_hlr_gsup);
Pau Espin Pedrolce9bc402017-05-31 13:19:22 +0200359
360 install_element(GSUP_NODE, &cfg_hlr_gsup_bind_ip_cmd);
Neels Hofmeyr183e7002017-10-06 02:59:54 +0200361
Harald Welte4956ae12018-06-15 22:04:28 +0200362 install_element(HLR_NODE, &cfg_euse_cmd);
363 install_element(HLR_NODE, &cfg_no_euse_cmd);
364 install_node(&euse_node, config_write_euse);
Harald Weltedab544e2018-07-29 16:14:48 +0200365 install_element(HLR_NODE, &cfg_ussd_route_pfx_int_cmd);
366 install_element(HLR_NODE, &cfg_ussd_route_pfx_ext_cmd);
367 install_element(HLR_NODE, &cfg_ussd_no_route_pfx_cmd);
368 install_element(HLR_NODE, &cfg_ussd_defaultroute_cmd);
369 install_element(HLR_NODE, &cfg_ussd_no_defaultroute_cmd);
Vadim Yanitskiyd157a562018-12-01 00:03:39 +0700370 install_element(HLR_NODE, &cfg_ncss_guard_timeout_cmd);
Harald Welte4956ae12018-06-15 22:04:28 +0200371
Harald Welted5807b82018-07-29 12:27:41 +0200372 hlr_vty_subscriber_init();
Neels Hofmeyr7685a782017-01-30 23:30:26 +0100373}