blob: 550fb87dfe2fbfa22a26d68c94810c6b0f37b2d3 [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 Hofmeyr012a17d2020-12-08 17:36:04 +010052 self.ctrl = None
Neels Hofmeyr798e5922017-05-18 15:24:02 +020053
54 def start(self):
55 self.log('Starting osmo-msc')
Pau Espin Pedrol2a2d8462020-05-11 10:56:52 +020056 self.run_dir = util.Dir(self.testenv.test().get_run_dir().new_dir(self.name()))
Neels Hofmeyr798e5922017-05-18 15:24:02 +020057 self.configure()
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020058 inst = util.Dir(os.path.abspath(self.testenv.suite().trial().get_inst('osmo-msc')))
Neels Hofmeyr798e5922017-05-18 15:24:02 +020059 binary = inst.child('bin', 'osmo-msc')
60 if not os.path.isfile(binary):
61 raise RuntimeError('Binary missing: %r' % binary)
62 lib = inst.child('lib')
63 if not os.path.isdir(lib):
64 raise RuntimeError('No lib/ in %r' % inst)
65
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020066 pcap_recorder.PcapRecorder(self.testenv, self.run_dir.new_dir('pcap'), None,
Neels Hofmeyr798e5922017-05-18 15:24:02 +020067 'host %s and port not 22' % self.addr())
68
69 env = { 'LD_LIBRARY_PATH': util.prepend_library_path(lib) }
70
71 self.dbg(run_dir=self.run_dir, binary=binary, env=env)
72 self.process = process.Process(self.name(), self.run_dir,
73 (binary, '-c',
74 os.path.abspath(self.config_file)),
75 env=env)
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020076 self.testenv.remember_to_stop(self.process)
Neels Hofmeyr798e5922017-05-18 15:24:02 +020077 self.process.launch()
78
Neels Hofmeyr012a17d2020-12-08 17:36:04 +010079 self.ctrl = OsmoMscCtrl(self)
80 self.ctrl.connect()
81
Neels Hofmeyr798e5922017-05-18 15:24:02 +020082 def configure(self):
83 self.config_file = self.run_dir.new_file('osmo-msc.cfg')
84 self.dbg(config_file=self.config_file)
85
86 values = dict(msc=config.get_defaults('msc'))
Pau Espin Pedrola442cb82020-05-05 12:54:37 +020087 config.overlay(values, self.testenv.suite().config())
Neels Hofmeyr798e5922017-05-18 15:24:02 +020088 config.overlay(values, dict(msc=dict(ip_address=self.ip_address)))
Pau Espin Pedrol86ea02f2018-02-26 12:14:46 +010089 config.overlay(values, self.mgw.conf_for_client())
Pau Espin Pedrol5bbb3002017-11-23 11:07:02 +010090 config.overlay(values, self.hlr.conf_for_client())
Pau Espin Pedrol1e1d3812017-11-16 18:06:37 +010091 config.overlay(values, self.stp.conf_for_client())
Pau Espin Pedrol2d16f6f2017-05-30 15:33:57 +020092 config.overlay(values, self.smsc.get_config())
Pau Espin Pedrolca126b12017-08-22 19:04:06 +020093
94 # runtime parameters:
95 if self.encryption is not None:
Pau Espin Pedrolabd556a2017-09-04 16:26:08 +020096 encryption_vty = util.encryption2osmovty(self.encryption)
97 else:
98 encryption_vty = util.encryption2osmovty(values['msc']['net']['encryption'])
99 config.overlay(values, dict(msc=dict(net=dict(encryption=encryption_vty))))
Pau Espin Pedrolb5a86142017-08-23 17:39:54 +0200100 if self.authentication is not None:
101 config.overlay(values, dict(msc=dict(net=dict(authentication=self.authentication))))
Pau Espin Pedrolfed578e2019-06-19 17:18:43 +0200102 config.overlay(values, dict(msc=dict(use_osmux=self.use_osmux)))
Pau Espin Pedrol680ba032020-10-14 14:49:05 +0200103 if self._emergency_call_msisdn is not None:
104 config.overlay(values, dict(msc=dict(emergency_call_msisdn=self._emergency_call_msisdn)))
Pau Espin Pedrolca126b12017-08-22 19:04:06 +0200105
Pau Espin Pedrol0e57aad2017-05-29 14:25:22 +0200106 self.config = values
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200107
108 self.dbg('MSC CONFIG:\n' + pprint.pformat(values))
109
110 with open(self.config_file, 'w') as f:
111 r = template.render('osmo-msc.cfg', values)
112 self.dbg(r)
113 f.write(r)
114
115 def addr(self):
116 return self.ip_address.get('addr')
117
Pau Espin Pedrolca126b12017-08-22 19:04:06 +0200118 def set_encryption(self, val):
119 self.encryption = val
120
Pau Espin Pedrolb5a86142017-08-23 17:39:54 +0200121 def set_authentication(self, val):
122 if val is None:
123 self.authroziation = None
124 return
125 self.authentication = "required" if val else "optional"
126
Pau Espin Pedrolfed578e2019-06-19 17:18:43 +0200127 def set_use_osmux(self, use=False, force=False):
128 if not use:
129 self.use_osmux = "off"
130 else:
131 if not force:
132 self.use_osmux = "on"
133 else:
134 self.use_osmux = "only"
135
Pau Espin Pedrol0e57aad2017-05-29 14:25:22 +0200136 def mcc(self):
137 return self.config['msc']['net']['mcc']
138
139 def mnc(self):
140 return self.config['msc']['net']['mnc']
141
142 def mcc_mnc(self):
143 return (self.mcc(), self.mnc())
144
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200145 def subscriber_attached(self, *modems):
146 return self.imsi_attached(*[m.imsi() for m in modems])
147
148 def imsi_attached(self, *imsis):
149 attached = self.imsi_list_attached()
Neels Hofmeyr1a7a3f02017-06-10 01:18:27 +0200150 log.dbg('attached:', attached)
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200151 return all([(imsi in attached) for imsi in imsis])
152
153 def imsi_list_attached(self):
Neels Hofmeyr012a17d2020-12-08 17:36:04 +0100154 return self.ctrl.subscriber_list_active()
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200155
Pau Espin Pedrol680ba032020-10-14 14:49:05 +0200156 def set_emergency_call_msisdn(self, msisdn):
157 self.dbg('Setting Emergency Call MSISDN', msisdn=msisdn)
158 self._emergency_call_msisdn = msisdn
159
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200160 def running(self):
161 return not self.process.terminated()
162
Neels Hofmeyr012a17d2020-12-08 17:36:04 +0100163 def cleanup(self):
164 if self.ctrl is not None:
165 self.ctrl.disconnect()
166 self.ctrl = None
167
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200168
Neels Hofmeyr5b04ef22020-12-01 04:48:38 +0100169class OsmoMscCtrl(osmo_ctrl.OsmoCtrl):
170 def __init__(self, msc, port=4255):
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200171 self.msc = msc
Neels Hofmeyr5b04ef22020-12-01 04:48:38 +0100172 super().__init__(self.msc.addr(), port)
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200173
174 def subscriber_list_active(self):
Neels Hofmeyr5b04ef22020-12-01 04:48:38 +0100175 return self.get_var('subscriber-list-active-v1').replace('\n', ' ')
Neels Hofmeyr798e5922017-05-18 15:24:02 +0200176
177# vim: expandtab tabstop=4 shiftwidth=4