blob: 69b247819fed06dc68344ccef6badafa66267b09 [file] [log] [blame]
Oliver Smith14ec6342019-12-10 13:07:27 +01001#!/usr/bin/env python3
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +02002
3# (C) 2013 by Jacob Erlbeck <jerlbeck@sysmocom.de>
Holger Hans Peter Freythereab2a3f2014-03-04 17:16:58 +01004# (C) 2014 by Holger Hans Peter Freyther
Jacob Erlbeck5741e1f2013-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
Max8a58bcd2017-12-19 18:12:11 +010032from osmopy.osmo_ipa import Ctrl, IPA
Max580fdb52017-05-02 16:10:07 +020033
Neels Hofmeyr6226a362017-02-24 17:55:11 +010034# to be able to find $top_srcdir/doc/...
35confpath = os.path.join(sys.path[0], '..')
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020036verbose = False
37
38class TestCtrlBase(unittest.TestCase):
39
40 def ctrl_command(self):
41 raise Exception("Needs to be implemented by a subclass")
42
43 def ctrl_app(self):
44 raise Exception("Needs to be implemented by a subclass")
45
46 def setUp(self):
47 osmo_ctrl_cmd = self.ctrl_command()[:]
48 config_index = osmo_ctrl_cmd.index('-c')
49 if config_index:
50 cfi = config_index + 1
51 osmo_ctrl_cmd[cfi] = os.path.join(confpath, osmo_ctrl_cmd[cfi])
52
53 try:
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020054 self.proc = osmoutil.popen_devnull(osmo_ctrl_cmd)
55 except OSError:
Oliver Smith14ec6342019-12-10 13:07:27 +010056 print("Current directory: %s" % os.getcwd(), file=sys.stderr)
57 print("Consider setting -b", file=sys.stderr)
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020058 time.sleep(2)
59
60 appstring = self.ctrl_app()[2]
61 appport = self.ctrl_app()[0]
62 self.connect("127.0.0.1", appport)
63 self.next_id = 1000
64
65 def tearDown(self):
66 self.disconnect()
67 osmoutil.end_proc(self.proc)
68
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020069 def disconnect(self):
70 if not (self.sock is None):
71 self.sock.close()
72
73 def connect(self, host, port):
74 if verbose:
Oliver Smith14ec6342019-12-10 13:07:27 +010075 print("Connecting to host %s:%i" % (host, port))
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020076
Neels Hofmeyrfd32c9f2017-02-27 02:01:37 +010077 retries = 30
78 while True:
79 try:
80 sck = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
81 sck.setblocking(1)
82 sck.connect((host, port))
83 except IOError:
84 retries -= 1
85 if retries <= 0:
86 raise
87 time.sleep(.1)
88 continue
89 break
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020090 self.sock = sck
91 return sck
92
93 def send(self, data):
94 if verbose:
Oliver Smith14ec6342019-12-10 13:07:27 +010095 print("Sending \"%s\"" %(data))
Max580fdb52017-05-02 16:10:07 +020096 data = Ctrl().add_header(data)
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +020097 return self.sock.send(data) == len(data)
98
99 def send_set(self, var, value, id):
100 setmsg = "SET %s %s %s" %(id, var, value)
101 return self.send(setmsg)
102
103 def send_get(self, var, id):
104 getmsg = "GET %s %s" %(id, var)
105 return self.send(getmsg)
106
107 def do_set(self, var, value):
108 id = self.next_id
109 self.next_id += 1
110 self.send_set(var, value, id)
111 return self.recv_msgs()[id]
112
113 def do_get(self, var):
114 id = self.next_id
115 self.next_id += 1
116 self.send_get(var, id)
117 return self.recv_msgs()[id]
118
119 def recv_msgs(self):
120 responses = {}
121 data = self.sock.recv(4096)
122 while (len(data)>0):
Max580fdb52017-05-02 16:10:07 +0200123 (head, data) = IPA().split_combined(data)
Oliver Smith14ec6342019-12-10 13:07:27 +0100124 answer = Ctrl().rem_header(head).decode()
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200125 if verbose:
Oliver Smith14ec6342019-12-10 13:07:27 +0100126 print("Got message:", answer)
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200127 (mtype, id, msg) = answer.split(None, 2)
128 id = int(id)
129 rsp = {'mtype': mtype, 'id': id}
130 if mtype == "ERROR":
131 rsp['error'] = msg
132 else:
Holger Hans Peter Freyther842137a2014-03-04 15:38:00 +0100133 split = msg.split(None, 1)
134 rsp['var'] = split[0]
135 if len(split) > 1:
136 rsp['value'] = split[1]
137 else:
138 rsp['value'] = None
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200139 responses[id] = rsp
140
141 if verbose:
Oliver Smith14ec6342019-12-10 13:07:27 +0100142 print("Decoded replies: ", responses)
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200143
144 return responses
145
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100146class TestCtrlSGSN(TestCtrlBase):
147 def ctrl_command(self):
Pau Espin Pedrola3646fb2019-08-30 20:47:02 +0200148 return ["./src/sgsn/osmo-sgsn", "-c",
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100149 "doc/examples/osmo-sgsn/osmo-sgsn.cfg"]
150
151 def ctrl_app(self):
Pau Espin Pedrola3646fb2019-08-30 20:47:02 +0200152 return (4251, "./src/sgsn/osmo-sgsn", "OsmoSGSN", "sgsn")
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100153
154 def testListSubscribers(self):
155 # TODO. Add command to mark a subscriber as active
156 r = self.do_get('subscriber-list-active-v1')
Oliver Smith14ec6342019-12-10 13:07:27 +0100157 self.assertEqual(r['mtype'], 'GET_REPLY')
158 self.assertEqual(r['var'], 'subscriber-list-active-v1')
159 self.assertEqual(r['value'], None)
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100160
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100161def add_sgsn_test(suite, workdir):
Pau Espin Pedrola3646fb2019-08-30 20:47:02 +0200162 if not os.path.isfile(os.path.join(workdir, "src/sgsn/osmo-sgsn")):
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100163 print("Skipping the SGSN test")
164 return
165 test = unittest.TestLoader().loadTestsFromTestCase(TestCtrlSGSN)
166 suite.addTest(test)
167
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200168if __name__ == '__main__':
169 import argparse
170 import sys
171
172 workdir = '.'
173
174 parser = argparse.ArgumentParser()
175 parser.add_argument("-v", "--verbose", dest="verbose",
176 action="store_true", help="verbose mode")
177 parser.add_argument("-p", "--pythonconfpath", dest="p",
178 help="searchpath for config")
179 parser.add_argument("-w", "--workdir", dest="w",
180 help="Working directory")
181 args = parser.parse_args()
182
183 verbose_level = 1
184 if args.verbose:
185 verbose_level = 2
186 verbose = True
187
188 if args.w:
189 workdir = args.w
190
191 if args.p:
192 confpath = args.p
193
Oliver Smith14ec6342019-12-10 13:07:27 +0100194 print("confpath %s, workdir %s" % (confpath, workdir))
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200195 os.chdir(workdir)
Oliver Smith14ec6342019-12-10 13:07:27 +0100196 print("Running tests for specific control commands")
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200197 suite = unittest.TestSuite()
Holger Hans Peter Freythere8e5ef22014-03-23 18:08:26 +0100198 add_sgsn_test(suite, workdir)
Jacob Erlbeck5741e1f2013-09-16 11:20:28 +0200199 res = unittest.TextTestRunner(verbosity=verbose_level).run(suite)
200 sys.exit(len(res.errors) + len(res.failures))