2 # -*- coding: utf-8 -*-
5 from nepi.util.constants import STATUS_FINISHED, TIME_NOW
6 from nepi.testbeds import planetlab
15 class PlanetLabExecuteTestCase(unittest.TestCase):
17 self.root_dir = tempfile.mkdtemp()
20 shutil.rmtree(self.root_dir)
22 def make_instance(self):
23 testbed_version = "01"
24 instance = planetlab.TestbedController(testbed_version)
25 slicename = "inria_nepi12"
26 pl_user, pl_pwd = test_util.pl_auth()
28 instance.defer_configure("homeDirectory", self.root_dir)
29 instance.defer_configure("slice", slicename)
30 instance.defer_configure("sliceSSHKey", "/user/%s/home/.ssh/id_rsa_planetlab" % (getpass.getuser(),))
31 instance.defer_configure("authUser", pl_user)
32 instance.defer_configure("authPass", pl_pwd)
36 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
37 def test_simple(self):
38 instance = self.make_instance()
40 instance.defer_create(2, "Node")
41 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
42 instance.defer_create(3, "Node")
43 instance.defer_create_set(3, "hostname", "onelab10.pl.sophia.inria.fr")
44 instance.defer_create(4, "NodeInterface")
45 instance.defer_connect(2, "devs", 4, "node")
46 instance.defer_create(5, "NodeInterface")
47 instance.defer_connect(3, "devs", 5, "node")
48 instance.defer_create(6, "Internet")
49 instance.defer_connect(4, "inet", 6, "devs")
50 instance.defer_connect(5, "inet", 6, "devs")
51 instance.defer_create(7, "Application")
52 instance.defer_create_set(7, "command", "ping -qc1 {#[GUID-5].addr[0].[Address]#}")
53 instance.defer_add_trace(7, "stdout")
54 instance.defer_add_trace(7, "stderr")
55 instance.defer_connect(7, "node", 2, "apps")
59 instance.do_connect_init()
60 instance.do_connect_compl()
61 instance.do_preconfigure()
63 # Manually replace netref
64 instance.set(TIME_NOW, 7, "command",
65 instance.get(TIME_NOW, 7, "command")
66 .replace("{#[GUID-5].addr[0].[Address]#}",
67 instance.get_address(5, 0, "Address") )
70 instance.do_configure()
73 while instance.status(7) != STATUS_FINISHED:
75 ping_result = instance.trace(7, "stdout") or ""
76 comp_result = r"""PING .* \(.*\) \d*\(\d*\) bytes of data.
78 --- .* ping statistics ---
79 1 packets transmitted, 1 received, 0% packet loss, time \d*ms.*
81 self.assertTrue(re.match(comp_result, ping_result, re.MULTILINE),
82 "Unexpected trace:\n" + ping_result)
86 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
87 def test_depends(self):
88 instance = self.make_instance()
90 instance.defer_create(2, "Node")
91 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
92 instance.defer_create(3, "NodeInterface")
93 instance.defer_connect(2, "devs", 3, "node")
94 instance.defer_create(4, "Internet")
95 instance.defer_connect(3, "inet", 4, "devs")
96 instance.defer_create(5, "Application")
97 instance.defer_create_set(5, "command", "gfortran --version")
98 instance.defer_create_set(5, "depends", "gcc-gfortran")
99 instance.defer_add_trace(5, "stdout")
100 instance.defer_add_trace(5, "stderr")
101 instance.defer_connect(5, "node", 2, "apps")
105 instance.do_connect_init()
106 instance.do_connect_compl()
107 instance.do_preconfigure()
108 instance.do_configure()
111 while instance.status(5) != STATUS_FINISHED:
113 ping_result = instance.trace(5, "stdout") or ""
114 comp_result = r".*GNU Fortran \(GCC\).*"
115 self.assertTrue(re.match(comp_result, ping_result, re.MULTILINE),
116 "Unexpected trace:\n" + ping_result)
120 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
121 def test_build(self):
122 instance = self.make_instance()
124 instance.defer_create(2, "Node")
125 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
126 instance.defer_create(3, "NodeInterface")
127 instance.defer_connect(2, "devs", 3, "node")
128 instance.defer_create(4, "Internet")
129 instance.defer_connect(3, "inet", 4, "devs")
130 instance.defer_create(10, "Application")
131 instance.defer_create_set(10, "command", "./consts")
132 instance.defer_create_set(10, "buildDepends", "gcc")
133 instance.defer_create_set(10, "build", "gcc ${SOURCES}/consts.c -o consts")
134 instance.defer_create_set(10, "install", "cp consts ${SOURCES}/consts")
135 instance.defer_create_set(10, "sources", os.path.join(os.path.dirname(planetlab.__file__),'scripts','consts.c'))
136 instance.defer_add_trace(10, "stdout")
137 instance.defer_add_trace(10, "stderr")
138 instance.defer_connect(10, "node", 2, "apps")
142 instance.do_connect_init()
143 instance.do_connect_compl()
144 instance.do_preconfigure()
145 instance.do_configure()
148 while instance.status(10) != STATUS_FINISHED:
150 ping_result = instance.trace(10, "stdout") or ""
152 r""".*ETH_P_ALL = 0x[0-9a-fA-F]{8}
153 ETH_P_IP = 0x[0-9a-fA-F]{8}
154 TUNSETIFF = 0x[0-9a-fA-F]{8}
155 IFF_NO_PI = 0x[0-9a-fA-F]{8}
156 IFF_TAP = 0x[0-9a-fA-F]{8}
157 IFF_TUN = 0x[0-9a-fA-F]{8}
158 IFF_VNET_HDR = 0x[0-9a-fA-F]{8}
159 TUN_PKT_STRIP = 0x[0-9a-fA-F]{8}
160 IFHWADDRLEN = 0x[0-9a-fA-F]{8}
161 IFNAMSIZ = 0x[0-9a-fA-F]{8}
162 IFREQ_SZ = 0x[0-9a-fA-F]{8}
163 FIONREAD = 0x[0-9a-fA-F]{8}.*
165 self.assertTrue(re.match(comp_result, ping_result, re.MULTILINE),
166 "Unexpected trace:\n" + ping_result)
170 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
171 def test_simple_vsys(self):
172 instance = self.make_instance()
174 instance.defer_create(2, "Node")
175 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
176 instance.defer_create_set(2, "emulation", True) # require emulation
177 instance.defer_create(3, "NodeInterface")
178 instance.defer_connect(2, "devs", 3, "node")
179 instance.defer_create(4, "Internet")
180 instance.defer_connect(3, "inet", 4, "devs")
181 instance.defer_create(5, "TunInterface")
182 instance.defer_add_address(5, "192.168.2.2", 24, False)
183 instance.defer_connect(2, "devs", 5, "node")
184 instance.defer_create(6, "Application")
185 instance.defer_create_set(6, "command", """
187 netconfig help > /dev/null
188 test -e /vsys/vif_up.in > /dev/null
189 test -e /vsys/vif_up.out > /dev/null
190 test -e /vsys/fd_tuntap.control > /dev/null
193 instance.defer_create_set(6, "sudo", True) # only sudo has access to /vsys
194 instance.defer_add_trace(6, "stdout")
195 instance.defer_add_trace(6, "stderr")
196 instance.defer_connect(6, "node", 2, "apps")
200 instance.do_connect_init()
201 instance.do_connect_compl()
202 instance.do_preconfigure()
203 instance.do_configure()
206 while instance.status(6) != STATUS_FINISHED:
208 test_result = (instance.trace(6, "stdout") or "").strip()
209 comp_result = "OKIDOKI"
210 self.assertEqual(comp_result, test_result)
214 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
215 def test_emulation(self):
216 instance = self.make_instance()
218 instance.defer_create(2, "Node")
219 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
220 instance.defer_create_set(2, "emulation", True) # require emulation
221 instance.defer_create(3, "NodeInterface")
222 instance.defer_connect(2, "devs", 3, "node")
223 instance.defer_create(4, "Internet")
224 instance.defer_connect(3, "inet", 4, "devs")
225 instance.defer_create(7, "NetPipe")
226 instance.defer_create_set(7, "mode", "CLIENT")
227 instance.defer_create_set(7, "portList", "80")
228 instance.defer_create_set(7, "bwOut", 12.0/1024.0) # 12kbps
229 instance.defer_create_set(7, "bwIn", 64.0/1024.0) # 64kbps
230 instance.defer_create_set(7, "plrOut", 0.01) # 1% plr outbound - high loss
231 instance.defer_create_set(7, "plrIn", 0.001) # 0.1% plr inbound - regular loss
232 instance.defer_create_set(7, "delayOut", int(1500 * 8 / (12.0/1024.0) / 1000)) # tx delay at 12kbps in ms
233 instance.defer_create_set(7, "delayIn", int(1500 * 8 / (64.0/1024.0) / 1000)) # rx delay at 64kbps in ms
234 instance.defer_add_trace(7, "netpipeStats")
235 instance.defer_connect(2, "pipes", 7, "node")
236 instance.defer_create(8, "Application")
237 instance.defer_create_set(8, "command", "time wget -q -O /dev/null http://www.google.com/") # Fetch ~10kb
238 instance.defer_add_trace(8, "stdout")
239 instance.defer_add_trace(8, "stderr")
240 instance.defer_connect(8, "node", 2, "apps")
244 instance.do_connect_init()
245 instance.do_connect_compl()
246 instance.do_preconfigure()
247 instance.do_configure()
250 while instance.status(8) != STATUS_FINISHED:
252 test_result = (instance.trace(8, "stderr") or "").strip()
253 comp_result = r".*real\s*(?P<min>[0-9]+)m(?P<sec>[0-9]+[.][0-9]+)s.*"
254 netpipe_stats = instance.trace(7, "netpipeStats")
259 # asserts at the end, to make sure there's proper cleanup
260 match = re.match(comp_result, test_result, re.MULTILINE)
261 self.assertTrue(match, "Unexpected output: %s" % (test_result,))
263 minutes = int(match.group("min"))
264 seconds = float(match.group("sec"))
265 self.assertTrue((minutes * 60 + seconds) > 1.0, "Emulation not effective: %s" % (test_result,))
267 self.assertTrue(netpipe_stats, "Unavailable netpipe stats")
269 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
270 def test_tun_emulation_requirement(self):
271 instance = self.make_instance()
273 instance.defer_create(2, "Node")
274 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
275 instance.defer_create(3, "NodeInterface")
276 instance.defer_connect(2, "devs", 3, "node")
277 instance.defer_create(4, "Internet")
278 instance.defer_connect(3, "inet", 4, "devs")
279 instance.defer_create(5, "TunInterface")
280 instance.defer_add_address(5, "192.168.2.2", 24, False)
281 instance.defer_connect(2, "devs", 5, "node")
282 instance.defer_create(6, "Application")
283 instance.defer_create_set(6, "command", "false")
284 instance.defer_add_trace(6, "stdout")
285 instance.defer_add_trace(6, "stderr")
286 instance.defer_connect(6, "node", 2, "apps")
291 instance.do_connect_init()
292 instance.do_connect_compl()
293 instance.do_preconfigure()
294 instance.do_configure()
295 self.fail("Usage of TUN without emulation should fail")
299 @test_util.skipUnless(test_util.pl_auth() is not None, "Test requires PlanetLab authentication info (PL_USER and PL_PASS environment variables)")
300 def test_tun_ping(self):
301 instance = self.make_instance()
303 instance.defer_create(2, "Node")
304 instance.defer_create_set(2, "hostname", "onelab11.pl.sophia.inria.fr")
305 instance.defer_create_set(2, "emulation", True) # require emulation
306 instance.defer_create(3, "Node")
307 instance.defer_create_set(3, "hostname", "onelab10.pl.sophia.inria.fr")
308 instance.defer_create_set(3, "emulation", True) # require emulation
309 instance.defer_create(4, "NodeInterface")
310 instance.defer_connect(2, "devs", 4, "node")
311 instance.defer_create(5, "Internet")
312 instance.defer_connect(4, "inet", 5, "devs")
313 instance.defer_create(6, "NodeInterface")
314 instance.defer_connect(3, "devs", 6, "node")
315 instance.defer_connect(6, "inet", 5, "devs")
316 instance.defer_create(7, "TunInterface")
317 instance.defer_add_address(7, "192.168.2.2", 24, False)
318 instance.defer_connect(2, "devs", 7, "node")
319 instance.defer_create(8, "TunInterface")
320 instance.defer_add_address(8, "192.168.2.3", 24, False)
321 instance.defer_connect(3, "devs", 8, "node")
322 instance.defer_connect(7, "tcp", 8, "tcp")
323 instance.defer_create(9, "Application")
324 instance.defer_create_set(9, "command", "ping -qc1 {#[GUID-8].addr[0].[Address]#}")
325 instance.defer_add_trace(9, "stdout")
326 instance.defer_add_trace(9, "stderr")
327 instance.defer_connect(9, "node", 2, "apps")
331 instance.do_connect_init()
332 instance.do_connect_compl()
333 instance.do_preconfigure()
335 # Manually replace netref
336 instance.set(TIME_NOW, 9, "command",
337 instance.get(TIME_NOW, 9, "command")
338 .replace("{#[GUID-8].addr[0].[Address]#}",
339 instance.get_address(8, 0, "Address") )
342 instance.do_configure()
345 while instance.status(9) != STATUS_FINISHED:
347 ping_result = instance.trace(9, "stdout") or ""
348 comp_result = r"""PING .* \(.*\) \d*\(\d*\) bytes of data.
350 --- .* ping statistics ---
351 1 packets transmitted, 1 received, 0% packet loss, time \d*ms.*
356 # asserts at the end, to make sure there's proper cleanup
357 self.assertTrue(re.match(comp_result, ping_result, re.MULTILINE),
358 "Unexpected trace:\n" + ping_result)
361 if __name__ == '__main__':