2017-04-05 11:32:45 +00:00
|
|
|
# Copyright 2016 The Chromium Authors. All rights reserved.
|
|
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
|
|
# found in the LICENSE file.
|
|
|
|
|
2017-04-19 17:36:00 +00:00
|
|
|
|
2017-04-05 11:32:45 +00:00
|
|
|
from recipe_engine import recipe_api
|
|
|
|
|
2018-05-24 13:14:18 +00:00
|
|
|
import default
|
|
|
|
import json # TODO(borenet): No! Remove this.
|
2017-04-05 11:32:45 +00:00
|
|
|
|
|
|
|
|
2018-05-24 13:14:18 +00:00
|
|
|
"""Chromebook flavor, used for running code on Chromebooks."""
|
|
|
|
|
|
|
|
|
|
|
|
class ChromebookFlavor(default.DefaultFlavor):
|
2017-04-05 11:32:45 +00:00
|
|
|
|
2017-04-07 14:04:08 +00:00
|
|
|
def __init__(self, m):
|
2018-05-24 13:14:18 +00:00
|
|
|
super(ChromebookFlavor, self).__init__(m)
|
2017-04-07 14:04:08 +00:00
|
|
|
self._user_ip = ''
|
|
|
|
|
2018-05-18 11:36:55 +00:00
|
|
|
self.chromeos_homedir = '/home/chronos/user/'
|
2018-05-24 13:14:18 +00:00
|
|
|
self.device_dirs = default.DeviceDirs(
|
2018-05-18 11:36:55 +00:00
|
|
|
bin_dir = self.chromeos_homedir + 'bin',
|
|
|
|
dm_dir = self.chromeos_homedir + 'dm_out',
|
|
|
|
perf_data_dir = self.chromeos_homedir + 'perf',
|
|
|
|
resource_dir = self.chromeos_homedir + 'resources',
|
|
|
|
images_dir = self.chromeos_homedir + 'images',
|
|
|
|
skp_dir = self.chromeos_homedir + 'skps',
|
|
|
|
svg_dir = self.chromeos_homedir + 'svgs',
|
|
|
|
tmp_dir = self.chromeos_homedir)
|
2017-04-07 14:04:08 +00:00
|
|
|
|
2017-04-19 17:36:00 +00:00
|
|
|
@property
|
|
|
|
def user_ip(self):
|
2017-04-19 17:23:31 +00:00
|
|
|
if not self._user_ip:
|
2017-04-19 17:36:00 +00:00
|
|
|
ssh_info = self.m.run(self.m.python.inline, 'read chromeos ip',
|
|
|
|
program="""
|
|
|
|
import os
|
|
|
|
SSH_MACHINE_FILE = os.path.expanduser('~/ssh_machine.json')
|
|
|
|
with open(SSH_MACHINE_FILE, 'r') as f:
|
|
|
|
print f.read()
|
|
|
|
""",
|
|
|
|
stdout=self.m.raw_io.output(),
|
|
|
|
infra_step=True).stdout
|
|
|
|
|
|
|
|
self._user_ip = json.loads(ssh_info).get(u'user_ip', 'ERROR')
|
|
|
|
return self._user_ip
|
2017-04-19 17:23:31 +00:00
|
|
|
|
2017-04-19 17:36:00 +00:00
|
|
|
def _ssh(self, title, *cmd, **kwargs):
|
2017-04-07 14:04:08 +00:00
|
|
|
if 'infra_step' not in kwargs:
|
|
|
|
kwargs['infra_step'] = True
|
|
|
|
|
|
|
|
ssh_cmd = ['ssh', '-oConnectTimeout=15', '-oBatchMode=yes',
|
2017-04-19 17:36:00 +00:00
|
|
|
'-t', '-t', self.user_ip] + list(cmd)
|
2017-04-07 14:04:08 +00:00
|
|
|
|
|
|
|
return self._run(title, ssh_cmd, **kwargs)
|
|
|
|
|
|
|
|
def install(self):
|
|
|
|
self._ssh('mkdir %s' % self.device_dirs.resource_dir, 'mkdir', '-p',
|
|
|
|
self.device_dirs.resource_dir)
|
|
|
|
|
|
|
|
# Ensure the home dir is marked executable
|
2018-05-18 11:36:55 +00:00
|
|
|
self._ssh('remount %s as exec' % self.chromeos_homedir,
|
2017-04-07 14:04:08 +00:00
|
|
|
'sudo', 'mount', '-i', '-o', 'remount,exec', '/home/chronos')
|
|
|
|
|
2018-05-18 11:36:55 +00:00
|
|
|
self.create_clean_device_dir(self.device_dirs.bin_dir)
|
2017-04-07 14:04:08 +00:00
|
|
|
|
|
|
|
def create_clean_device_dir(self, path):
|
|
|
|
# use -f to silently return if path doesn't exist
|
|
|
|
self._ssh('rm %s' % path, 'rm', '-rf', path)
|
|
|
|
self._ssh('mkdir %s' % path, 'mkdir', '-p', path)
|
|
|
|
|
2017-04-24 19:59:55 +00:00
|
|
|
def read_file_on_device(self, path, **kwargs):
|
|
|
|
rv = self._ssh('read %s' % path,
|
|
|
|
'cat', path, stdout=self.m.raw_io.output(),
|
|
|
|
**kwargs)
|
|
|
|
return rv.stdout.rstrip() if rv and rv.stdout else None
|
2017-04-07 14:04:08 +00:00
|
|
|
|
|
|
|
def remove_file_on_device(self, path):
|
|
|
|
# use -f to silently return if path doesn't exist
|
|
|
|
self._ssh('rm %s' % path, 'rm', '-f', path)
|
|
|
|
|
|
|
|
def _prefix_device_path(self, device_path):
|
2017-04-19 17:36:00 +00:00
|
|
|
return '%s:%s' % (self.user_ip, device_path)
|
2017-04-07 14:04:08 +00:00
|
|
|
|
|
|
|
def copy_file_to_device(self, host_path, device_path):
|
|
|
|
device_path = self._prefix_device_path(device_path)
|
|
|
|
# Recipe
|
|
|
|
self.m.python.inline(str('scp %s %s' % (host_path, device_path)),
|
|
|
|
"""
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
host = sys.argv[1]
|
|
|
|
device = sys.argv[2]
|
|
|
|
print subprocess.check_output(['scp', host, device])
|
|
|
|
""", args=[host_path, device_path], infra_step=True)
|
|
|
|
|
|
|
|
def _copy_dir(self, src, dest):
|
|
|
|
# We can't use rsync to communicate with the chromebooks because the
|
|
|
|
# chromebooks don't have rsync installed on them.
|
|
|
|
self.m.python.inline(str('scp -r %s %s' % (src, dest)),
|
|
|
|
"""
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
src = sys.argv[1] + '/*'
|
|
|
|
dest = sys.argv[2]
|
|
|
|
print subprocess.check_output('scp -r %s %s' % (src, dest), shell=True)
|
|
|
|
""", args=[src, dest], infra_step=True)
|
|
|
|
|
|
|
|
def copy_directory_contents_to_device(self, host_path, device_path):
|
|
|
|
self._copy_dir(host_path, self._prefix_device_path(device_path))
|
|
|
|
|
|
|
|
def copy_directory_contents_to_host(self, device_path, host_path):
|
|
|
|
self._copy_dir(self._prefix_device_path(device_path), host_path)
|
|
|
|
|
|
|
|
def step(self, name, cmd, **kwargs):
|
|
|
|
# Push and run either dm or nanobench
|
|
|
|
|
|
|
|
name = cmd[0]
|
2017-10-12 19:15:47 +00:00
|
|
|
|
|
|
|
if name == 'dm':
|
2018-05-18 11:36:55 +00:00
|
|
|
self.create_clean_host_dir(self.host_dirs.dm_dir)
|
2017-10-12 19:15:47 +00:00
|
|
|
if name == 'nanobench':
|
2018-05-18 11:36:55 +00:00
|
|
|
self.create_clean_host_dir(self.host_dirs.perf_data_dir)
|
2017-10-12 19:15:47 +00:00
|
|
|
|
2018-06-01 14:08:53 +00:00
|
|
|
app = self.host_dirs.bin_dir.join(cmd[0])
|
2017-04-07 14:04:08 +00:00
|
|
|
|
2018-05-18 11:36:55 +00:00
|
|
|
cmd[0] = '%s/%s' % (self.device_dirs.bin_dir, cmd[0])
|
2017-04-07 14:04:08 +00:00
|
|
|
self.copy_file_to_device(app, cmd[0])
|
|
|
|
|
|
|
|
self._ssh('chmod %s' % name, 'chmod', '+x', cmd[0])
|
|
|
|
self._ssh(str(name), *cmd)
|