blob: 55cb546b607462b5b632f1f80b1be5da329bc91b [file] [log] [blame]
Pau Espin Pedrol8a725862018-10-26 15:54:28 +02001# osmo_gsm_tester: specifics for running an iperf3 client and server
2#
3# Copyright (C) 2018 by sysmocom - s.f.m.c. GmbH
4#
5# Author: Pau Espin Pedrol <pespin@sysmocom.de>
6#
7# This program is free software: you can redistribute it and/or modify
8# it under the terms of the GNU General Public License as
9# 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
15# GNU General Public License for more details.
16#
17# You should have received a copy of the GNU General Public License
18# along with this program. If not, see <http://www.gnu.org/licenses/>.
19
20import os
21import json
22
23from . import log, util, process, pcap_recorder
24
Pau Espin Pedrol64f0b1b2018-11-09 17:44:10 +010025def iperf3_result_to_json(file):
26 with open(file) as f:
27 # Sometimes iperf3 provides 2 dictionaries, the 2nd one being an error about being interrupted (by us).
28 # json parser doesn't support (raises exception) parsing several dictionaries at a time (not a valid json object).
29 # We are only interested in the first dictionary, the regular results one:
30 d = f.read().split("\n}\n")[0] + "\n}\n"
31 data = json.loads(d)
32 return data
33
34
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020035class IPerf3Server(log.Origin):
36
Pau Espin Pedrolcf6a3602018-11-14 16:38:19 +010037 DEFAULT_SRV_PORT = 5003
38
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020039 def __init__(self, suite_run, ip_address):
40 super().__init__(log.C_RUN, 'iperf3-srv_%s' % ip_address.get('addr'))
41 self.run_dir = None
42 self.config_file = None
43 self.process = None
44 self.suite_run = suite_run
45 self.ip_address = ip_address
Pau Espin Pedrolcf6a3602018-11-14 16:38:19 +010046 self._port = IPerf3Server.DEFAULT_SRV_PORT
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020047
48 def start(self):
49 self.log('Starting iperf3-srv')
50 self.run_dir = util.Dir(self.suite_run.get_test_run_dir().new_dir(self.name()))
51
52 pcap_recorder.PcapRecorder(self.suite_run, self.run_dir.new_dir('pcap'), None,
53 'host %s and port not 22' % self.addr())
54
55 self.log_file = self.run_dir.new_file('iperf3_srv.json')
56 self.process = process.Process(self.name(), self.run_dir,
57 ('iperf3', '-s', '-B', self.addr(),
58 '-p', str(self._port), '-J',
59 '--logfile', os.path.abspath(self.log_file)),
60 env={})
61 self.suite_run.remember_to_stop(self.process)
62 self.process.launch()
63
Pau Espin Pedrolcf6a3602018-11-14 16:38:19 +010064 def set_port(self, port):
65 self._port = port
66
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020067 def stop(self):
68 self.suite_run.stop_process(self.process)
69
70 def get_results(self):
Pau Espin Pedrol64f0b1b2018-11-09 17:44:10 +010071 return iperf3_result_to_json(self.log_file)
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020072
73 def addr(self):
74 return self.ip_address.get('addr')
75
76 def port(self):
77 return self._port
78
Pau Espin Pedrol0df63172018-11-14 16:39:30 +010079 def __str__(self):
80 return "%s:%u" %(self.addr(), self.port())
81
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020082 def running(self):
83 return not self.process.terminated()
84
85 def create_client(self):
86 return IPerf3Client(self.suite_run, self)
87
88class IPerf3Client(log.Origin):
89
90 def __init__(self, suite_run, iperf3srv):
91 super().__init__(log.C_RUN, 'iperf3-cli_%s' % iperf3srv.addr())
92 self.run_dir = None
93 self.config_file = None
94 self.process = None
95 self.server = iperf3srv
96 self.suite_run = suite_run
97
Pau Espin Pedrol0fc74372018-11-14 16:44:57 +010098 def prepare_test_proc(self, netns=None):
Pau Espin Pedrol8a725862018-10-26 15:54:28 +020099 self.log('Starting iperf3-client connecting to %s:%d' % (self.server.addr(), self.server.port()))
100 self.run_dir = util.Dir(self.suite_run.get_test_run_dir().new_dir(self.name()))
101
102 pcap_recorder.PcapRecorder(self.suite_run, self.run_dir.new_dir('pcap'), None,
103 'host %s and port not 22' % self.server.addr(), netns)
104
105 self.log_file = self.run_dir.new_file('iperf3_cli.json')
106 popen_args = ('iperf3', '-c', self.server.addr(),
107 '-p', str(self.server.port()), '-J',
108 '--logfile', os.path.abspath(self.log_file))
109 if netns:
110 self.process = process.NetNSProcess(self.name(), self.run_dir, netns, popen_args, env={})
111 else:
112 self.process = process.Process(self.name(), self.run_dir, popen_args, env={})
Pau Espin Pedrol0fc74372018-11-14 16:44:57 +0100113 return self.process
114
115 def run_test_sync(self, netns=None):
116 self.prepare_test_proc(netns)
Pau Espin Pedrol79df7392018-11-12 18:15:30 +0100117 self.process.launch_sync()
Pau Espin Pedrol8a725862018-10-26 15:54:28 +0200118 return self.get_results()
119
120 def get_results(self):
Pau Espin Pedrol64f0b1b2018-11-09 17:44:10 +0100121 return iperf3_result_to_json(self.log_file)
Pau Espin Pedrol8a725862018-10-26 15:54:28 +0200122
123# vim: expandtab tabstop=4 shiftwidth=4