| OLD | NEW | 
|---|
| 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
| 2 # Use of this source code is governed by a BSD-style license that can be | 2 # Use of this source code is governed by a BSD-style license that can be | 
| 3 # found in the LICENSE file. | 3 # found in the LICENSE file. | 
| 4 | 4 | 
| 5 """Functions that deals with local and device ports.""" | 5 """Functions that deal with local and device ports.""" | 
| 6 | 6 | 
| 7 import contextlib | 7 import contextlib | 
| 8 import fcntl | 8 import fcntl | 
| 9 import httplib | 9 import httplib | 
| 10 import logging | 10 import logging | 
| 11 import os | 11 import os | 
| 12 import re | 12 import re | 
| 13 import socket | 13 import socket | 
| 14 import traceback | 14 import traceback | 
| 15 | 15 | 
| 16 import cmd_helper | 16 import cmd_helper | 
| 17 import constants | 17 import constants | 
| 18 | 18 | 
| 19 | 19 | 
| 20 #The following two methods are used to allocate the port source for various | 20 # The following two methods are used to allocate the port source for various | 
| 21 # types of test servers. Because some net relates tests can be run on shards | 21 # types of test servers. Because some net-related tests can be run on shards at | 
| 22 # at same time, it's important to have a mechanism to allocate the port process | 22 # same time, it's important to have a mechanism to allocate the port | 
| 23 # safe. In here, we implement the safe port allocation by leveraging flock. | 23 # process-safe. In here, we implement the safe port allocation by leveraging | 
|  | 24 # flock. | 
| 24 def ResetTestServerPortAllocation(): | 25 def ResetTestServerPortAllocation(): | 
| 25   """Reset the port allocation to start from TEST_SERVER_PORT_FIRST. | 26   """Resets the port allocation to start from TEST_SERVER_PORT_FIRST. | 
| 26 | 27 | 
| 27   Returns: | 28   Returns: | 
| 28     Returns True if reset successes. Otherwise returns False. | 29     Returns True if reset successes. Otherwise returns False. | 
| 29   """ | 30   """ | 
| 30   try: | 31   try: | 
| 31     with open(constants.TEST_SERVER_PORT_FILE, 'w') as fp: | 32     with open(constants.TEST_SERVER_PORT_FILE, 'w') as fp: | 
| 32       fp.write('%d' % constants.TEST_SERVER_PORT_FIRST) | 33       fp.write('%d' % constants.TEST_SERVER_PORT_FIRST) | 
| 33     if os.path.exists(constants.TEST_SERVER_PORT_LOCKFILE): | 34     if os.path.exists(constants.TEST_SERVER_PORT_LOCKFILE): | 
| 34       os.unlink(constants.TEST_SERVER_PORT_LOCKFILE) | 35       os.unlink(constants.TEST_SERVER_PORT_LOCKFILE) | 
| 35     return True | 36     return True | 
| 36   except Exception as e: | 37   except Exception as e: | 
| 37     logging.error(e) | 38     logging.error(e) | 
| 38   return False | 39   return False | 
| 39 | 40 | 
| 40 | 41 | 
| 41 def AllocateTestServerPort(): | 42 def AllocateTestServerPort(): | 
| 42   """Allocate a port incrementally. | 43   """Allocates a port incrementally. | 
| 43 | 44 | 
| 44   Returns: | 45   Returns: | 
| 45     Returns a valid port which should be in between TEST_SERVER_PORT_FIRST and | 46     Returns a valid port which should be in between TEST_SERVER_PORT_FIRST and | 
| 46     TEST_SERVER_PORT_LAST. Returning 0 means no more valid port can be used. | 47     TEST_SERVER_PORT_LAST. Returning 0 means no more valid port can be used. | 
| 47   """ | 48   """ | 
| 48   port = 0 | 49   port = 0 | 
| 49   ports_tried = [] | 50   ports_tried = [] | 
| 50   try: | 51   try: | 
| 51     fp_lock = open(constants.TEST_SERVER_PORT_LOCKFILE, 'w') | 52     fp_lock = open(constants.TEST_SERVER_PORT_LOCKFILE, 'w') | 
| 52     fcntl.flock(fp_lock, fcntl.LOCK_EX) | 53     fcntl.flock(fp_lock, fcntl.LOCK_EX) | 
| (...skipping 30 matching lines...) Expand all  Loading... | 
| 83   """Checks whether the specified host port is used or not. | 84   """Checks whether the specified host port is used or not. | 
| 84 | 85 | 
| 85   Uses -n -P to inhibit the conversion of host/port numbers to host/port names. | 86   Uses -n -P to inhibit the conversion of host/port numbers to host/port names. | 
| 86 | 87 | 
| 87   Args: | 88   Args: | 
| 88     host_port: Port on host we want to check. | 89     host_port: Port on host we want to check. | 
| 89 | 90 | 
| 90   Returns: | 91   Returns: | 
| 91     True if the port on host is already used, otherwise returns False. | 92     True if the port on host is already used, otherwise returns False. | 
| 92   """ | 93   """ | 
| 93   port_info = '(127\.0\.0\.1)|(localhost)\:%d' % host_port | 94   port_info = '(\*)|(127\.0\.0\.1)|(localhost):%d' % host_port | 
| 94   # TODO(jnd): Find a better way to filter the port. | 95   # TODO(jnd): Find a better way to filter the port. Note that connecting to the | 
|  | 96   # socket and closing it would leave it in the TIME_WAIT state. Setting | 
|  | 97   # SO_LINGER on it and then closing it makes the Python HTTP server crash. | 
| 95   re_port = re.compile(port_info, re.MULTILINE) | 98   re_port = re.compile(port_info, re.MULTILINE) | 
| 96   if re_port.findall(cmd_helper.GetCmdOutput(['lsof', '-nPi:%d' % host_port])): | 99   if re_port.search(cmd_helper.GetCmdOutput(['lsof', '-nPi:%d' % host_port])): | 
| 97     return True | 100     return True | 
| 98   return False | 101   return False | 
| 99 | 102 | 
| 100 | 103 | 
| 101 def IsDevicePortUsed(adb, device_port, state=''): | 104 def IsDevicePortUsed(adb, device_port, state=''): | 
| 102   """Checks whether the specified device port is used or not. | 105   """Checks whether the specified device port is used or not. | 
| 103 | 106 | 
| 104   Args: | 107   Args: | 
| 105     adb: Instance of AndroidCommands for talking to the device. | 108     adb: Instance of AndroidCommands for talking to the device. | 
| 106     device_port: Port on device we want to check. | 109     device_port: Port on device we want to check. | 
| 107     state: String of the specified state. Default is empty string, which | 110     state: String of the specified state. Default is empty string, which | 
| 108            means any state. | 111            means any state. | 
| 109 | 112 | 
| 110   Returns: | 113   Returns: | 
| 111     True if the port on device is already used, otherwise returns False. | 114     True if the port on device is already used, otherwise returns False. | 
| 112   """ | 115   """ | 
| 113   base_url = '127.0.0.1:%d' % device_port | 116   base_url = '127.0.0.1:%d' % device_port | 
| 114   netstat_results = adb.RunShellCommand('netstat', log_result=False) | 117   netstat_results = adb.RunShellCommand('netstat', log_result=False) | 
| 115   for single_connect in netstat_results: | 118   for single_connect in netstat_results: | 
| 116     # Column 3 is the local address which we want to check with. | 119     # Column 3 is the local address which we want to check with. | 
| 117     connect_results = single_connect.split() | 120     connect_results = single_connect.split() | 
|  | 121     if connect_results[0] != 'tcp': | 
|  | 122       continue | 
|  | 123     if len(connect_results) < 6: | 
|  | 124       raise Exception('Unexpected format while parsing netstat line: ' + | 
|  | 125                       single_connect) | 
| 118     is_state_match = connect_results[5] == state if state else True | 126     is_state_match = connect_results[5] == state if state else True | 
| 119     if connect_results[3] == base_url and is_state_match: | 127     if connect_results[3] == base_url and is_state_match: | 
| 120       return True | 128       return True | 
| 121   return False | 129   return False | 
| 122 | 130 | 
| 123 | 131 | 
| 124 def IsHttpServerConnectable(host, port, tries=3, command='GET', path='/', | 132 def IsHttpServerConnectable(host, port, tries=3, command='GET', path='/', | 
| 125                             expected_read='', timeout=2): | 133                             expected_read='', timeout=2): | 
| 126   """Checks whether the specified http server is ready to serve request or not. | 134   """Checks whether the specified http server is ready to serve request or not. | 
| 127 | 135 | 
| (...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 159         client_error = ('Bad response: %s %s version %s\n  ' % | 167         client_error = ('Bad response: %s %s version %s\n  ' % | 
| 160                         (r.status, r.reason, r.version) + | 168                         (r.status, r.reason, r.version) + | 
| 161                         '\n  '.join([': '.join(h) for h in r.getheaders()])) | 169                         '\n  '.join([': '.join(h) for h in r.getheaders()])) | 
| 162     except (httplib.HTTPException, socket.error) as e: | 170     except (httplib.HTTPException, socket.error) as e: | 
| 163       # Probably too quick connecting: try again. | 171       # Probably too quick connecting: try again. | 
| 164       exception_error_msgs = traceback.format_exception_only(type(e), e) | 172       exception_error_msgs = traceback.format_exception_only(type(e), e) | 
| 165       if exception_error_msgs: | 173       if exception_error_msgs: | 
| 166         client_error = ''.join(exception_error_msgs) | 174         client_error = ''.join(exception_error_msgs) | 
| 167   # Only returns last client_error. | 175   # Only returns last client_error. | 
| 168   return (False, client_error or 'Timeout') | 176   return (False, client_error or 'Timeout') | 
| OLD | NEW | 
|---|