blob: 67e1d31eeeefe2ee28e7942b09c8813426416065 [file] [log] [blame]
Neels Hofmeyr798e5922017-05-18 15:24:02 +02001# osmo_gsm_tester: specifics for running an osmo-msc
2#
3# Copyright (C) 2016-2017 by sysmocom - s.f.m.c. GmbH
4#
5# Author: Neels Hofmeyr <neels@hofmeyr.de>
6#
7# This program is free software: you can redistribute it and/or modify
Harald Welte27205342017-06-03 09:51:45 +02008# it under the terms of the GNU General Public License as
Neels Hofmeyr798e5922017-05-18 15:24:02 +02009# published by the Free Software Foundation, either version 3 of the
10# License, or (at your option) any later version.
11#
12# This program is distributed in the hope that it will be useful,
13# but WITHOUT ANY WARRANTY; without even the implied warranty of
14# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Welte27205342017-06-03 09:51:45 +020015# GNU General Public License for more details.
Neels Hofmeyr798e5922017-05-18 15:24:02 +020016#
Harald Welte27205342017-06-03 09:51:45 +020017# You should have received a copy of the GNU General Public License
Neels Hofmeyr798e5922017-05-18 15:24:02 +020018# along with this program. If not, see <http://www.gnu.org/licenses/>.
19
20import os
21import pprint
22
Pau Espin Pedrole1a58bd2020-04-10 20:46:07 +020023from ..core import log, util, config, template, process
Pau Espin Pedrolea8c3d42020-05-04 12:05:05 +020024from ..core import schema
Pau Espin Pedrole8bbcbf2020-04-10 19:51:31 +020025from . import osmo_ctrl, pcap_recorder, smsc
Neels Hofmeyr798e5922017-05-18 15:24:02 +020026
Pau Espin Pedrolea8c3d42020-05-04 12:05:05 +020027def on_register_schemas():
28 resource_schema = {
29 'path': schema.STR,
Pau Espin Pedrol680ba032020-10-14 14:49:05 +020030 'emergency_call_msisdn': schema.MSISDN,
Pau Espin Pedrolea8c3d42020-05-04 12:05:05 +020031 }
32 schema.register_resource_schema('modem', resource_schema)
33
Neels Hofmeyr798e5922017-05-18 15:24:02 +020034class OsmoMsc(log.Origin):
Neels Hofmeyr798e5922017-05-18 15:24:02 +020035
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020036 def __init__(self, testenv, hlr, mgw, stp, ip_address):
Neels Hofmeyr1a7a3f02017-06-10 01:18:27 +020037 super().__init__(log.C_RUN, 'osmo-msc_%s' % ip_address.get('addr'))
Pau Espin Pedrol58603672018-08-09 13:45:55 +020038 self.run_dir = None
39 self.config_file = None
40 self.process = None
41 self.config = None
42 self.encryption = None
43 self.authentication = None
Pau Espin Pedrolfed578e2019-06-19 17:18:43 +020044 self.use_osmux = "off"
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020045 self.testenv = testenv
Neels Hofmeyr798e5922017-05-18 15:24:02 +020046 self.ip_address = ip_address
Pau Espin Pedrol680ba032020-10-14 14:49:05 +020047 self._emergency_call_msisdn = None
Neels Hofmeyr798e5922017-05-18 15:24:02 +020048 self.hlr = hlr
Pau Espin Pedrol86ea02f2018-02-26 12:14:46 +010049 self.mgw = mgw
Pau Espin Pedrol1e1d3812017-11-16 18:06:37 +010050 self.stp = stp
Pau Espin Pedrol2d16f6f2017-05-30 15:33:57 +020051 self.smsc = smsc.Smsc((ip_address.get('addr'), 2775))
Neels Hofmeyr798e5922017-05-18 15:24:02 +020052
53 def start(self):
54 self.log('Starting osmo-msc')
Pau Espin Pedrol2a2d8462020-05-11 10:56:52 +020055 self.run_dir = util.Dir(self.testenv.test().get_run_dir().new_dir(self.name()))
Neels Hofmeyr798e5922017-05-18 15:24:02 +020056 self.configure()
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020057 inst = util.Dir(os.path.abspath(self.testenv.suite().trial().get_inst('osmo-msc')))
Neels Hofmeyr798e5922017-05-18 15:24:02 +020058 binary = inst.child('bin', 'osmo-msc')
59 if not os.path.isfile(binary):
60 raise RuntimeError('Binary missing: %r' % binary)
61 lib = inst.child('lib')
62 if not os.path.isdir(lib):
63 raise RuntimeError('No lib/ in %r' % inst)
64
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020065 pcap_recorder.PcapRecorder(self.testenv, self.run_dir.new_dir('pcap'), None,
Neels Hofmeyr798e5922017-05-18 15:24:02 +020066 'host %s and port not 22' % self.addr())
67
68 env = { 'LD_LIBRARY_PATH': util.prepend_library_path(lib) }
69
70 self.dbg(run_dir=self.run_dir, binary=binary, env=env)
71 self.process = process.Process(self.name(), self.run_dir,
72 (binary, '-c',
73 os.path.abspath(self.config_file)),
74 env=env)
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020075 self.testenv.remember_to_stop(self.process)
Neels Hofmeyr798e5922017-05-18 15:24:02 +020076 self.process.launch()
77
78 def configure(self):
79 self.config_file = self.run_dir.new_file('osmo-msc.cfg')
80 self.dbg(config_file=self.config_file)
81
82 values = dict(msc=config.get_defaults('msc'))
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020083 config.overlay(values, self.testenv.suite().config())
Neels Hofmeyr798e5922017-05-18 15:24:02 +020084 config.overlay(values, dict(msc=dict(ip_address=self.ip_address)))
Pau Espin Pedrol86ea02f2018-02-26 12:14:46 +010085 config.overlay(values, self.mgw.conf_for_client())
Pau Espin Pedrol5bbb3002017-11-23 11:07:02 +010086 config.overlay(values, self.hlr.conf_for_client())
Pau Espin Pedrol1e1d3812017-11-16 18:06:37 +010087 config.overlay(values, self.stp.conf_for_client())
Pau Espin Pedrol2d16f6f2017-05-30 15:33:57 +020088 config.overlay(values, self.smsc.get_config())
Pau Espin Pedrolca126b12017-08-22 19:04:06 +020089
90 # runtime parameters:
91 if self.encryption is not None:
Pau Espin Pedrolabd556a2017-09-04 16:26:08 +020092 encryption_vty = util.encryption2osmovty(self.encryption)
93 else:
94 encryption_vty = util.encryption2osmovty(values['msc']['net']['encryption'])
95 config.overlay(values, dict(msc=dict(net=dict(encryption=encryption_vty))))
Pau Espin Pedrolb5a86142017-08-23 17:39:54 +020096 if self.authentication is not None:
97 config.overlay(values, dict(msc=dict(net=dict(authentication=self.authentication))))
Pau Espin Pedrolfed578e2019-06-19 17:18:43 +020098 config.overlay(values, dict(msc=dict(use_osmux=self.use_osmux)))
Pau Espin Pedrol680ba032020-10-14 14:49:05 +020099 if self._emergency_call_msisdn is not None:
100 config.overlay(values, dict(msc=dict(emergency_call_msisdn=self._emergency_call_msisdn)))
Pau Espin Pedrolca126b12017-08-22 19:04:06 +0200101
Pau Espin Pedrol0e57aad2017-05-29 14:25:22 +0200102 self.config = values
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200103
104 self.dbg('MSC CONFIG:\n' + pprint.pformat(values))
105
106 with open(self.config_file, 'w') as f:
107 r = template.render('osmo-msc.cfg', values)
108 self.dbg(r)
109 f.write(r)
110
111 def addr(self):
112 return self.ip_address.get('addr')
113
Pau Espin Pedrolca126b12017-08-22 19:04:06 +0200114 def set_encryption(self, val):
115 self.encryption = val
116
Pau Espin Pedrolb5a86142017-08-23 17:39:54 +0200117 def set_authentication(self, val):
118 if val is None:
119 self.authroziation = None
120 return
121 self.authentication = "required" if val else "optional"
122
Pau Espin Pedrolfed578e2019-06-19 17:18:43 +0200123 def set_use_osmux(self, use=False, force=False):
124 if not use:
125 self.use_osmux = "off"
126 else:
127 if not force:
128 self.use_osmux = "on"
129 else:
130 self.use_osmux = "only"
131
Pau Espin Pedrol0e57aad2017-05-29 14:25:22 +0200132 def mcc(self):
133 return self.config['msc']['net']['mcc']
134
135 def mnc(self):
136 return self.config['msc']['net']['mnc']
137
138 def mcc_mnc(self):
139 return (self.mcc(), self.mnc())
140
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200141 def subscriber_attached(self, *modems):
142 return self.imsi_attached(*[m.imsi() for m in modems])
143
144 def imsi_attached(self, *imsis):
145 attached = self.imsi_list_attached()
Neels Hofmeyr1a7a3f02017-06-10 01:18:27 +0200146 log.dbg('attached:', attached)
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200147 return all([(imsi in attached) for imsi in imsis])
148
149 def imsi_list_attached(self):
Neels Hofmeyr1a7a3f02017-06-10 01:18:27 +0200150 return OsmoMscCtrl(self).subscriber_list_active()
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200151
Pau Espin Pedrol680ba032020-10-14 14:49:05 +0200152 def set_emergency_call_msisdn(self, msisdn):
153 self.dbg('Setting Emergency Call MSISDN', msisdn=msisdn)
154 self._emergency_call_msisdn = msisdn
155
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200156 def running(self):
157 return not self.process.terminated()
158
159
160class OsmoMscCtrl(log.Origin):
161 PORT = 4255
162 SUBSCR_LIST_ACTIVE_VAR = 'subscriber-list-active-v1'
163
164 def __init__(self, msc):
165 self.msc = msc
Neels Hofmeyr1a7a3f02017-06-10 01:18:27 +0200166 super().__init__(log.C_BUS, 'CTRL(%s:%d)' % (self.msc.addr(), self.PORT))
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200167
168 def ctrl(self):
169 return osmo_ctrl.OsmoCtrl(self.msc.addr(), self.PORT)
170
171 def subscriber_list_active(self):
172 aslist_str = ""
173 with self.ctrl() as ctrl:
174 ctrl.do_get(self.SUBSCR_LIST_ACTIVE_VAR)
175 # This is legacy code from the old osmo-gsm-tester.
176 # looks like this doesn't work for long data.
177 data = ctrl.receive()
178 while (len(data) > 0):
179 (answer, data) = ctrl.remove_ipa_ctrl_header(data)
180 answer_str = answer.decode('utf-8')
181 answer_str = answer_str.replace('\n', ' ')
182 aslist_str = answer_str
183 return aslist_str
184
185# vim: expandtab tabstop=4 shiftwidth=4