# Copyright (c) 2011 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import re, socket, subprocess from autotest_lib.client.common_lib import error class HostRoute(object): """ Host Route: A utility for retrieving information about our route to a host """ def __init__(self, host): self.host = host # Remote host self.calculate() def calculate(self): output = self.run_command(["ip", "route", "get", self.host]) # This converts "172.22.18.53 via 10.0.0.1 dev eth0 src 10.0.0.200 \n.." # into ("via", "10.0.0.1", "dev", "eth0", "src", "10.0.0.200") route_info = re.split("\s*", output.split("\n")[0].rstrip(' '))[1:] # Further, convert the list into a dict {"via": "10.0.0.1", ...} self.route_info = dict(tuple(route_info[i:i+2]) for i in range(0, len(route_info), 2)) if 'src' not in self.route_info: raise error.TestFail('Cannot find route to host %s' % self.host) class LocalHostRoute(HostRoute): """ Self Host Route: Retrieve host route for the test-host machine """ def __init__(self, host): # TODO(pstew): If we could depend on the host having the "ip" command # we would just be able to do this: # # HostRoute.__init__(self, host) # # but alas, we can't depend on this, so we fake it by creating a # socket and figuring out what local address we bound to if we # connected to the client sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) sock.connect((host, 22)) # NB: Port doesn't matter self.route_info = { 'src': sock.getsockname()[0] } def run_command(self, args): return subprocess.Popen(args, stdout=subprocess.PIPE).communicate()[0] class RemoteHostRoute(HostRoute): """ Remote Host Route: Retrieve host route for a remote (DUT, server) machine """ def __init__(self, remote, host): self.remote = remote HostRoute.__init__(self, host) def run_command(self, args): return self.remote.run(' '.join(args)).stdout