mirror of
https://github.com/ARM-software/workload-automation.git
synced 2025-01-19 12:24:32 +00:00
301 lines
13 KiB
Python
301 lines
13 KiB
Python
|
# Copyright 2013-2015 ARM Limited
|
||
|
#
|
||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
# you may not use this file except in compliance with the License.
|
||
|
# You may obtain a copy of the License at
|
||
|
#
|
||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||
|
#
|
||
|
# Unless required by applicable law or agreed to in writing, software
|
||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
# See the License for the specific language governing permissions and
|
||
|
# limitations under the License.
|
||
|
#
|
||
|
|
||
|
|
||
|
import os
|
||
|
import stat
|
||
|
import string
|
||
|
import textwrap
|
||
|
import argparse
|
||
|
import shutil
|
||
|
import getpass
|
||
|
|
||
|
from wlauto import ExtensionLoader, Command, settings
|
||
|
from wlauto.exceptions import CommandError
|
||
|
from wlauto.utils.cli import init_argument_parser
|
||
|
from wlauto.utils.misc import (capitalize, check_output,
|
||
|
ensure_file_directory_exists as _f, ensure_directory_exists as _d)
|
||
|
from wlauto.utils.types import identifier
|
||
|
from wlauto.utils.doc import format_body
|
||
|
|
||
|
|
||
|
__all__ = ['create_workload']
|
||
|
|
||
|
|
||
|
TEMPLATES_DIR = os.path.join(os.path.dirname(__file__), 'templates')
|
||
|
|
||
|
UIAUTO_BUILD_SCRIPT = """#!/bin/bash
|
||
|
|
||
|
class_dir=bin/classes/com/arm/wlauto/uiauto
|
||
|
base_class=`python -c "import os, wlauto; print os.path.join(os.path.dirname(wlauto.__file__), 'common', 'android', 'BaseUiAutomation.class')"`
|
||
|
mkdir -p $$class_dir
|
||
|
cp $$base_class $$class_dir
|
||
|
|
||
|
ant build
|
||
|
|
||
|
if [[ -f bin/${package_name}.jar ]]; then
|
||
|
cp bin/${package_name}.jar ..
|
||
|
fi
|
||
|
"""
|
||
|
|
||
|
|
||
|
class CreateSubcommand(object):
|
||
|
|
||
|
name = None
|
||
|
help = None
|
||
|
usage = None
|
||
|
description = None
|
||
|
epilog = None
|
||
|
formatter_class = None
|
||
|
|
||
|
def __init__(self, logger, subparsers):
|
||
|
self.logger = logger
|
||
|
self.group = subparsers
|
||
|
parser_params = dict(help=(self.help or self.description), usage=self.usage,
|
||
|
description=format_body(textwrap.dedent(self.description), 80),
|
||
|
epilog=self.epilog)
|
||
|
if self.formatter_class:
|
||
|
parser_params['formatter_class'] = self.formatter_class
|
||
|
self.parser = subparsers.add_parser(self.name, **parser_params)
|
||
|
init_argument_parser(self.parser) # propagate top-level options
|
||
|
self.initialize()
|
||
|
|
||
|
def initialize(self):
|
||
|
pass
|
||
|
|
||
|
|
||
|
class CreateWorkloadSubcommand(CreateSubcommand):
|
||
|
|
||
|
name = 'workload'
|
||
|
description = '''Create a new workload. By default, a basic workload template will be
|
||
|
used but you can use options to specify a different template.'''
|
||
|
|
||
|
def initialize(self):
|
||
|
self.parser.add_argument('name', metavar='NAME',
|
||
|
help='Name of the workload to be created')
|
||
|
self.parser.add_argument('-p', '--path', metavar='PATH', default=None,
|
||
|
help='The location at which the workload will be created. If not specified, ' +
|
||
|
'this defaults to "~/.workload_automation/workloads".')
|
||
|
self.parser.add_argument('-f', '--force', action='store_true',
|
||
|
help='Create the new workload even if a workload with the specified ' +
|
||
|
'name already exists.')
|
||
|
|
||
|
template_group = self.parser.add_mutually_exclusive_group()
|
||
|
template_group.add_argument('-A', '--android-benchmark', action='store_true',
|
||
|
help='Use android benchmark template. This template allows you to specify ' +
|
||
|
' an APK file that will be installed and run on the device. You should ' +
|
||
|
' place the APK file into the workload\'s directory at the same level ' +
|
||
|
'as the __init__.py.')
|
||
|
template_group.add_argument('-U', '--ui-automation', action='store_true',
|
||
|
help='Use UI automation template. This template generates a UI automation ' +
|
||
|
'Android project as well as the Python class. This a more general ' +
|
||
|
'version of the android benchmark template that makes no assumptions ' +
|
||
|
'about the nature of your workload, apart from the fact that you need ' +
|
||
|
'UI automation. If you need to install an APK, start an app on device, ' +
|
||
|
'etc., you will need to do that explicitly in your code.')
|
||
|
template_group.add_argument('-B', '--android-uiauto-benchmark', action='store_true',
|
||
|
help='Use android uiauto benchmark template. This generates a UI automation ' +
|
||
|
'project as well as a Python class. This template should be used ' +
|
||
|
'if you have a APK file that needs to be run on the device. You ' +
|
||
|
'should place the APK file into the workload\'s directory at the ' +
|
||
|
'same level as the __init__.py.')
|
||
|
|
||
|
def execute(self, args): # pylint: disable=R0201
|
||
|
where = args.path or 'local'
|
||
|
check_name = not args.force
|
||
|
|
||
|
if args.android_benchmark:
|
||
|
kind = 'android'
|
||
|
elif args.ui_automation:
|
||
|
kind = 'uiauto'
|
||
|
elif args.android_uiauto_benchmark:
|
||
|
kind = 'android_uiauto'
|
||
|
else:
|
||
|
kind = 'basic'
|
||
|
|
||
|
try:
|
||
|
create_workload(args.name, kind, where, check_name)
|
||
|
except CommandError, e:
|
||
|
print "ERROR:", e
|
||
|
|
||
|
|
||
|
class CreatePackageSubcommand(CreateSubcommand):
|
||
|
|
||
|
name = 'package'
|
||
|
description = '''Create a new empty Python package for WA extensions. On installation,
|
||
|
this package will "advertise" itself to WA so that Extensions with in it will
|
||
|
be loaded by WA when it runs.'''
|
||
|
|
||
|
def initialize(self):
|
||
|
self.parser.add_argument('name', metavar='NAME',
|
||
|
help='Name of the package to be created')
|
||
|
self.parser.add_argument('-p', '--path', metavar='PATH', default=None,
|
||
|
help='The location at which the new pacakge will be created. If not specified, ' +
|
||
|
'current working directory will be used.')
|
||
|
self.parser.add_argument('-f', '--force', action='store_true',
|
||
|
help='Create the new package even if a file or directory with the same name '
|
||
|
'already exists at the specified location.')
|
||
|
|
||
|
def execute(self, args): # pylint: disable=R0201
|
||
|
package_dir = args.path or os.path.abspath('.')
|
||
|
template_path = os.path.join(TEMPLATES_DIR, 'setup.template')
|
||
|
self.create_extensions_package(package_dir, args.name, template_path, args.force)
|
||
|
|
||
|
def create_extensions_package(self, location, name, setup_template_path, overwrite=False):
|
||
|
package_path = os.path.join(location, name)
|
||
|
if os.path.exists(package_path):
|
||
|
if overwrite:
|
||
|
self.logger.info('overwriting existing "{}"'.format(package_path))
|
||
|
shutil.rmtree(package_path)
|
||
|
else:
|
||
|
raise CommandError('Location "{}" already exists.'.format(package_path))
|
||
|
actual_package_path = os.path.join(package_path, name)
|
||
|
os.makedirs(actual_package_path)
|
||
|
setup_text = render_template(setup_template_path, {'package_name': name, 'user': getpass.getuser()})
|
||
|
with open(os.path.join(package_path, 'setup.py'), 'w') as wfh:
|
||
|
wfh.write(setup_text)
|
||
|
touch(os.path.join(actual_package_path, '__init__.py'))
|
||
|
|
||
|
|
||
|
class CreateCommand(Command):
|
||
|
|
||
|
name = 'create'
|
||
|
description = '''Used to create various WA-related objects (see positional arguments list for what
|
||
|
objects may be created).\n\nUse "wa create <object> -h" for object-specific arguments.'''
|
||
|
formatter_class = argparse.RawDescriptionHelpFormatter
|
||
|
subcmd_classes = [CreateWorkloadSubcommand, CreatePackageSubcommand]
|
||
|
|
||
|
def initialize(self):
|
||
|
subparsers = self.parser.add_subparsers(dest='what')
|
||
|
self.subcommands = [] # pylint: disable=W0201
|
||
|
for subcmd_cls in self.subcmd_classes:
|
||
|
subcmd = subcmd_cls(self.logger, subparsers)
|
||
|
self.subcommands.append(subcmd)
|
||
|
|
||
|
def execute(self, args):
|
||
|
for subcmd in self.subcommands:
|
||
|
if subcmd.name == args.what:
|
||
|
subcmd.execute(args)
|
||
|
break
|
||
|
else:
|
||
|
raise CommandError('Not a valid create parameter: {}'.format(args.name))
|
||
|
|
||
|
|
||
|
def create_workload(name, kind='basic', where='local', check_name=True, **kwargs):
|
||
|
if check_name:
|
||
|
extloader = ExtensionLoader(packages=settings.extension_packages, paths=settings.extension_paths)
|
||
|
if name in [wl.name for wl in extloader.list_workloads()]:
|
||
|
raise CommandError('Workload with name "{}" already exists.'.format(name))
|
||
|
|
||
|
class_name = get_class_name(name)
|
||
|
if where == 'local':
|
||
|
workload_dir = _d(os.path.join(settings.environment_root, 'workloads', name))
|
||
|
else:
|
||
|
workload_dir = _d(os.path.join(where, name))
|
||
|
|
||
|
if kind == 'basic':
|
||
|
create_basic_workload(workload_dir, name, class_name, **kwargs)
|
||
|
elif kind == 'uiauto':
|
||
|
create_uiautomator_workload(workload_dir, name, class_name, **kwargs)
|
||
|
elif kind == 'android':
|
||
|
create_android_benchmark(workload_dir, name, class_name, **kwargs)
|
||
|
elif kind == 'android_uiauto':
|
||
|
create_android_uiauto_benchmark(workload_dir, name, class_name, **kwargs)
|
||
|
else:
|
||
|
raise CommandError('Unknown workload type: {}'.format(kind))
|
||
|
|
||
|
print 'Workload created in {}'.format(workload_dir)
|
||
|
|
||
|
|
||
|
def create_basic_workload(path, name, class_name):
|
||
|
source_file = os.path.join(path, '__init__.py')
|
||
|
with open(source_file, 'w') as wfh:
|
||
|
wfh.write(render_template('basic_workload', {'name': name, 'class_name': class_name}))
|
||
|
|
||
|
|
||
|
def create_uiautomator_workload(path, name, class_name):
|
||
|
uiauto_path = _d(os.path.join(path, 'uiauto'))
|
||
|
create_uiauto_project(uiauto_path, name)
|
||
|
source_file = os.path.join(path, '__init__.py')
|
||
|
with open(source_file, 'w') as wfh:
|
||
|
wfh.write(render_template('uiauto_workload', {'name': name, 'class_name': class_name}))
|
||
|
|
||
|
|
||
|
def create_android_benchmark(path, name, class_name):
|
||
|
source_file = os.path.join(path, '__init__.py')
|
||
|
with open(source_file, 'w') as wfh:
|
||
|
wfh.write(render_template('android_benchmark', {'name': name, 'class_name': class_name}))
|
||
|
|
||
|
|
||
|
def create_android_uiauto_benchmark(path, name, class_name):
|
||
|
uiauto_path = _d(os.path.join(path, 'uiauto'))
|
||
|
create_uiauto_project(uiauto_path, name)
|
||
|
source_file = os.path.join(path, '__init__.py')
|
||
|
with open(source_file, 'w') as wfh:
|
||
|
wfh.write(render_template('android_uiauto_benchmark', {'name': name, 'class_name': class_name}))
|
||
|
|
||
|
|
||
|
def create_uiauto_project(path, name, target='1'):
|
||
|
sdk_path = get_sdk_path()
|
||
|
android_path = os.path.join(sdk_path, 'tools', 'android')
|
||
|
package_name = 'com.arm.wlauto.uiauto.' + name.lower()
|
||
|
|
||
|
# ${ANDROID_HOME}/tools/android create uitest-project -n com.arm.wlauto.uiauto.linpack -t 1 -p ../test2
|
||
|
command = '{} create uitest-project --name {} --target {} --path {}'.format(android_path,
|
||
|
package_name,
|
||
|
target,
|
||
|
path)
|
||
|
check_output(command, shell=True)
|
||
|
|
||
|
build_script = os.path.join(path, 'build.sh')
|
||
|
with open(build_script, 'w') as wfh:
|
||
|
template = string.Template(UIAUTO_BUILD_SCRIPT)
|
||
|
wfh.write(template.substitute({'package_name': package_name}))
|
||
|
os.chmod(build_script, stat.S_IRWXU | stat.S_IRWXG | stat.S_IRWXO)
|
||
|
|
||
|
source_file = _f(os.path.join(path, 'src',
|
||
|
os.sep.join(package_name.split('.')[:-1]),
|
||
|
'UiAutomation.java'))
|
||
|
with open(source_file, 'w') as wfh:
|
||
|
wfh.write(render_template('UiAutomation.java', {'name': name, 'package_name': package_name}))
|
||
|
|
||
|
|
||
|
# Utility functions
|
||
|
|
||
|
def get_sdk_path():
|
||
|
sdk_path = os.getenv('ANDROID_HOME')
|
||
|
if not sdk_path:
|
||
|
raise CommandError('Please set ANDROID_HOME environment variable to point to ' +
|
||
|
'the locaton of Android SDK')
|
||
|
return sdk_path
|
||
|
|
||
|
|
||
|
def get_class_name(name, postfix=''):
|
||
|
name = identifier(name)
|
||
|
return ''.join(map(capitalize, name.split('_'))) + postfix
|
||
|
|
||
|
|
||
|
def render_template(name, params):
|
||
|
filepath = os.path.join(TEMPLATES_DIR, name)
|
||
|
with open(filepath) as fh:
|
||
|
text = fh.read()
|
||
|
template = string.Template(text)
|
||
|
return template.substitute(params)
|
||
|
|
||
|
|
||
|
def touch(path):
|
||
|
with open(path, 'w') as wfh: # pylint: disable=unused-variable
|
||
|
pass
|