Commit 0ee00fb2 authored by Ian Fischer's avatar Ian Fischer

Add a verbose flag to sky_tool that logs all of shell commands being run.

Also clean up some formatting issues.
parent d2e32b80
...@@ -75,6 +75,7 @@ def _start_http_server(port, root): ...@@ -75,6 +75,7 @@ def _start_http_server(port, root):
server_command = [ server_command = [
PUB_PATH, 'run', 'sky_tools:sky_server', str(port), PUB_PATH, 'run', 'sky_tools:sky_server', str(port),
] ]
logging.info(' '.join(server_command))
return subprocess.Popen(server_command, cwd=root).pid return subprocess.Popen(server_command, cwd=root).pid
...@@ -158,6 +159,7 @@ class StartSky(object): ...@@ -158,6 +159,7 @@ class StartSky(object):
def _is_package_installed(self, package_name): def _is_package_installed(self, package_name):
pm_path_cmd = [ADB_PATH, 'shell', 'pm', 'path', package_name] pm_path_cmd = [ADB_PATH, 'shell', 'pm', 'path', package_name]
logging.info(' '.join(pm_path_cmd))
return subprocess.check_output(pm_path_cmd).strip() != '' return subprocess.check_output(pm_path_cmd).strip() != ''
def _is_valid_script_path(self): def _is_valid_script_path(self):
...@@ -167,7 +169,9 @@ class StartSky(object): ...@@ -167,7 +169,9 @@ class StartSky(object):
def _get_device_apk_sha1(self, apk_path): def _get_device_apk_sha1(self, apk_path):
# We might need to install a new APK, so check SHA1 # We might need to install a new APK, so check SHA1
return subprocess.check_output([ADB_PATH, 'shell', 'cat', SHA1_PATH]) cmd = [ADB_PATH, 'shell', 'cat', SHA1_PATH]
logging.info(' '.join(cmd))
return subprocess.check_output(cmd)
def run(self, args, pids): def run(self, args, pids):
if not args.poke: if not args.poke:
...@@ -221,12 +225,15 @@ class StartSky(object): ...@@ -221,12 +225,15 @@ class StartSky(object):
return 2 return 2
cmd = [ADB_PATH, 'install', '-r', apk_path] cmd = [ADB_PATH, 'install', '-r', apk_path]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
# record the SHA1 of the APK we just pushed # record the SHA1 of the APK we just pushed
with tempfile.NamedTemporaryFile() as fp: with tempfile.NamedTemporaryFile() as fp:
fp.write(source_sha1) fp.write(source_sha1)
fp.seek(0) fp.seek(0)
subprocess.check_call([ADB_PATH, 'push', fp.name, SHA1_PATH]) cmd = [ADB_PATH, 'push', fp.name, SHA1_PATH]
logging.info(' '.join(cmd))
subprocess.check_call(cmd)
# Install on connected iOS device # Install on connected iOS device
if IOSDevice.is_connected() and args.ios_build_available: if IOSDevice.is_connected() and args.ios_build_available:
...@@ -246,9 +253,14 @@ class StartSky(object): ...@@ -246,9 +253,14 @@ class StartSky(object):
# Set up port forwarding for observatory # Set up port forwarding for observatory
observatory_port_string = 'tcp:%s' % OBSERVATORY_PORT observatory_port_string = 'tcp:%s' % OBSERVATORY_PORT
subprocess.check_call([ cmd = [
ADB_PATH, 'forward', observatory_port_string, observatory_port_string ADB_PATH,
]) 'forward',
observatory_port_string,
observatory_port_string
]
logging.info(' '.join(cmd))
subprocess.check_call(cmd)
sky_server_port = SKY_SERVER_PORT sky_server_port = SKY_SERVER_PORT
pids['sky_server_port'] = sky_server_port pids['sky_server_port'] = sky_server_port
...@@ -261,9 +273,14 @@ class StartSky(object): ...@@ -261,9 +273,14 @@ class StartSky(object):
pids['sky_server_root'] = sky_server_root pids['sky_server_root'] = sky_server_root
port_string = 'tcp:%s' % sky_server_port port_string = 'tcp:%s' % sky_server_port
subprocess.check_call([ cmd = [
ADB_PATH, 'reverse', port_string, port_string ADB_PATH,
]) 'reverse',
port_string,
port_string
]
logging.info(' '.join(cmd))
subprocess.check_call(cmd)
pids['remote_sky_server_port'] = sky_server_port pids['remote_sky_server_port'] = sky_server_port
# The load happens on the remote device, use the remote port. # The load happens on the remote device, use the remote port.
...@@ -283,7 +300,7 @@ class StartSky(object): ...@@ -283,7 +300,7 @@ class StartSky(object):
cmd += [ '--ez', 'enable-checked-mode', 'true' ] cmd += [ '--ez', 'enable-checked-mode', 'true' ]
cmd += [ ANDROID_COMPONENT ] cmd += [ ANDROID_COMPONENT ]
logging.info(' '.join(cmd))
subprocess.check_output(cmd) subprocess.check_output(cmd)
...@@ -295,6 +312,7 @@ class StopSky(object): ...@@ -295,6 +312,7 @@ class StopSky(object):
def _run(self, args): def _run(self, args):
with open('/dev/null', 'w') as dev_null: with open('/dev/null', 'w') as dev_null:
logging.info(' '.join(args))
subprocess.call(args, stdout=dev_null, stderr=dev_null) subprocess.call(args, stdout=dev_null, stderr=dev_null)
def run(self, args, pids): def run(self, args, pids):
...@@ -320,6 +338,7 @@ class IOSDevice(object): ...@@ -320,6 +338,7 @@ class IOSDevice(object):
'which', 'which',
'ios-deploy' 'ios-deploy'
] ]
logging.info(' '.join(cmd))
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd)
match = re.search(r'ios-deploy', out) match = re.search(r'ios-deploy', out)
cls._has_ios_deploy = match is not None cls._has_ios_deploy = match is not None
...@@ -340,6 +359,7 @@ class IOSDevice(object): ...@@ -340,6 +359,7 @@ class IOSDevice(object):
'--timeout', '--timeout',
'1' '1'
] ]
logging.info(' '.join(cmd))
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd)
match = re.search(r'\[\.\.\.\.\] Found [^\)]*\) connected', out) match = re.search(r'\[\.\.\.\.\] Found [^\)]*\) connected', out)
cls._is_connected = match is not None cls._is_connected = match is not None
...@@ -357,6 +377,7 @@ class IOSDevice(object): ...@@ -357,6 +377,7 @@ class IOSDevice(object):
'--bundle', '--bundle',
ios_app_path ios_app_path
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
@classmethod @classmethod
...@@ -374,6 +395,7 @@ class IOSDevice(object): ...@@ -374,6 +395,7 @@ class IOSDevice(object):
'--to', '--to',
device_path device_path
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
...@@ -393,6 +415,7 @@ class IOSSimulator(object): ...@@ -393,6 +415,7 @@ class IOSSimulator(object):
'list', 'list',
'devices', 'devices',
] ]
logging.info(' '.join(cmd))
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd)
match = re.search(r'[^\(]+\(([^\)]+)\) \(Booted\)', out) match = re.search(r'[^\(]+\(([^\)]+)\) \(Booted\)', out)
if match is not None and match.group(1) is not None: if match is not None and match.group(1) is not None:
...@@ -431,6 +454,7 @@ class IOSSimulator(object): ...@@ -431,6 +454,7 @@ class IOSSimulator(object):
'-name', '-name',
SKY_SHELL_APP_ID SKY_SHELL_APP_ID
] ]
logging.info(' '.join(cmd))
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd)
match = re.search(r'Data\/Application\/([^\/]+)\/Documents\/' + SKY_SHELL_APP_ID, out) match = re.search(r'Data\/Application\/([^\/]+)\/Documents\/' + SKY_SHELL_APP_ID, out)
if match is not None and match.group(1) is not None: if match is not None and match.group(1) is not None:
...@@ -464,82 +488,91 @@ class IOSSimulator(object): ...@@ -464,82 +488,91 @@ class IOSSimulator(object):
os.path.abspath(__file__), os.path.abspath(__file__),
'ios_sim', 'ios_sim',
'-p', '-p',
# This path manipulation is to work around an issue where simctl fails to correctly parse
# paths that start with ../
ios_app_path, ios_app_path,
'launch' 'launch'
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
def get_application_identifier(self, path): def get_application_identifier(self, path):
identifier = subprocess.check_output(PLIST_BUDDY_PATH + [ cmd = PLIST_BUDDY_PATH + [
'-c', '-c',
'Print CFBundleIdentifier', 'Print CFBundleIdentifier',
os.path.join(path, 'Info.plist') os.path.join(path, 'Info.plist')
]) ]
return identifier.strip() logging.info(' '.join(cmd))
identifier = subprocess.check_output(cmd)
return identifier.strip()
def is_simulator_booted(self): def is_simulator_booted(self):
devices = subprocess.check_output(SIMCTL_PATH + [ 'list', 'devices' ]).strip().split('\n') cmd = SIMCTL_PATH + [ 'list', 'devices' ]
for device in devices: logging.info(' '.join(cmd))
if re.search(r'\(Booted\)', device): devices = subprocess.check_output(cmd).strip().split('\n')
return True for device in devices:
return False if re.search(r'\(Booted\)', device):
return True
return False
# Launch whatever simulator the user last used, rather than try to guess which of their simulators they might want to use # Launch whatever simulator the user last used, rather than try to guess which of their simulators they might want to use
def boot_simulator(self, args, pids): def boot_simulator(self, args, pids):
if self.is_simulator_booted(): if self.is_simulator_booted():
return return
# Use Popen here because launching the simulator from the command line in this manner doesn't return, so we can't check the result. # Use Popen here because launching the simulator from the command line in this manner doesn't return, so we can't check the result.
if args.ios_sim_path: if args.ios_sim_path:
subprocess.Popen(args.ios_sim_path) logging.info(args.ios_sim_path)
else: subprocess.Popen(args.ios_sim_path)
subprocess.Popen(IOS_SIM_PATH) else:
while not self.is_simulator_booted(): logging.info(IOS_SIM_PATH)
print('Waiting for iOS Simulator to boot...') subprocess.Popen(IOS_SIM_PATH)
time.sleep(0.5) while not self.is_simulator_booted():
print('Waiting for iOS Simulator to boot...')
time.sleep(0.5)
def install_app(self, args, pids): def install_app(self, args, pids):
self.boot_simulator(args, pids) self.boot_simulator(args, pids)
cmd = SIMCTL_PATH + [ cmd = SIMCTL_PATH + [
'install', 'install',
'booted', 'booted',
args.path, args.path,
] ]
return subprocess.check_call(cmd) logging.info(' '.join(cmd))
return subprocess.check_call(cmd)
def install_launch_and_wait(self, args, pids, wait): def install_launch_and_wait(self, args, pids, wait):
res = self.install_app(args, pids) res = self.install_app(args, pids)
if res != 0: if res != 0:
return res return res
identifier = self.get_application_identifier(args.path) identifier = self.get_application_identifier(args.path)
launch_args = [ 'launch' ] launch_args = SIMCTL_PATH + ['launch']
if wait: if wait:
launch_args += [ '-w' ] launch_args += [ '-w' ]
launch_args += [ launch_args += [
'booted', 'booted',
identifier, identifier,
'-target', '-target',
args.target, args.target,
'-server', '-server',
args.server args.server
] ]
return subprocess.check_output(SIMCTL_PATH + launch_args).strip() logging.info(' '.join(launch_args))
return subprocess.check_output(launch_args).strip()
def launch_app(self, args, pids): def launch_app(self, args, pids):
self.install_launch_and_wait(args, pids, False) self.install_launch_and_wait(args, pids, False)
def debug_app(self, args, pids): def debug_app(self, args, pids):
launch_res = self.install_launch_and_wait(args, pids, True) launch_res = self.install_launch_and_wait(args, pids, True)
launch_pid = re.search('.*: (\d+)', launch_res).group(1) launch_pid = re.search('.*: (\d+)', launch_res).group(1)
return os.system(' '.join(XCRUN_PATH + [ cmd = XCRUN_PATH + [
'lldb', 'lldb',
# TODO(iansf): get this working again # TODO(iansf): get this working again
# '-s', # '-s',
# os.path.join(os.path.dirname(__file__), 'lldb_start_commands.txt'), # os.path.join(os.path.dirname(__file__), 'lldb_start_commands.txt'),
'-p', '-p',
launch_pid, launch_pid,
])) ]
logging.info(' '.join(cmd))
return subprocess.call(cmd)
def add_subparser(self, subparsers): def add_subparser(self, subparsers):
simulator_parser = subparsers.add_parser('ios_sim', simulator_parser = subparsers.add_parser('ios_sim',
...@@ -588,6 +621,7 @@ class StartListening(object): ...@@ -588,6 +621,7 @@ class StartListening(object):
'which', 'which',
'inotifywait' 'inotifywait'
] ]
logging.info(' '.join(cmd))
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd)
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
logging.error('"listen" command is only useful if you have installed inotifywait on Linux. Run "apt-get install inotify-tools" or equivalent to install it.') logging.error('"listen" command is only useful if you have installed inotifywait on Linux. Run "apt-get install inotify-tools" or equivalent to install it.')
...@@ -606,6 +640,7 @@ class StartListening(object): ...@@ -606,6 +640,7 @@ class StartListening(object):
'which', 'which',
'fswatch' 'fswatch'
] ]
logging.info(' '.join(cmd))
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd)
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
logging.error('"listen" command is only useful if you have installed fswatch on Mac. Run "brew install fswatch" to install it with homebrew.') logging.error('"listen" command is only useful if you have installed fswatch on Mac. Run "brew install fswatch" to install it with homebrew.')
...@@ -622,6 +657,7 @@ class StartListening(object): ...@@ -622,6 +657,7 @@ class StartListening(object):
logging.error('"listen" command is only available on Mac and Linux.') logging.error('"listen" command is only available on Mac and Linux.')
return False return False
logging.info(' '.join(self.watch_cmd))
subprocess.check_call(self.watch_cmd) subprocess.check_call(self.watch_cmd)
return True return True
...@@ -643,6 +679,7 @@ class StartListening(object): ...@@ -643,6 +679,7 @@ class StartListening(object):
'start', 'start',
'--poke' '--poke'
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
if args.local_build: if args.local_build:
...@@ -664,6 +701,7 @@ class StartListening(object): ...@@ -664,6 +701,7 @@ class StartListening(object):
'--snapshot=' + os.path.join(tempdir, 'snapshot_blob.bin'), '--snapshot=' + os.path.join(tempdir, 'snapshot_blob.bin'),
os.path.join('lib', 'main.dart') os.path.join('lib', 'main.dart')
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
os.chdir(tempdir) os.chdir(tempdir)
...@@ -677,6 +715,7 @@ class StartListening(object): ...@@ -677,6 +715,7 @@ class StartListening(object):
'content', 'content',
'navigation' 'navigation'
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
os.chdir(currdir) os.chdir(currdir)
...@@ -688,6 +727,7 @@ class StartListening(object): ...@@ -688,6 +727,7 @@ class StartListening(object):
os.path.join(tempdir, 'app.skyx'), os.path.join(tempdir, 'app.skyx'),
simulator_app_documents_dir simulator_app_documents_dir
] ]
logging.info(' '.join(cmd))
subprocess.check_call(cmd) subprocess.check_call(cmd)
# Copy the app.skyx to the attached iOS device # Copy the app.skyx to the attached iOS device
...@@ -706,9 +746,16 @@ class StartTracing(object): ...@@ -706,9 +746,16 @@ class StartTracing(object):
start_tracing_parser.set_defaults(func=self.run) start_tracing_parser.set_defaults(func=self.run)
def run(self, args, pids): def run(self, args, pids):
subprocess.check_output([ADB_PATH, 'shell', cmd = [
'am', 'broadcast', ADB_PATH,
'-a', 'org.domokit.sky.shell.TRACING_START']) 'shell',
'am',
'broadcast',
'-a',
'org.domokit.sky.shell.TRACING_START'
]
logging.info(' '.join(cmd))
subprocess.check_output(cmd)
TRACE_COMPLETE_REGEXP = re.compile('Trace complete') TRACE_COMPLETE_REGEXP = re.compile('Trace complete')
...@@ -722,15 +769,28 @@ class StopTracing(object): ...@@ -722,15 +769,28 @@ class StopTracing(object):
stop_tracing_parser.set_defaults(func=self.run) stop_tracing_parser.set_defaults(func=self.run)
def run(self, args, pids): def run(self, args, pids):
subprocess.check_output([ADB_PATH, 'logcat', '-c']) cmd = [ADB_PATH, 'logcat', '-c']
subprocess.check_output([ADB_PATH, 'shell', logging.info(' '.join(cmd))
'am', 'broadcast', subprocess.check_output(cmd)
'-a', 'org.domokit.sky.shell.TRACING_STOP'])
cmd = [
ADB_PATH,
'shell',
'am',
'broadcast',
'-a',
'org.domokit.sky.shell.TRACING_STOP'
]
logging.info(' '.join(cmd))
subprocess.check_output(cmd)
device_path = None device_path = None
is_complete = False is_complete = False
while not is_complete: while not is_complete:
time.sleep(0.2) time.sleep(0.2)
log = subprocess.check_output([ADB_PATH, 'logcat', '-d']) cmd = [ADB_PATH, 'logcat', '-d']
logging.info(' '.join(cmd))
log = subprocess.check_output(cmd)
if device_path is None: if device_path is None:
result = TRACE_FILE_REGEXP.search(log) result = TRACE_FILE_REGEXP.search(log)
if result: if result:
...@@ -740,8 +800,13 @@ class StopTracing(object): ...@@ -740,8 +800,13 @@ class StopTracing(object):
logging.info('Downloading trace %s ...' % os.path.basename(device_path)) logging.info('Downloading trace %s ...' % os.path.basename(device_path))
if device_path: if device_path:
subprocess.check_output([ADB_PATH, 'pull', device_path]) cmd = [ADB_PATH, 'pull', device_path]
subprocess.check_output([ADB_PATH, 'shell', 'rm', device_path]) logging.info(' '.join(cmd))
subprocess.check_output(cmd)
cmd = [ADB_PATH, 'shell', 'rm', device_path]
logging.info(' '.join(cmd))
subprocess.check_output(cmd)
class SkyShellRunner(object): class SkyShellRunner(object):
...@@ -771,11 +836,15 @@ class SkyShellRunner(object): ...@@ -771,11 +836,15 @@ class SkyShellRunner(object):
def _check_for_adb(self): def _check_for_adb(self):
try: try:
adb_version = subprocess.check_output([ADB_PATH, 'version']) cmd = [ADB_PATH, 'version']
logging.info(' '.join(cmd))
adb_version = subprocess.check_output(cmd)
if self._is_valid_adb_version(adb_version): if self._is_valid_adb_version(adb_version):
return True return True
adb_path = subprocess.check_output(['which', ADB_PATH]).rstrip() cmd = ['which', ADB_PATH]
logging.info(' '.join(cmd))
adb_path = subprocess.check_output(cmd).rstrip()
logging.error('"%s" is too old. Need 1.0.32 or later. ' logging.error('"%s" is too old. Need 1.0.32 or later. '
'Try setting ANDROID_HOME.' % adb_path) 'Try setting ANDROID_HOME.' % adb_path)
return False return False
...@@ -791,10 +860,13 @@ class SkyShellRunner(object): ...@@ -791,10 +860,13 @@ class SkyShellRunner(object):
# output lines like this, which we want to ignore: # output lines like this, which we want to ignore:
# adb server is out of date. killing.. # adb server is out of date. killing..
# * daemon started successfully * # * daemon started successfully *
cmd = [ADB_PATH, 'start-server']
logging.info(' '.join(cmd))
subprocess.call(cmd)
subprocess.call([ADB_PATH, 'start-server']) cmd = [ADB_PATH, 'shell', 'getprop', 'ro.build.version.sdk']
sdk_version = subprocess.check_output( logging.info(' '.join(cmd))
[ADB_PATH, 'shell', 'getprop', 'ro.build.version.sdk']).rstrip() sdk_version = subprocess.check_output(cmd).rstrip()
# Sample output: '22' # Sample output: '22'
if not sdk_version.isdigit(): if not sdk_version.isdigit():
logging.error('Unexpected response from getprop: "%s".' % sdk_version) logging.error('Unexpected response from getprop: "%s".' % sdk_version)
...@@ -812,14 +884,16 @@ class SkyShellRunner(object): ...@@ -812,14 +884,16 @@ class SkyShellRunner(object):
def _check_for_dart(self): def _check_for_dart(self):
try: try:
subprocess.check_output([DART_PATH, '--version'], stderr=subprocess.STDOUT) cmd = [DART_PATH, '--version']
logging.info(' '.join(cmd))
subprocess.check_output(cmd, stderr=subprocess.STDOUT)
except OSError: except OSError:
logging.error('"dart" (from the Dart SDK) not in $PATH, cannot continue.') logging.error('"dart" (from the Dart SDK) not in $PATH, cannot continue.')
return False return False
return True return True
def main(self): def main(self):
logging.basicConfig(format='%(levelname)s:%(message)s', level=logging.INFO) logging.basicConfig(format='%(levelname)s: %(message)s', level=logging.WARNING)
self._update_paths() self._update_paths()
...@@ -832,6 +906,8 @@ class SkyShellRunner(object): ...@@ -832,6 +906,8 @@ class SkyShellRunner(object):
sys.exit(2) sys.exit(2)
parser = argparse.ArgumentParser(description='Sky App Runner') parser = argparse.ArgumentParser(description='Sky App Runner')
parser.add_argument('--verbose', dest='verbose', action='store_true',
help='Noisy logging, including all shell commands executed')
parser.add_argument('--release', dest='use_release', action='store_true', parser.add_argument('--release', dest='use_release', action='store_true',
help='Set this if you are building Sky locally and want to use the release build products. ' help='Set this if you are building Sky locally and want to use the release build products. '
'When set, attempts to automaticaly determine sky-src-path if sky-src-path is ' 'When set, attempts to automaticaly determine sky-src-path if sky-src-path is '
...@@ -876,6 +952,9 @@ class SkyShellRunner(object): ...@@ -876,6 +952,9 @@ class SkyShellRunner(object):
command.add_subparser(subparsers) command.add_subparser(subparsers)
args = parser.parse_args() args = parser.parse_args()
if args.verbose:
logging.getLogger().setLevel(logging.INFO)
if args.use_release: if args.use_release:
args.local_build = True args.local_build = True
...@@ -933,11 +1012,11 @@ class SkyShellRunner(object): ...@@ -933,11 +1012,11 @@ class SkyShellRunner(object):
atexit.register(pids.write_to, PID_FILE_PATH) atexit.register(pids.write_to, PID_FILE_PATH)
exit_code = 0 exit_code = 0
try: try:
exit_code = args.func(args, pids) exit_code = args.func(args, pids)
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
# Don't print a stack trace if the adb command fails. # Don't print a stack trace if the adb command fails.
logging.error(e) logging.error(e)
exit_code = 2 exit_code = 2
sys.exit(exit_code) sys.exit(exit_code)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment