blob: bed685adaff241cf26b6d115a2f13d4a5828dddc [file] [log] [blame]
Jacob Erlbeck0760a832013-09-16 11:20:28 +02001#!/usr/bin/env python
2
3# (C) 2013 by Jacob Erlbeck <jerlbeck@sysmocom.de>
Holger Hans Peter Freyther3adb7722014-03-04 17:16:58 +01004# (C) 2014 by Holger Hans Peter Freyther
Jacob Erlbeck0760a832013-09-16 11:20:28 +02005# based on vty_test_runner.py:
6# (C) 2013 by Katerina Barone-Adesi <kat.obsc@gmail.com>
7# (C) 2013 by Holger Hans Peter Freyther
8# based on bsc_control.py.
9
10# This program is free software: you can redistribute it and/or modify
11# it under the terms of the GNU General Public License as published by
12# the Free Software Foundation, either version 3 of the License, or
13# (at your option) any later version.
14
15# This program is distributed in the hope that it will be useful,
16# but WITHOUT ANY WARRANTY; without even the implied warranty of
17# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18# GNU General Public License for more details.
19
20# You should have received a copy of the GNU General Public License
21# along with this program. If not, see <http://www.gnu.org/licenses/>.
22
23import os
24import time
25import unittest
26import socket
27import sys
28import struct
29
30import osmopy.obscvty as obscvty
31import osmopy.osmoutil as osmoutil
32
33confpath = '.'
34verbose = False
35
36class TestCtrlBase(unittest.TestCase):
37
38 def ctrl_command(self):
39 raise Exception("Needs to be implemented by a subclass")
40
41 def ctrl_app(self):
42 raise Exception("Needs to be implemented by a subclass")
43
44 def setUp(self):
45 osmo_ctrl_cmd = self.ctrl_command()[:]
46 config_index = osmo_ctrl_cmd.index('-c')
47 if config_index:
48 cfi = config_index + 1
49 osmo_ctrl_cmd[cfi] = os.path.join(confpath, osmo_ctrl_cmd[cfi])
50
51 try:
52 print "Launch: %s from %s" % (' '.join(osmo_ctrl_cmd), os.getcwd())
53 self.proc = osmoutil.popen_devnull(osmo_ctrl_cmd)
54 except OSError:
55 print >> sys.stderr, "Current directory: %s" % os.getcwd()
56 print >> sys.stderr, "Consider setting -b"
57 time.sleep(2)
58
59 appstring = self.ctrl_app()[2]
60 appport = self.ctrl_app()[0]
61 self.connect("127.0.0.1", appport)
62 self.next_id = 1000
63
64 def tearDown(self):
65 self.disconnect()
66 osmoutil.end_proc(self.proc)
67
68 def prefix_ipa_ctrl_header(self, data):
69 return struct.pack(">HBB", len(data)+1, 0xee, 0) + data
70
71 def remove_ipa_ctrl_header(self, data):
72 if (len(data) < 4):
73 raise BaseException("Answer too short!")
74 (plen, ipa_proto, osmo_proto) = struct.unpack(">HBB", data[:4])
75 if (plen + 3 > len(data)):
76 print "Warning: Wrong payload length (expected %i, got %i)" % (plen, len(data) - 3)
77 if (ipa_proto != 0xee or osmo_proto != 0):
78 raise BaseException("Wrong protocol in answer!")
79
80 return data[4:plen+3], data[plen+3:]
81
82 def disconnect(self):
83 if not (self.sock is None):
84 self.sock.close()
85
86 def connect(self, host, port):
87 if verbose:
88 print "Connecting to host %s:%i" % (host, port)
89
90 sck = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
91 sck.setblocking(1)
92 sck.connect((host, port))
93 self.sock = sck
94 return sck
95
96 def send(self, data):
97 if verbose:
98 print "Sending \"%s\"" %(data)
99 data = self.prefix_ipa_ctrl_header(data)
100 return self.sock.send(data) == len(data)
101
102 def send_set(self, var, value, id):
103 setmsg = "SET %s %s %s" %(id, var, value)
104 return self.send(setmsg)
105
106 def send_get(self, var, id):
107 getmsg = "GET %s %s" %(id, var)
108 return self.send(getmsg)
109
110 def do_set(self, var, value):
111 id = self.next_id
112 self.next_id += 1
113 self.send_set(var, value, id)
114 return self.recv_msgs()[id]
115
116 def do_get(self, var):
117 id = self.next_id
118 self.next_id += 1
119 self.send_get(var, id)
120 return self.recv_msgs()[id]
121
122 def recv_msgs(self):
123 responses = {}
124 data = self.sock.recv(4096)
125 while (len(data)>0):
126 (answer, data) = self.remove_ipa_ctrl_header(data)
127 if verbose:
128 print "Got message:", answer
129 (mtype, id, msg) = answer.split(None, 2)
130 id = int(id)
131 rsp = {'mtype': mtype, 'id': id}
132 if mtype == "ERROR":
133 rsp['error'] = msg
134 else:
Holger Hans Peter Freyther4ecc6872014-03-04 15:38:00 +0100135 split = msg.split(None, 1)
136 rsp['var'] = split[0]
137 if len(split) > 1:
138 rsp['value'] = split[1]
139 else:
140 rsp['value'] = None
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200141 responses[id] = rsp
142
143 if verbose:
144 print "Decoded replies: ", responses
145
146 return responses
147
148
149class TestCtrlBSC(TestCtrlBase):
150
151 def tearDown(self):
152 TestCtrlBase.tearDown(self)
153 os.unlink("tmp_dummy_sock")
154
155 def ctrl_command(self):
156 return ["./src/osmo-bsc/osmo-bsc", "-r", "tmp_dummy_sock", "-c",
157 "doc/examples/osmo-bsc/osmo-bsc.cfg"]
158
159 def ctrl_app(self):
160 return (4249, "./src/osmo-bsc/osmo-bsc", "OsmoBSC", "bsc")
161
162 def testCtrlErrs(self):
163 r = self.do_get('invalid')
164 self.assertEquals(r['mtype'], 'ERROR')
165 self.assertEquals(r['error'], 'Command not found')
166
Jacob Erlbeck4f13d032013-09-16 11:20:29 +0200167 r = self.do_set('rf_locked', '999')
168 self.assertEquals(r['mtype'], 'ERROR')
169 self.assertEquals(r['error'], 'Value failed verification.')
170
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200171 r = self.do_get('bts')
172 self.assertEquals(r['mtype'], 'ERROR')
173 self.assertEquals(r['error'], 'Error while parsing the index.')
174
175 r = self.do_get('bts.999')
176 self.assertEquals(r['mtype'], 'ERROR')
177 self.assertEquals(r['error'], 'Error while resolving object')
178
Holger Hans Peter Freyther054bc242014-11-10 11:41:03 +0100179 def testBtsLac(self):
180 r = self.do_get('bts.0.location-area-code')
181 self.assertEquals(r['mtype'], 'GET_REPLY')
182 self.assertEquals(r['var'], 'bts.0.location-area-code')
183 self.assertEquals(r['value'], '1')
184
185 r = self.do_set('bts.0.location-area-code', '23')
186 self.assertEquals(r['mtype'], 'SET_REPLY')
187 self.assertEquals(r['var'], 'bts.0.location-area-code')
188 self.assertEquals(r['value'], '23')
189
190 r = self.do_get('bts.0.location-area-code')
191 self.assertEquals(r['mtype'], 'GET_REPLY')
192 self.assertEquals(r['var'], 'bts.0.location-area-code')
193 self.assertEquals(r['value'], '23')
194
195 r = self.do_set('bts.0.location-area-code', '-1')
196 self.assertEquals(r['mtype'], 'ERROR')
197 self.assertEquals(r['error'], 'Input not within the range')
198
Holger Hans Peter Freyther8a641412014-11-21 10:54:42 +0100199 def testBtsCi(self):
200 r = self.do_get('bts.0.cell-identity')
201 self.assertEquals(r['mtype'], 'GET_REPLY')
202 self.assertEquals(r['var'], 'bts.0.cell-identity')
203 self.assertEquals(r['value'], '0')
204
205 r = self.do_set('bts.0.cell-identity', '23')
206 self.assertEquals(r['mtype'], 'SET_REPLY')
207 self.assertEquals(r['var'], 'bts.0.cell-identity')
208 self.assertEquals(r['value'], '23')
209
210 r = self.do_get('bts.0.cell-identity')
211 self.assertEquals(r['mtype'], 'GET_REPLY')
212 self.assertEquals(r['var'], 'bts.0.cell-identity')
213 self.assertEquals(r['value'], '23')
214
215 r = self.do_set('bts.0.cell-identity', '-1')
216 self.assertEquals(r['mtype'], 'ERROR')
217 self.assertEquals(r['error'], 'Input not within the range')
218
Holger Hans Peter Freythera49b2c02014-11-21 11:18:45 +0100219 def testBtsGenerateSystemInformation(self):
220 r = self.do_get('bts.0.send-new-system-informations')
221 self.assertEquals(r['mtype'], 'ERROR')
222 self.assertEquals(r['error'], 'Write only attribute')
223
224 # No RSL links so it will fail
225 r = self.do_set('bts.0.send-new-system-informations', '1')
226 self.assertEquals(r['mtype'], 'ERROR')
227 self.assertEquals(r['error'], 'Failed to generate SI')
228
Holger Hans Peter Freyther5eebb7a2014-12-05 12:03:24 +0100229 def testBtsChannelLoad(self):
230 r = self.do_set('bts.0.channel-load', '1')
231 self.assertEquals(r['mtype'], 'ERROR')
232 self.assertEquals(r['error'], 'Read only attribute')
233
234 # No RSL link so everything is 0
235 r = self.do_get('bts.0.channel-load')
236 self.assertEquals(r['mtype'], 'GET_REPLY')
Neels Hofmeyr17a6bab2016-07-23 16:14:06 +0200237 self.assertEquals(r['value'],
238 'CCCH+SDCCH4,0,0 TCH/F,0,0 TCH/H,0,0 SDCCH8,0,0'
239 + ' TCH/F_PDCH,0,0 CCCH+SDCCH4+CBCH,0,0'
240 + ' SDCCH8+CBCH,0,0 TCH/F_TCH/H_PDCH,0,0')
Holger Hans Peter Freyther5eebb7a2014-12-05 12:03:24 +0100241
Holger Hans Peter Freyther1f6cce72014-12-05 14:44:21 +0100242 def testBtsOmlConnectionState(self):
243 """Check OML state. It will not be connected"""
244 r = self.do_set('bts.0.oml-connection-state', '1')
245 self.assertEquals(r['mtype'], 'ERROR')
246 self.assertEquals(r['error'], 'Read only attribute')
247
248 # No RSL link so everything is 0
249 r = self.do_get('bts.0.oml-connection-state')
250 self.assertEquals(r['mtype'], 'GET_REPLY')
251 self.assertEquals(r['value'], 'disconnected')
252
Holger Hans Peter Freytherd092f482014-03-23 11:17:27 +0100253 def testTrxPowerRed(self):
254 r = self.do_get('bts.0.trx.0.max-power-reduction')
255 self.assertEquals(r['mtype'], 'GET_REPLY')
256 self.assertEquals(r['var'], 'bts.0.trx.0.max-power-reduction')
257 self.assertEquals(r['value'], '20')
258
259 r = self.do_set('bts.0.trx.0.max-power-reduction', '22')
260 self.assertEquals(r['mtype'], 'SET_REPLY')
261 self.assertEquals(r['var'], 'bts.0.trx.0.max-power-reduction')
262 self.assertEquals(r['value'], '22')
263
264 r = self.do_get('bts.0.trx.0.max-power-reduction')
265 self.assertEquals(r['mtype'], 'GET_REPLY')
266 self.assertEquals(r['var'], 'bts.0.trx.0.max-power-reduction')
267 self.assertEquals(r['value'], '22')
268
269 r = self.do_set('bts.0.trx.0.max-power-reduction', '1')
270 self.assertEquals(r['mtype'], 'ERROR')
271 self.assertEquals(r['error'], 'Value must be even')
272
Holger Hans Peter Freyther175a2402013-01-09 19:55:04 +0100273 def testTrxArfcn(self):
274 r = self.do_get('bts.0.trx.0.arfcn')
275 self.assertEquals(r['mtype'], 'GET_REPLY')
276 self.assertEquals(r['var'], 'bts.0.trx.0.arfcn')
277 self.assertEquals(r['value'], '871')
278
279 r = self.do_set('bts.0.trx.0.arfcn', '873')
280 self.assertEquals(r['mtype'], 'SET_REPLY')
281 self.assertEquals(r['var'], 'bts.0.trx.0.arfcn')
282 self.assertEquals(r['value'], '873')
283
284 r = self.do_get('bts.0.trx.0.arfcn')
285 self.assertEquals(r['mtype'], 'GET_REPLY')
286 self.assertEquals(r['var'], 'bts.0.trx.0.arfcn')
287 self.assertEquals(r['value'], '873')
288
289 r = self.do_set('bts.0.trx.0.arfcn', '2000')
290 self.assertEquals(r['mtype'], 'ERROR')
291 self.assertEquals(r['error'], 'Input not within the range')
292
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200293 def testRfLock(self):
294 r = self.do_get('bts.0.rf_state')
295 self.assertEquals(r['mtype'], 'GET_REPLY')
296 self.assertEquals(r['var'], 'bts.0.rf_state')
297 self.assertEquals(r['value'], 'inoperational,unlocked,on')
298
299 r = self.do_set('rf_locked', '1')
300 self.assertEquals(r['mtype'], 'SET_REPLY')
301 self.assertEquals(r['var'], 'rf_locked')
302 self.assertEquals(r['value'], '1')
303
304 time.sleep(1.5)
305
306 r = self.do_get('bts.0.rf_state')
307 self.assertEquals(r['mtype'], 'GET_REPLY')
308 self.assertEquals(r['var'], 'bts.0.rf_state')
309 self.assertEquals(r['value'], 'inoperational,locked,off')
310
Holger Hans Peter Freyther66105fd2015-02-10 23:03:25 +0100311 r = self.do_get('rf_locked')
312 self.assertEquals(r['mtype'], 'GET_REPLY')
313 self.assertEquals(r['var'], 'rf_locked')
314 self.assertEquals(r['value'], 'state=off,policy=off')
315
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200316 r = self.do_set('rf_locked', '0')
317 self.assertEquals(r['mtype'], 'SET_REPLY')
318 self.assertEquals(r['var'], 'rf_locked')
319 self.assertEquals(r['value'], '0')
320
321 time.sleep(1.5)
322
323 r = self.do_get('bts.0.rf_state')
324 self.assertEquals(r['mtype'], 'GET_REPLY')
325 self.assertEquals(r['var'], 'bts.0.rf_state')
326 self.assertEquals(r['value'], 'inoperational,unlocked,on')
327
Holger Hans Peter Freyther66105fd2015-02-10 23:03:25 +0100328 r = self.do_get('rf_locked')
329 self.assertEquals(r['mtype'], 'GET_REPLY')
330 self.assertEquals(r['var'], 'rf_locked')
331 self.assertEquals(r['value'], 'state=off,policy=on')
332
Jacob Erlbeckcc391b82013-10-01 13:26:42 +0200333 def testTimezone(self):
334 r = self.do_get('bts.0.timezone')
335 self.assertEquals(r['mtype'], 'GET_REPLY')
336 self.assertEquals(r['var'], 'bts.0.timezone')
337 self.assertEquals(r['value'], 'off')
338
339 r = self.do_set('bts.0.timezone', '-2,15,2')
340 self.assertEquals(r['mtype'], 'SET_REPLY')
341 self.assertEquals(r['var'], 'bts.0.timezone')
342 self.assertEquals(r['value'], '-2,15,2')
343
344 r = self.do_get('bts.0.timezone')
345 self.assertEquals(r['mtype'], 'GET_REPLY')
346 self.assertEquals(r['var'], 'bts.0.timezone')
347 self.assertEquals(r['value'], '-2,15,2')
348
349 # Test invalid input
350 r = self.do_set('bts.0.timezone', '-2,15,2,5,6,7')
351 self.assertEquals(r['mtype'], 'SET_REPLY')
352 self.assertEquals(r['var'], 'bts.0.timezone')
353 self.assertEquals(r['value'], '-2,15,2')
354
355 r = self.do_set('bts.0.timezone', '-2,15')
356 self.assertEquals(r['mtype'], 'ERROR')
357 r = self.do_set('bts.0.timezone', '-2')
358 self.assertEquals(r['mtype'], 'ERROR')
359 r = self.do_set('bts.0.timezone', '1')
360
361 r = self.do_set('bts.0.timezone', 'off')
362 self.assertEquals(r['mtype'], 'SET_REPLY')
363 self.assertEquals(r['var'], 'bts.0.timezone')
364 self.assertEquals(r['value'], 'off')
365
366 r = self.do_get('bts.0.timezone')
367 self.assertEquals(r['mtype'], 'GET_REPLY')
368 self.assertEquals(r['var'], 'bts.0.timezone')
369 self.assertEquals(r['value'], 'off')
370
Holger Hans Peter Freythere9faa6f2014-04-24 10:30:05 +0200371 def testMcc(self):
372 r = self.do_set('mcc', '23')
373 r = self.do_get('mcc')
374 self.assertEquals(r['mtype'], 'GET_REPLY')
375 self.assertEquals(r['var'], 'mcc')
376 self.assertEquals(r['value'], '23')
377
378 r = self.do_set('mcc', '023')
379 r = self.do_get('mcc')
380 self.assertEquals(r['mtype'], 'GET_REPLY')
381 self.assertEquals(r['var'], 'mcc')
382 self.assertEquals(r['value'], '23')
383
384 def testMnc(self):
385 r = self.do_set('mnc', '9')
386 r = self.do_get('mnc')
387 self.assertEquals(r['mtype'], 'GET_REPLY')
388 self.assertEquals(r['var'], 'mnc')
389 self.assertEquals(r['value'], '9')
390
391 r = self.do_set('mnc', '09')
392 r = self.do_get('mnc')
393 self.assertEquals(r['mtype'], 'GET_REPLY')
394 self.assertEquals(r['var'], 'mnc')
395 self.assertEquals(r['value'], '9')
396
397
Holger Hans Peter Freyther3adb7722014-03-04 17:16:58 +0100398 def testMccMncApply(self):
399 # Test some invalid input
400 r = self.do_set('mcc-mnc-apply', 'WRONG')
401 self.assertEquals(r['mtype'], 'ERROR')
402
403 r = self.do_set('mcc-mnc-apply', '1,')
404 self.assertEquals(r['mtype'], 'ERROR')
405
406 r = self.do_set('mcc-mnc-apply', '200,3')
407 self.assertEquals(r['mtype'], 'SET_REPLY')
408 self.assertEquals(r['var'], 'mcc-mnc-apply')
409 self.assertEquals(r['value'], 'Tried to drop the BTS')
410
411 # Set it again
412 r = self.do_set('mcc-mnc-apply', '200,3')
413 self.assertEquals(r['mtype'], 'SET_REPLY')
414 self.assertEquals(r['var'], 'mcc-mnc-apply')
415 self.assertEquals(r['value'], 'Nothing changed')
416
417 # Change it
418 r = self.do_set('mcc-mnc-apply', '200,4')
419 self.assertEquals(r['mtype'], 'SET_REPLY')
420 self.assertEquals(r['var'], 'mcc-mnc-apply')
421 self.assertEquals(r['value'], 'Tried to drop the BTS')
422
423 # Change it
424 r = self.do_set('mcc-mnc-apply', '201,4')
425 self.assertEquals(r['mtype'], 'SET_REPLY')
426 self.assertEquals(r['var'], 'mcc-mnc-apply')
427 self.assertEquals(r['value'], 'Tried to drop the BTS')
428
429 # Verify
430 r = self.do_get('mnc')
431 self.assertEquals(r['mtype'], 'GET_REPLY')
432 self.assertEquals(r['var'], 'mnc')
433 self.assertEquals(r['value'], '4')
434
435 r = self.do_get('mcc')
436 self.assertEquals(r['mtype'], 'GET_REPLY')
437 self.assertEquals(r['var'], 'mcc')
438 self.assertEquals(r['value'], '201')
439
Holger Hans Peter Freythere9faa6f2014-04-24 10:30:05 +0200440 # Change it
441 r = self.do_set('mcc-mnc-apply', '202,03')
442 self.assertEquals(r['mtype'], 'SET_REPLY')
443 self.assertEquals(r['var'], 'mcc-mnc-apply')
444 self.assertEquals(r['value'], 'Tried to drop the BTS')
445
446 r = self.do_get('mnc')
447 self.assertEquals(r['mtype'], 'GET_REPLY')
448 self.assertEquals(r['var'], 'mnc')
449 self.assertEquals(r['value'], '3')
450
451 r = self.do_get('mcc')
452 self.assertEquals(r['mtype'], 'GET_REPLY')
453 self.assertEquals(r['var'], 'mcc')
454 self.assertEquals(r['value'], '202')
455
Holger Hans Peter Freyther9dbc3f82014-03-23 12:06:36 +0100456class TestCtrlNITB(TestCtrlBase):
457
458 def tearDown(self):
459 TestCtrlBase.tearDown(self)
460 os.unlink("test_hlr.sqlite3")
461
462 def ctrl_command(self):
463 return ["./src/osmo-nitb/osmo-nitb", "-c",
464 "doc/examples/osmo-nitb/nanobts/openbsc.cfg", "-l", "test_hlr.sqlite3"]
465
466 def ctrl_app(self):
467 return (4249, "./src/osmo-nitb/osmo-nitb", "OsmoBSC", "nitb")
468
Holger Hans Peter Freytherca415192015-02-10 21:55:37 +0100469 def testNumberOfBTS(self):
470 r = self.do_get('number-of-bts')
471 self.assertEquals(r['mtype'], 'GET_REPLY')
472 self.assertEquals(r['var'], 'number-of-bts')
473 self.assertEquals(r['value'], '1')
474
Holger Hans Peter Freyther9bcb1a52016-04-06 22:41:12 +0200475 def testSubscriberAddWithKi(self):
476 """Test that we can set the algorithm to none, xor, comp128v1"""
477
478 r = self.do_set('subscriber-modify-v1', '2620345,445566')
479 self.assertEquals(r['mtype'], 'SET_REPLY')
480 self.assertEquals(r['var'], 'subscriber-modify-v1')
481 self.assertEquals(r['value'], 'OK')
482
483 r = self.do_set('subscriber-modify-v1', '2620345,445566,none')
484 self.assertEquals(r['mtype'], 'SET_REPLY')
485 self.assertEquals(r['var'], 'subscriber-modify-v1')
486 self.assertEquals(r['value'], 'OK')
487
488 r = self.do_set('subscriber-modify-v1', '2620345,445566,xor')
489 self.assertEquals(r['mtype'], 'ERROR')
490 self.assertEquals(r['error'], 'Value failed verification.')
491
492 r = self.do_set('subscriber-modify-v1', '2620345,445566,comp128v1,00112233445566778899AABBCCDDEEFF')
493 self.assertEquals(r['mtype'], 'SET_REPLY')
494 self.assertEquals(r['var'], 'subscriber-modify-v1')
495 self.assertEquals(r['value'], 'OK')
496
497 r = self.do_set('subscriber-modify-v1', '2620345,445566,none')
498 self.assertEquals(r['mtype'], 'SET_REPLY')
499 self.assertEquals(r['var'], 'subscriber-modify-v1')
500 self.assertEquals(r['value'], 'OK')
501
Holger Hans Peter Freyther2d99eeb2014-03-23 14:01:08 +0100502 def testSubscriberAddRemove(self):
Holger Hans Peter Freyther9dbc3f82014-03-23 12:06:36 +0100503 r = self.do_set('subscriber-modify-v1', '2620345,445566')
504 self.assertEquals(r['mtype'], 'SET_REPLY')
505 self.assertEquals(r['var'], 'subscriber-modify-v1')
506 self.assertEquals(r['value'], 'OK')
507
508 r = self.do_set('subscriber-modify-v1', '2620345,445567')
509 self.assertEquals(r['mtype'], 'SET_REPLY')
510 self.assertEquals(r['var'], 'subscriber-modify-v1')
511 self.assertEquals(r['value'], 'OK')
512
513 # TODO. verify that the entry has been created and modified? Invoke
514 # the sqlite3 CLI or do it through the DB libraries?
515
Holger Hans Peter Freyther2d99eeb2014-03-23 14:01:08 +0100516 r = self.do_set('subscriber-delete-v1', '2620345')
517 self.assertEquals(r['mtype'], 'SET_REPLY')
518 self.assertEquals(r['value'], 'Removed')
519
520 r = self.do_set('subscriber-delete-v1', '2620345')
521 self.assertEquals(r['mtype'], 'ERROR')
522 self.assertEquals(r['error'], 'Failed to find subscriber')
523
Holger Hans Peter Freytherd883db02014-03-23 16:22:55 +0100524 def testSubscriberList(self):
525 # TODO. Add command to mark a subscriber as active
526 r = self.do_get('subscriber-list-active-v1')
527 self.assertEquals(r['mtype'], 'GET_REPLY')
528 self.assertEquals(r['var'], 'subscriber-list-active-v1')
529 self.assertEquals(r['value'], None)
530
Holger Hans Peter Freytherb1461152014-11-21 10:20:29 +0100531 def testApplyConfiguration(self):
532 r = self.do_get('bts.0.apply-configuration')
533 self.assertEquals(r['mtype'], 'ERROR')
534 self.assertEquals(r['error'], 'Write only attribute')
535
536 r = self.do_set('bts.0.apply-configuration', '1')
537 self.assertEquals(r['mtype'], 'SET_REPLY')
538 self.assertEquals(r['value'], 'Tried to drop the BTS')
539
Holger Hans Peter Freyther4e13a8f2015-01-31 22:16:00 +0100540 def testGprsMode(self):
541 r = self.do_get('bts.0.gprs-mode')
542 self.assertEquals(r['mtype'], 'GET_REPLY')
543 self.assertEquals(r['var'], 'bts.0.gprs-mode')
544 self.assertEquals(r['value'], 'none')
545
546 r = self.do_set('bts.0.gprs-mode', 'bla')
547 self.assertEquals(r['mtype'], 'ERROR')
548 self.assertEquals(r['error'], 'Mode is not known')
549
550 r = self.do_set('bts.0.gprs-mode', 'egprs')
551 self.assertEquals(r['mtype'], 'SET_REPLY')
552 self.assertEquals(r['value'], 'egprs')
553
554 r = self.do_get('bts.0.gprs-mode')
555 self.assertEquals(r['mtype'], 'GET_REPLY')
556 self.assertEquals(r['var'], 'bts.0.gprs-mode')
557 self.assertEquals(r['value'], 'egprs')
558
Holger Hans Peter Freyther4ecc6872014-03-04 15:38:00 +0100559class TestCtrlNAT(TestCtrlBase):
560
561 def ctrl_command(self):
562 return ["./src/osmo-bsc_nat/osmo-bsc_nat", "-c",
563 "doc/examples/osmo-bsc_nat/osmo-bsc_nat.cfg"]
564
565 def ctrl_app(self):
566 return (4250, "./src/osmo-bsc_nat/osmo-bsc_nat", "OsmoNAT", "nat")
567
568 def testAccessList(self):
569 r = self.do_get('net.0.bsc_cfg.0.access-list-name')
570 self.assertEquals(r['mtype'], 'GET_REPLY')
571 self.assertEquals(r['var'], 'net')
572 self.assertEquals(r['value'], None)
573
574 r = self.do_set('net.0.bsc_cfg.0.access-list-name', 'bla')
575 self.assertEquals(r['mtype'], 'SET_REPLY')
576 self.assertEquals(r['var'], 'net')
577 self.assertEquals(r['value'], 'bla')
578
579 r = self.do_get('net.0.bsc_cfg.0.access-list-name')
580 self.assertEquals(r['mtype'], 'GET_REPLY')
581 self.assertEquals(r['var'], 'net')
582 self.assertEquals(r['value'], 'bla')
583
584 r = self.do_set('net.0.bsc_cfg.0.no-access-list-name', '1')
585 self.assertEquals(r['mtype'], 'SET_REPLY')
586 self.assertEquals(r['var'], 'net')
587 self.assertEquals(r['value'], None)
588
Holger Hans Peter Freyther416c08f2014-12-09 19:13:00 +0100589 r = self.do_get('net.0.bsc_cfg.0.access-list-name')
590 self.assertEquals(r['mtype'], 'GET_REPLY')
Holger Hans Peter Freyther4ecc6872014-03-04 15:38:00 +0100591 self.assertEquals(r['var'], 'net')
592 self.assertEquals(r['value'], None)
Holger Hans Peter Freyther3adb7722014-03-04 17:16:58 +0100593
Holger Hans Peter Freytherab94ca12015-04-05 15:15:36 +0200594 def testAccessListManagement(self):
595 r = self.do_set("net.0.add.allow.access-list.404", "abc")
596 self.assertEquals(r['mtype'], 'ERROR')
597
598 r = self.do_set("net.0.add.allow.access-list.bla", "^234$")
599 self.assertEquals(r['mtype'], 'SET_REPLY')
600 self.assertEquals(r['var'], 'net.0.add.allow.access-list.bla')
601 self.assertEquals(r['value'], 'IMSI allow added to access list')
602
603 # TODO.. find a way to actually see if this rule has been
604 # added. e.g. by implementing a get for the list.
605
Holger Hans Peter Freythera2730302014-03-23 18:08:26 +0100606class TestCtrlSGSN(TestCtrlBase):
607 def ctrl_command(self):
608 return ["./src/gprs/osmo-sgsn", "-c",
609 "doc/examples/osmo-sgsn/osmo-sgsn.cfg"]
610
611 def ctrl_app(self):
612 return (4251, "./src/gprs/osmo-sgsn", "OsmoSGSN", "sgsn")
613
614 def testListSubscribers(self):
615 # TODO. Add command to mark a subscriber as active
616 r = self.do_get('subscriber-list-active-v1')
617 self.assertEquals(r['mtype'], 'GET_REPLY')
618 self.assertEquals(r['var'], 'subscriber-list-active-v1')
619 self.assertEquals(r['value'], None)
620
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200621def add_bsc_test(suite, workdir):
622 if not os.path.isfile(os.path.join(workdir, "src/osmo-bsc/osmo-bsc")):
623 print("Skipping the BSC test")
624 return
625 test = unittest.TestLoader().loadTestsFromTestCase(TestCtrlBSC)
626 suite.addTest(test)
627
Holger Hans Peter Freyther9dbc3f82014-03-23 12:06:36 +0100628def add_nitb_test(suite, workdir):
629 test = unittest.TestLoader().loadTestsFromTestCase(TestCtrlNITB)
630 suite.addTest(test)
631
Holger Hans Peter Freyther4ecc6872014-03-04 15:38:00 +0100632def add_nat_test(suite, workdir):
633 if not os.path.isfile(os.path.join(workdir, "src/osmo-bsc_nat/osmo-bsc_nat")):
634 print("Skipping the NAT test")
635 return
636 test = unittest.TestLoader().loadTestsFromTestCase(TestCtrlNAT)
637 suite.addTest(test)
638
Holger Hans Peter Freythera2730302014-03-23 18:08:26 +0100639def add_sgsn_test(suite, workdir):
640 if not os.path.isfile(os.path.join(workdir, "src/gprs/osmo-sgsn")):
641 print("Skipping the SGSN test")
642 return
643 test = unittest.TestLoader().loadTestsFromTestCase(TestCtrlSGSN)
644 suite.addTest(test)
645
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200646if __name__ == '__main__':
647 import argparse
648 import sys
649
650 workdir = '.'
651
652 parser = argparse.ArgumentParser()
653 parser.add_argument("-v", "--verbose", dest="verbose",
654 action="store_true", help="verbose mode")
655 parser.add_argument("-p", "--pythonconfpath", dest="p",
656 help="searchpath for config")
657 parser.add_argument("-w", "--workdir", dest="w",
658 help="Working directory")
659 args = parser.parse_args()
660
661 verbose_level = 1
662 if args.verbose:
663 verbose_level = 2
664 verbose = True
665
666 if args.w:
667 workdir = args.w
668
669 if args.p:
670 confpath = args.p
671
672 print "confpath %s, workdir %s" % (confpath, workdir)
673 os.chdir(workdir)
674 print "Running tests for specific control commands"
675 suite = unittest.TestSuite()
676 add_bsc_test(suite, workdir)
Holger Hans Peter Freyther9dbc3f82014-03-23 12:06:36 +0100677 add_nitb_test(suite, workdir)
Holger Hans Peter Freyther4ecc6872014-03-04 15:38:00 +0100678 add_nat_test(suite, workdir)
Holger Hans Peter Freythera2730302014-03-23 18:08:26 +0100679 add_sgsn_test(suite, workdir)
Jacob Erlbeck0760a832013-09-16 11:20:28 +0200680 res = unittest.TextTestRunner(verbosity=verbose_level).run(suite)
681 sys.exit(len(res.errors) + len(res.failures))