mirror of
https://github.com/ARM-software/workload-automation.git
synced 2025-02-20 20:09:11 +00:00
Merge pull request #434 from marcbonnici/next_uiauto2
Next: Fixes and uiauto2 workloads
This commit is contained in:
commit
61a44dd91d
@ -46,12 +46,42 @@ class SubClass(TestClass):
|
||||
def __init__(self):
|
||||
super(SubClass, self).__init__()
|
||||
|
||||
@once
|
||||
def initilize_once(self):
|
||||
super(SubClass, self).initilize_once()
|
||||
self.count += 1
|
||||
|
||||
@once_per_class
|
||||
def initilize_once_per_class(self):
|
||||
super(SubClass, self).initilize_once_per_class()
|
||||
self.count += 1
|
||||
|
||||
@once_per_instance
|
||||
def initilize_once_per_instance(self):
|
||||
super(SubClass, self).initilize_once_per_instance()
|
||||
self.count += 1
|
||||
|
||||
|
||||
class SubSubClass(SubClass):
|
||||
|
||||
def __init__(self):
|
||||
super(SubSubClass, self).__init__()
|
||||
|
||||
@once
|
||||
def initilize_once(self):
|
||||
super(SubSubClass, self).initilize_once()
|
||||
self.count += 1
|
||||
|
||||
@once_per_class
|
||||
def initilize_once_per_class(self):
|
||||
super(SubSubClass, self).initilize_once_per_class()
|
||||
self.count += 1
|
||||
|
||||
@once_per_instance
|
||||
def initilize_once_per_instance(self):
|
||||
super(SubSubClass, self).initilize_once_per_instance()
|
||||
self.count += 1
|
||||
|
||||
|
||||
class AnotherClass(object):
|
||||
|
||||
@ -262,8 +292,8 @@ class OncePerInstanceEnvironmentTest(TestCase):
|
||||
|
||||
sc.initilize_once_per_instance()
|
||||
sc.initilize_once_per_instance()
|
||||
assert_equal(sc.count, 1)
|
||||
assert_equal(sc.count, 2)
|
||||
|
||||
ss.initilize_once_per_instance()
|
||||
ss.initilize_once_per_instance()
|
||||
assert_equal(ss.count, 1)
|
||||
assert_equal(ss.count, 3)
|
||||
|
@ -241,7 +241,7 @@ def update_config_from_source(final_config, source, state):
|
||||
cfg_point.set_value(final_config, value, check_mandatory=False)
|
||||
|
||||
if state.generic_config[source]:
|
||||
msg = 'Unexected values for {}: {}'
|
||||
msg = 'Unexpected values for {}: {}'
|
||||
raise ConfigError(msg.format(state.generic_name,
|
||||
state.generic_config[source]))
|
||||
|
||||
@ -254,6 +254,6 @@ def update_config_from_source(final_config, source, state):
|
||||
cfg_point.set_value(final_config, value, check_mandatory=False)
|
||||
|
||||
if state.specific_config[source]:
|
||||
msg = 'Unexected values for {}: {}'
|
||||
msg = 'Unexpected values for {}: {}'
|
||||
raise ConfigError(msg.format(state.specific_name,
|
||||
state.specific_config[source]))
|
||||
|
@ -6,6 +6,8 @@ from wa.framework.configuration.core import Status
|
||||
|
||||
class Job(object):
|
||||
|
||||
_workload_cache = {}
|
||||
|
||||
@property
|
||||
def id(self):
|
||||
return self.spec.id
|
||||
@ -40,11 +42,15 @@ class Job(object):
|
||||
|
||||
def load(self, target, loader=pluginloader):
|
||||
self.logger.info('Loading job {}'.format(self.id))
|
||||
self.workload = loader.get_workload(self.spec.workload_name,
|
||||
target,
|
||||
**self.spec.workload_parameters)
|
||||
self.workload.init_resources(self.context)
|
||||
self.workload.validate()
|
||||
if self.iteration == 1:
|
||||
self.workload = loader.get_workload(self.spec.workload_name,
|
||||
target,
|
||||
**self.spec.workload_parameters)
|
||||
self.workload.init_resources(self.context)
|
||||
self.workload.validate()
|
||||
self._workload_cache[self.id] = self.workload
|
||||
else:
|
||||
self.workload = self._workload_cache[self.id]
|
||||
|
||||
def initialize(self, context):
|
||||
self.logger.info('Initializing job {}'.format(self.id))
|
||||
|
@ -1,5 +1,5 @@
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
package="com.arm.wlauto.uiauto">
|
||||
package="com.arm.wa.uiauto">
|
||||
|
||||
<uses-permission android:name="android.permission.READ_LOGS"/>
|
||||
|
||||
|
@ -38,6 +38,14 @@ import java.util.concurrent.TimeoutException;
|
||||
|
||||
public class BaseUiAutomation {
|
||||
|
||||
public enum FindByCriteria {BY_ID, BY_TEXT, BY_DESC}
|
||||
|
||||
// Time in milliseconds
|
||||
public long uiAutoTimeout = 4 * 1000;
|
||||
|
||||
public static final int CLICK_REPEAT_INTERVAL_MINIMUM = 5;
|
||||
public static final int CLICK_REPEAT_INTERVAL_DEFAULT = 50;
|
||||
|
||||
public Instrumentation mInstrumentation;
|
||||
public Context mContext;
|
||||
public UiDevice mDevice;
|
||||
@ -140,4 +148,218 @@ public class BaseUiAutomation {
|
||||
throw new TimeoutException("Timed out waiting for Logcat text \"%s\"".format(searchText));
|
||||
}
|
||||
}
|
||||
|
||||
public void repeatClickUiObject(UiObject view, int repeatCount, int intervalInMillis) throws Exception {
|
||||
int repeatInterval = intervalInMillis > CLICK_REPEAT_INTERVAL_MINIMUM
|
||||
? intervalInMillis : CLICK_REPEAT_INTERVAL_DEFAULT;
|
||||
if (repeatCount < 1 || !view.isClickable()) {
|
||||
return;
|
||||
}
|
||||
|
||||
for (int i = 0; i < repeatCount; ++i) {
|
||||
view.click();
|
||||
SystemClock.sleep(repeatInterval); // in order to register as separate click
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public UiObject clickUiObject(FindByCriteria criteria, String matching) throws Exception {
|
||||
return clickUiObject(criteria, matching, null, false);
|
||||
}
|
||||
|
||||
public UiObject clickUiObject(FindByCriteria criteria, String matching, boolean wait) throws Exception {
|
||||
return clickUiObject(criteria, matching, null, wait);
|
||||
}
|
||||
|
||||
public UiObject clickUiObject(FindByCriteria criteria, String matching, String clazz) throws Exception {
|
||||
return clickUiObject(criteria, matching, clazz, false);
|
||||
}
|
||||
|
||||
public UiObject clickUiObject(FindByCriteria criteria, String matching, String clazz, boolean wait) throws Exception {
|
||||
UiObject view;
|
||||
|
||||
switch (criteria) {
|
||||
case BY_ID:
|
||||
view = (clazz == null)
|
||||
? getUiObjectByResourceId(matching) : getUiObjectByResourceId(matching, clazz);
|
||||
break;
|
||||
case BY_DESC:
|
||||
view = (clazz == null)
|
||||
? getUiObjectByDescription(matching) : getUiObjectByDescription(matching, clazz);
|
||||
break;
|
||||
case BY_TEXT:
|
||||
default:
|
||||
view = (clazz == null)
|
||||
? getUiObjectByText(matching) : getUiObjectByText(matching, clazz);
|
||||
break;
|
||||
}
|
||||
|
||||
if (wait) {
|
||||
view.clickAndWaitForNewWindow();
|
||||
} else {
|
||||
view.click();
|
||||
}
|
||||
return view;
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByResourceId(String resourceId, String className) throws Exception {
|
||||
return getUiObjectByResourceId(resourceId, className, uiAutoTimeout);
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByResourceId(String resourceId, String className, long timeout) throws Exception {
|
||||
UiObject object = mDevice.findObject(new UiSelector().resourceId(resourceId)
|
||||
.className(className));
|
||||
if (!object.waitForExists(timeout)) {
|
||||
throw new UiObjectNotFoundException(String.format("Could not find \"%s\" \"%s\"",
|
||||
resourceId, className));
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByResourceId(String id) throws Exception {
|
||||
UiObject object = mDevice.findObject(new UiSelector().resourceId(id));
|
||||
|
||||
if (!object.waitForExists(uiAutoTimeout)) {
|
||||
throw new UiObjectNotFoundException("Could not find view with resource ID: " + id);
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByDescription(String description, String className) throws Exception {
|
||||
return getUiObjectByDescription(description, className, uiAutoTimeout);
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByDescription(String description, String className, long timeout) throws Exception {
|
||||
UiObject object = mDevice.findObject(new UiSelector().descriptionContains(description)
|
||||
.className(className));
|
||||
if (!object.waitForExists(timeout)) {
|
||||
throw new UiObjectNotFoundException(String.format("Could not find \"%s\" \"%s\"",
|
||||
description, className));
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByDescription(String desc) throws Exception {
|
||||
UiObject object = mDevice.findObject(new UiSelector().descriptionContains(desc));
|
||||
|
||||
if (!object.waitForExists(uiAutoTimeout)) {
|
||||
throw new UiObjectNotFoundException("Could not find view with description: " + desc);
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByText(String text, String className) throws Exception {
|
||||
return getUiObjectByText(text, className, uiAutoTimeout);
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByText(String text, String className, long timeout) throws Exception {
|
||||
UiObject object = mDevice.findObject(new UiSelector().textContains(text)
|
||||
.className(className));
|
||||
if (!object.waitForExists(timeout)) {
|
||||
throw new UiObjectNotFoundException(String.format("Could not find \"%s\" \"%s\"",
|
||||
text, className));
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
public UiObject getUiObjectByText(String text) throws Exception {
|
||||
UiObject object = mDevice.findObject(new UiSelector().textContains(text));
|
||||
|
||||
if (!object.waitForExists(uiAutoTimeout)) {
|
||||
throw new UiObjectNotFoundException("Could not find view with text: " + text);
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
// Override getParams function to decode a url encoded parameter bundle before
|
||||
// passing it to workloads.
|
||||
public Bundle getParams() {
|
||||
// Get the original parameter bundle
|
||||
Bundle parameters = getArguments();
|
||||
|
||||
// Decode each parameter in the bundle, except null values and "class", as this
|
||||
// used to control instrumentation and therefore not encoded.
|
||||
for (String key : parameters.keySet()) {
|
||||
String param = parameters.getString(key);
|
||||
if (param != null && !key.equals("class")) {
|
||||
param = android.net.Uri.decode(param);
|
||||
parameters = decode(parameters, key, param);
|
||||
}
|
||||
}
|
||||
return parameters;
|
||||
}
|
||||
|
||||
// Helper function to decode a string and insert it as an appropriate type
|
||||
// into a provided bundle with its key.
|
||||
// Each bundle parameter will be a urlencoded string with 2 characters prefixed to the value
|
||||
// used to store the original type information, e.g. 'fl' -> list of floats.
|
||||
private Bundle decode(Bundle parameters, String key, String value) {
|
||||
char value_type = value.charAt(0);
|
||||
char value_dimension = value.charAt(1);
|
||||
String param = value.substring(2);
|
||||
|
||||
if (value_dimension == 's') {
|
||||
if (value_type == 's') {
|
||||
parameters.putString(key, param);
|
||||
} else if (value_type == 'f') {
|
||||
parameters.putFloat(key, Float.parseFloat(param));
|
||||
} else if (value_type == 'd') {
|
||||
parameters.putDouble(key, Double.parseDouble(param));
|
||||
} else if (value_type == 'b') {
|
||||
parameters.putBoolean(key, Boolean.parseBoolean(param));
|
||||
} else if (value_type == 'i') {
|
||||
parameters.putInt(key, Integer.parseInt(param));
|
||||
} else if (value_type == 'n') {
|
||||
parameters.putString(key, "None");
|
||||
} else {
|
||||
throw new IllegalArgumentException("Error decoding:" + key + value
|
||||
+ " - unknown format");
|
||||
}
|
||||
} else if (value_dimension == 'l') {
|
||||
return decodeArray(parameters, key, value_type, param);
|
||||
} else {
|
||||
throw new IllegalArgumentException("Error decoding:" + key + value
|
||||
+ " - unknown format");
|
||||
}
|
||||
return parameters;
|
||||
}
|
||||
|
||||
// Helper function to deal with decoding arrays and update the bundle with
|
||||
// an appropriate array type. The string "0newelement0" is used to distinguish
|
||||
// each element from each other in the array when encoded.
|
||||
private Bundle decodeArray(Bundle parameters, String key, char type, String value) {
|
||||
String[] string_list = value.split("0newelement0");
|
||||
if (type == 's') {
|
||||
parameters.putStringArray(key, string_list);
|
||||
}
|
||||
else if (type == 'i') {
|
||||
int[] int_list = new int[string_list.length];
|
||||
for (int i = 0; i < string_list.length; i++){
|
||||
int_list[i] = Integer.parseInt(string_list[i]);
|
||||
}
|
||||
parameters.putIntArray(key, int_list);
|
||||
} else if (type == 'f') {
|
||||
float[] float_list = new float[string_list.length];
|
||||
for (int i = 0; i < string_list.length; i++){
|
||||
float_list[i] = Float.parseFloat(string_list[i]);
|
||||
}
|
||||
parameters.putFloatArray(key, float_list);
|
||||
} else if (type == 'd') {
|
||||
double[] double_list = new double[string_list.length];
|
||||
for (int i = 0; i < string_list.length; i++){
|
||||
double_list[i] = Double.parseDouble(string_list[i]);
|
||||
}
|
||||
parameters.putDoubleArray(key, double_list);
|
||||
} else if (type == 'b') {
|
||||
boolean[] boolean_list = new boolean[string_list.length];
|
||||
for (int i = 0; i < string_list.length; i++){
|
||||
boolean_list[i] = Boolean.parseBoolean(string_list[i]);
|
||||
}
|
||||
parameters.putBooleanArray(key, boolean_list);
|
||||
} else {
|
||||
throw new IllegalArgumentException("Error decoding array: " +
|
||||
value + " - unknown format");
|
||||
}
|
||||
return parameters;
|
||||
}
|
||||
}
|
||||
|
Binary file not shown.
@ -22,8 +22,9 @@ from wa.framework.plugin import TargetedPlugin
|
||||
from wa.framework.resource import (ApkFile, JarFile, ReventFile, NO_ONE,
|
||||
Executable, File)
|
||||
from wa.framework.exception import WorkloadError
|
||||
from wa.utils.types import ParameterDict
|
||||
from wa.utils.revent import ReventRecorder
|
||||
from wa.utils.exec_control import once
|
||||
from wa.utils.exec_control import once_per_instance
|
||||
|
||||
from devlib.utils.android import ApkInfo
|
||||
from devlib.exception import TargetError
|
||||
@ -160,6 +161,7 @@ class ApkWorkload(Workload):
|
||||
def init_resources(self, context):
|
||||
pass
|
||||
|
||||
@once_per_instance
|
||||
def initialize(self, context):
|
||||
self.apk.initialize(context)
|
||||
|
||||
@ -176,7 +178,7 @@ class ApkWorkload(Workload):
|
||||
def teardown(self, context):
|
||||
self.apk.teardown()
|
||||
|
||||
@once
|
||||
@once_per_instance
|
||||
def finalize(self, context):
|
||||
pass
|
||||
|
||||
@ -190,14 +192,14 @@ class ApkUIWorkload(ApkWorkload):
|
||||
def init_resources(self, context):
|
||||
super(ApkUIWorkload, self).init_resources(context)
|
||||
self.gui.init_resources(context.resolver)
|
||||
self.gui.init_commands()
|
||||
|
||||
@once_per_instance
|
||||
def initialize(self, context):
|
||||
super(ApkUIWorkload, self).initialize(context)
|
||||
self.gui.deploy()
|
||||
|
||||
def setup(self, context):
|
||||
super(ApkUIWorkload, self).setup(context)
|
||||
self.gui.deploy()
|
||||
self.gui.setup()
|
||||
|
||||
def run(self, context):
|
||||
@ -212,7 +214,7 @@ class ApkUIWorkload(ApkWorkload):
|
||||
self.gui.teardown()
|
||||
super(ApkUIWorkload, self).teardown(context)
|
||||
|
||||
@once
|
||||
@once_per_instance
|
||||
def finalize(self, context):
|
||||
super(ApkUIWorkload, self).finalize(context)
|
||||
self.gui.remove()
|
||||
@ -226,6 +228,11 @@ class ApkUiautoWorkload(ApkUIWorkload):
|
||||
super(ApkUiautoWorkload, self).__init__(target, **kwargs)
|
||||
self.gui = UiAutomatorGUI(self)
|
||||
|
||||
def setup(self, context):
|
||||
self.gui.uiauto_params['package_name'] = self.apk.apk_info.package
|
||||
self.gui.init_commands()
|
||||
super(ApkUiautoWorkload, self).setup(context)
|
||||
|
||||
|
||||
class ReventWorkload(ApkUIWorkload):
|
||||
|
||||
@ -261,7 +268,7 @@ class UiAutomatorGUI(object):
|
||||
self.logger = logging.getLogger('gui')
|
||||
self.uiauto_file = None
|
||||
self.commands = {}
|
||||
self.uiauto_params = {}
|
||||
self.uiauto_params = ParameterDict()
|
||||
|
||||
def init_resources(self, resolver):
|
||||
self.uiauto_file = resolver.get(ApkFile(self.owner, uiauto=True))
|
||||
@ -273,7 +280,7 @@ class UiAutomatorGUI(object):
|
||||
params_dict = self.uiauto_params
|
||||
params_dict['workdir'] = self.target.working_directory
|
||||
params = ''
|
||||
for k, v in self.uiauto_params.iteritems():
|
||||
for k, v in params_dict.iter_encoded_items():
|
||||
params += ' -e {} {}'.format(k, v)
|
||||
|
||||
for stage in self.stages:
|
||||
@ -286,7 +293,9 @@ class UiAutomatorGUI(object):
|
||||
instrumentation_string)
|
||||
|
||||
def deploy(self):
|
||||
self.target.install_apk(self.uiauto_file, replace=True)
|
||||
if self.target.package_is_installed(self.uiauto_package):
|
||||
self.target.uninstall_package(self.uiauto_package)
|
||||
self.target.install_apk(self.uiauto_file)
|
||||
|
||||
def set(self, name, value):
|
||||
self.uiauto_params[name] = value
|
||||
@ -395,9 +404,6 @@ class ReventGUI(object):
|
||||
def remove(self):
|
||||
self.revent_recorder.remove()
|
||||
|
||||
def init_commands(self):
|
||||
pass
|
||||
|
||||
def _check_revent_files(self):
|
||||
if not self.revent_run_file:
|
||||
# pylint: disable=too-few-format-args
|
||||
@ -435,9 +441,9 @@ class PackageHandler(object):
|
||||
|
||||
def initialize(self, context):
|
||||
self.resolve_package(context)
|
||||
self.initialize_package(context)
|
||||
|
||||
def setup(self, context):
|
||||
self.initialize_package(context)
|
||||
self.start_activity()
|
||||
self.target.execute('am kill-all') # kill all *background* activities
|
||||
self.target.clear_logcat()
|
||||
@ -514,9 +520,12 @@ class PackageHandler(object):
|
||||
self.apk_version = host_version
|
||||
|
||||
def start_activity(self):
|
||||
cmd = 'am start -W -n {}/{}'
|
||||
output = self.target.execute(cmd.format(self.apk_info.package,
|
||||
self.apk_info.activity))
|
||||
if not self.apk_info.activity:
|
||||
cmd = 'am start -W {}'.format(self.apk_info.package)
|
||||
else:
|
||||
cmd = 'am start -W -n {}/{}'.format(self.apk_info.package,
|
||||
self.apk_info.activity)
|
||||
output = self.target.execute(cmd)
|
||||
if 'Error:' in output:
|
||||
# this will dismiss any error dialogs
|
||||
self.target.execute('am force-stop {}'.format(self.apk_info.package))
|
||||
@ -528,7 +537,8 @@ class PackageHandler(object):
|
||||
self.target.execute('pm clear {}'.format(self.apk_info.package))
|
||||
|
||||
def install_apk(self, context):
|
||||
output = self.target.install_apk(self.apk_file, self.install_timeout)
|
||||
output = self.target.install_apk(self.apk_file, self.install_timeout,
|
||||
replace=True, allow_downgrade=True)
|
||||
if 'Failure' in output:
|
||||
if 'ALREADY_EXISTS' in output:
|
||||
msg = 'Using already installed APK (did not unistall properly?)'
|
||||
|
@ -59,7 +59,7 @@ def once_per_instance(method):
|
||||
def wrapper(*args, **kwargs):
|
||||
if __active_environment is None:
|
||||
activate_environment('default')
|
||||
func_id = repr(args[0])
|
||||
func_id = repr(method.__hash__()) + repr(args[0])
|
||||
if func_id in __environments[__active_environment]:
|
||||
return
|
||||
else:
|
||||
|
@ -32,6 +32,7 @@ import numbers
|
||||
import shlex
|
||||
import string
|
||||
from bisect import insort
|
||||
from urllib import quote, unquote
|
||||
from collections import defaultdict, MutableMapping
|
||||
from copy import copy
|
||||
|
||||
@ -327,7 +328,7 @@ class prioritylist(object):
|
||||
raise ValueError('Invalid index {}'.format(index))
|
||||
current_global_offset = 0
|
||||
priority_counts = {priority: count for (priority, count) in
|
||||
zip(self.priorities, [len(self.elements[p])
|
||||
zip(self.priorities, [len(self.elements[p])
|
||||
for p in self.priorities])}
|
||||
for priority in self.priorities:
|
||||
if not index_range:
|
||||
@ -586,3 +587,118 @@ def enum(args, start=0, step=1):
|
||||
|
||||
return Enum
|
||||
|
||||
|
||||
class ParameterDict(dict):
|
||||
"""
|
||||
A dict-like object that automatically encodes various types into a url safe string,
|
||||
and enforces a single type for the contents in a list.
|
||||
Each value is first prefixed with 2 letters to preserve type when encoding to a string.
|
||||
The format used is "value_type, value_dimension" e.g a 'list of floats' would become 'fl'.
|
||||
"""
|
||||
|
||||
# Function to determine the appropriate prefix based on the parameters type
|
||||
@staticmethod
|
||||
def _get_prefix(obj):
|
||||
if isinstance(obj, basestring):
|
||||
prefix = 's'
|
||||
elif isinstance(obj, float):
|
||||
prefix = 'f'
|
||||
elif isinstance(obj, long):
|
||||
prefix = 'd'
|
||||
elif isinstance(obj, bool):
|
||||
prefix = 'b'
|
||||
elif isinstance(obj, int):
|
||||
prefix = 'i'
|
||||
elif obj is None:
|
||||
prefix = 'n'
|
||||
else:
|
||||
raise ValueError('Unable to encode {} {}'.format(obj, type(obj)))
|
||||
return prefix
|
||||
|
||||
# Function to add prefix and urlencode a provided parameter.
|
||||
@staticmethod
|
||||
def _encode(obj):
|
||||
if isinstance(obj, list):
|
||||
t = type(obj[0])
|
||||
prefix = ParameterDict._get_prefix(obj[0]) + 'l'
|
||||
for item in obj:
|
||||
if not isinstance(item, t):
|
||||
msg = 'Lists must only contain a single type, contains {} and {}'
|
||||
raise ValueError(msg.format(t, type(item)))
|
||||
obj = '0newelement0'.join(str(x) for x in obj)
|
||||
else:
|
||||
prefix = ParameterDict._get_prefix(obj) + 's'
|
||||
return quote(prefix + str(obj))
|
||||
|
||||
# Function to decode a string and return a value of the original parameter type.
|
||||
# pylint: disable=too-many-return-statements
|
||||
@staticmethod
|
||||
def _decode(string):
|
||||
value_type = string[:1]
|
||||
value_dimension = string[1:2]
|
||||
value = unquote(string[2:])
|
||||
if value_dimension == 's':
|
||||
if value_type == 's':
|
||||
return str(value)
|
||||
elif value_type == 'b':
|
||||
return boolean(value)
|
||||
elif value_type == 'd':
|
||||
return long(value)
|
||||
elif value_type == 'f':
|
||||
return float(value)
|
||||
elif value_type == 'i':
|
||||
return int(value)
|
||||
elif value_type == 'n':
|
||||
return None
|
||||
elif value_dimension == 'l':
|
||||
return [ParameterDict._decode(value_type + 's' + x)
|
||||
for x in value.split('0newelement0')]
|
||||
else:
|
||||
raise ValueError('Unknown {} {}'.format(type(string), string))
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
for k, v in kwargs.iteritems():
|
||||
self.__setitem__(k, v)
|
||||
dict.__init__(self, *args)
|
||||
|
||||
def __setitem__(self, name, value):
|
||||
dict.__setitem__(self, name, self._encode(value))
|
||||
|
||||
def __getitem__(self, name):
|
||||
return self._decode(dict.__getitem__(self, name))
|
||||
|
||||
def __contains__(self, item):
|
||||
return dict.__contains__(self, self._encode(item))
|
||||
|
||||
def __iter__(self):
|
||||
return iter((k, self._decode(v)) for (k, v) in self.items())
|
||||
|
||||
def iteritems(self):
|
||||
return self.__iter__()
|
||||
|
||||
def get(self, name):
|
||||
return self._decode(dict.get(self, name))
|
||||
|
||||
def pop(self, key):
|
||||
return self._decode(dict.pop(self, key))
|
||||
|
||||
def popitem(self):
|
||||
key, value = dict.popitem(self)
|
||||
return (key, self._decode(value))
|
||||
|
||||
def iter_encoded_items(self):
|
||||
return dict.iteritems(self)
|
||||
|
||||
def get_encoded_value(self, name):
|
||||
return dict.__getitem__(self, name)
|
||||
|
||||
def values(self):
|
||||
return [self[k] for k in dict.keys(self)]
|
||||
|
||||
def update(self, *args, **kwargs):
|
||||
for d in list(args) + [kwargs]:
|
||||
if isinstance(d, ParameterDict):
|
||||
dict.update(self, d)
|
||||
else:
|
||||
for k, v in d.iteritems():
|
||||
self[k] = v
|
||||
|
Binary file not shown.
@ -1,6 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
package="com.arm.wlauto.uiauto.benchmarkpi"
|
||||
package="com.arm.wa.uiauto.benchmarkpi"
|
||||
android:versionCode="1"
|
||||
android:versionName="1.0">
|
||||
|
||||
|
@ -33,6 +33,9 @@ public class UiAutomation extends BaseUiAutomation {
|
||||
|
||||
public static String TAG = "benchmarkpi";
|
||||
|
||||
public Bundle parameters;
|
||||
public String packageID;
|
||||
|
||||
@Test
|
||||
public void runWorkload() throws Exception {
|
||||
startTest();
|
||||
@ -43,7 +46,7 @@ public class UiAutomation extends BaseUiAutomation {
|
||||
public void extractResults() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject resultsText = mDevice.findObject(selector.textContains("You calculated Pi in")
|
||||
.className("android.widget.TextView"));
|
||||
.className("android.widget.TextView"));
|
||||
Log.v(TAG, resultsText.getText());
|
||||
}
|
||||
|
||||
|
@ -29,7 +29,7 @@ if [[ ! -f gradlew ]]; then
|
||||
exit 9
|
||||
fi
|
||||
|
||||
# Copy base class library from wlauto dist
|
||||
# Copy base class library from wa dist
|
||||
libs_dir=app/libs
|
||||
base_class=`python -c "import os, wa; print os.path.join(os.path.dirname(wa.__file__), 'framework', 'uiauto', 'uiauto.aar')"`
|
||||
mkdir -p $libs_dir
|
||||
|
236
wa/workloads/vellamo/__init__.py
Normal file
236
wa/workloads/vellamo/__init__.py
Normal file
@ -0,0 +1,236 @@
|
||||
# Copyright 2014-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 logging
|
||||
import json
|
||||
import re
|
||||
|
||||
from HTMLParser import HTMLParser
|
||||
from collections import defaultdict, OrderedDict
|
||||
from distutils.version import StrictVersion
|
||||
|
||||
from wa import ApkUiautoWorkload, Parameter
|
||||
from wa.utils.types import list_of_strs, numeric
|
||||
from wa.framework.exception import WorkloadError
|
||||
|
||||
|
||||
class Vellamo(ApkUiautoWorkload):
|
||||
|
||||
name = 'vellamo'
|
||||
description = """
|
||||
Android benchmark designed by Qualcomm.
|
||||
|
||||
Vellamo began as a mobile web benchmarking tool that today has expanded
|
||||
to include three primary chapters. The Browser Chapter evaluates mobile
|
||||
web browser performance, the Multicore chapter measures the synergy of
|
||||
multiple CPU cores, and the Metal Chapter measures the CPU subsystem
|
||||
performance of mobile processors. Through click-and-go test suites,
|
||||
organized by chapter, Vellamo is designed to evaluate: UX, 3D graphics,
|
||||
and memory read/write and peak bandwidth performance, and much more!
|
||||
|
||||
Note: Vellamo v3.0 fails to run on Juno
|
||||
|
||||
"""
|
||||
package_names = ['com.quicinc.vellamo']
|
||||
run_timeout = 15 * 60
|
||||
benchmark_types = {
|
||||
'2.0.3': ['html5', 'metal'],
|
||||
'3.0': ['Browser', 'Metal', 'Multi'],
|
||||
'3.2.4': ['Browser', 'Metal', 'Multi'],
|
||||
}
|
||||
valid_versions = benchmark_types.keys()
|
||||
summary_metrics = None
|
||||
|
||||
parameters = [
|
||||
Parameter('version', kind=str, allowed_values=valid_versions, default=sorted(benchmark_types, reverse=True)[0], override=True,
|
||||
description=('Specify the version of Vellamo to be run. '
|
||||
'If not specified, the latest available version will be used.')),
|
||||
Parameter('benchmarks', kind=list_of_strs, allowed_values=benchmark_types['3.0'], default=benchmark_types['3.0'],
|
||||
description=('Specify which benchmark sections of Vellamo to be run. Only valid on version 3.0 and newer.'
|
||||
'\nNOTE: Browser benchmark can be problematic and seem to hang,'
|
||||
'just wait and it will progress after ~5 minutes')),
|
||||
Parameter('browser', kind=int, default=1,
|
||||
description=('Specify which of the installed browsers will be used for the tests. The number refers to '
|
||||
'the order in which browsers are listed by Vellamo. E.g. ``1`` will select the first browser '
|
||||
'listed, ``2`` -- the second, etc. Only valid for version ``3.0``.'))
|
||||
]
|
||||
|
||||
|
||||
def setup(self, context):
|
||||
self.gui.uiauto_params['version'] = self.version
|
||||
self.gui.uiauto_params['browserToUse'] = self.browser
|
||||
self.gui.uiauto_params['metal'] = 'Metal' in self.benchmarks
|
||||
self.gui.uiauto_params['browser'] = 'Browser' in self.benchmarks
|
||||
self.gui.uiauto_params['multicore'] = 'Multi' in self.benchmarks
|
||||
super(Vellamo, self).setup(context)
|
||||
|
||||
def validate(self):
|
||||
super(Vellamo, self).validate()
|
||||
if self.version == '2.0.3' or not self.benchmarks: # pylint: disable=access-member-before-definition
|
||||
self.benchmarks = self.benchmark_types[self.version] # pylint: disable=attribute-defined-outside-init
|
||||
else:
|
||||
for benchmark in self.benchmarks:
|
||||
if benchmark not in self.benchmark_types[self.version]:
|
||||
raise WorkloadError('Version {} does not support {} benchmarks'.format(self.version, benchmark))
|
||||
|
||||
def update_output(self, context):
|
||||
super(Vellamo, self).update_output(context)
|
||||
|
||||
# Get total scores from logcat
|
||||
self.non_root_update_result(context)
|
||||
|
||||
if not self.target.is_rooted:
|
||||
return
|
||||
elif self.version == '3.0.0':
|
||||
self.update_output_v3(context)
|
||||
elif self.version == '3.2.4':
|
||||
self.update_output_v3_2(context)
|
||||
|
||||
def update_output_v3(self, context):
|
||||
for test in self.benchmarks: # Get all scores from HTML files
|
||||
filename = None
|
||||
if test == "Browser":
|
||||
result_folder = self.target.path.join(self.target.package_data_directory,
|
||||
self.apk.apk_info.package, 'files')
|
||||
for result_file in self.target.listdir(result_folder, as_root=True):
|
||||
if result_file.startswith("Browser"):
|
||||
filename = result_file
|
||||
else:
|
||||
filename = '{}_results.html'.format(test)
|
||||
|
||||
device_file = self.target.path.join(self.target.package_data_directory,
|
||||
self.apk.apk_info.package, 'files', filename)
|
||||
host_file = os.path.join(context.output_directory, filename)
|
||||
self.target.pull(device_file, host_file, as_root=True)
|
||||
with open(host_file) as fh:
|
||||
parser = VellamoResultParser()
|
||||
parser.feed(fh.read())
|
||||
for benchmark in parser.benchmarks:
|
||||
benchmark.name = benchmark.name.replace(' ', '_')
|
||||
context.add_metric('{}_Total'.format(benchmark.name),
|
||||
benchmark.score)
|
||||
for name, score in benchmark.metrics.items():
|
||||
name = name.replace(' ', '_')
|
||||
context.add_metric('{}_{}'.format(benchmark.name,
|
||||
name), score)
|
||||
context.add_artifact('vellamo_output', kind='raw',
|
||||
path=filename)
|
||||
|
||||
def update_output_v3_2(self, context):
|
||||
device_file = self.target.path.join(self.target.package_data_directory,
|
||||
self.apk.apk_info.package,
|
||||
'files',
|
||||
'chapterscores.json')
|
||||
host_file = os.path.join(context.output_directory, 'vellamo.json')
|
||||
self.target.pull(device_file, host_file, as_root=True)
|
||||
context.add_artifact('vellamo_output', kind='raw', path=host_file)
|
||||
# context.add_iteration_artifact('vellamo_output', kind='raw', path=host_file)
|
||||
with open(host_file) as results_file:
|
||||
data = json.load(results_file)
|
||||
for chapter in data:
|
||||
for result in chapter['benchmark_results']:
|
||||
name = result['id']
|
||||
score = result['score']
|
||||
context.add_metric(name, score)
|
||||
|
||||
def non_root_update_result(self, context):
|
||||
failed = []
|
||||
logcat_file = context.get_artifact_path('logcat')
|
||||
with open(logcat_file) as fh:
|
||||
iteration_result_regex = re.compile("VELLAMO RESULT: (Browser|Metal|Multicore) (\d+)")
|
||||
for line in fh:
|
||||
if 'VELLAMO ERROR:' in line:
|
||||
msg = "Browser crashed during benchmark, results may not be accurate"
|
||||
self.logger.warning(msg)
|
||||
result = iteration_result_regex.findall(line)
|
||||
if result:
|
||||
for (metric, score) in result:
|
||||
if not score:
|
||||
failed.append(metric)
|
||||
else:
|
||||
context.add_metric(metric, score)
|
||||
if failed:
|
||||
raise WorkloadError("The following benchmark groups failed: {}".format(", ".join(failed)))
|
||||
|
||||
|
||||
class VellamoResult(object):
|
||||
|
||||
def __init__(self, name):
|
||||
self.name = name
|
||||
self.score = None
|
||||
self.metrics = {}
|
||||
|
||||
def add_metric(self, data):
|
||||
split_data = data.split(":")
|
||||
name = split_data[0].strip()
|
||||
score = split_data[1].strip()
|
||||
|
||||
if name in self.metrics:
|
||||
raise KeyError("A metric of that name is already present")
|
||||
self.metrics[name] = float(score)
|
||||
|
||||
|
||||
class VellamoResultParser(HTMLParser):
|
||||
|
||||
class StopParsingException(Exception):
|
||||
pass
|
||||
|
||||
def __init__(self):
|
||||
HTMLParser.__init__(self)
|
||||
self.inside_div = False
|
||||
self.inside_span = 0
|
||||
self.inside_li = False
|
||||
self.got_data = False
|
||||
self.failed = False
|
||||
self.benchmarks = []
|
||||
|
||||
def feed(self, text):
|
||||
try:
|
||||
HTMLParser.feed(self, text)
|
||||
except self.StopParsingException:
|
||||
pass
|
||||
|
||||
def handle_starttag(self, tag, attrs):
|
||||
if tag == 'div':
|
||||
self.inside_div = True
|
||||
if tag == 'span':
|
||||
self.inside_span += 1
|
||||
if tag == 'li':
|
||||
self.inside_li = True
|
||||
|
||||
def handle_endtag(self, tag):
|
||||
if tag == 'div':
|
||||
self.inside_div = False
|
||||
self.inside_span = 0
|
||||
self.got_data = False
|
||||
self.failed = False
|
||||
if tag == 'li':
|
||||
self.inside_li = False
|
||||
|
||||
def handle_data(self, data):
|
||||
if self.inside_div and not self.failed:
|
||||
if "Problem" in data:
|
||||
self.failed = True
|
||||
elif self.inside_span == 1:
|
||||
self.benchmarks.append(VellamoResult(data))
|
||||
elif self.inside_span == 3 and not self.got_data:
|
||||
self.benchmarks[-1].score = int(data)
|
||||
self.got_data = True
|
||||
elif self.inside_li and self.got_data:
|
||||
if 'failed' not in data:
|
||||
self.benchmarks[-1].add_metric(data)
|
||||
else:
|
||||
self.failed = True
|
BIN
wa/workloads/vellamo/com.arm.wa.uiauto.vellamo.apk
Normal file
BIN
wa/workloads/vellamo/com.arm.wa.uiauto.vellamo.apk
Normal file
Binary file not shown.
35
wa/workloads/vellamo/uiauto/app/build.gradle
Normal file
35
wa/workloads/vellamo/uiauto/app/build.gradle
Normal file
@ -0,0 +1,35 @@
|
||||
apply plugin: 'com.android.application'
|
||||
|
||||
def packageName = "com.arm.wa.uiauto.vellamo"
|
||||
|
||||
android {
|
||||
compileSdkVersion 25
|
||||
buildToolsVersion "25.0.3"
|
||||
defaultConfig {
|
||||
applicationId "${packageName}"
|
||||
minSdkVersion 18
|
||||
targetSdkVersion 25
|
||||
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
||||
}
|
||||
buildTypes {
|
||||
applicationVariants.all { variant ->
|
||||
variant.outputs.each { output ->
|
||||
output.outputFile = file("$project.buildDir/apk/${packageName}.apk")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
dependencies {
|
||||
compile fileTree(dir: 'libs', include: ['*.jar'])
|
||||
compile 'com.android.support.test:runner:0.5'
|
||||
compile 'com.android.support.test:rules:0.5'
|
||||
compile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2'
|
||||
compile(name: 'uiauto', ext:'aar')
|
||||
}
|
||||
|
||||
repositories {
|
||||
flatDir {
|
||||
dirs 'libs'
|
||||
}
|
||||
}
|
13
wa/workloads/vellamo/uiauto/app/src/main/AndroidManifest.xml
Normal file
13
wa/workloads/vellamo/uiauto/app/src/main/AndroidManifest.xml
Normal file
@ -0,0 +1,13 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
package="com.arm.wa.uiauto.vellamo"
|
||||
android:versionCode="1"
|
||||
android:versionName="1.0">
|
||||
|
||||
|
||||
<instrumentation
|
||||
android:name="android.support.test.runner.AndroidJUnitRunner"
|
||||
android:targetPackage="${applicationId}"/>
|
||||
|
||||
</manifest>
|
||||
|
@ -0,0 +1,311 @@
|
||||
|
||||
/* Copyright 2014-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.
|
||||
*/
|
||||
|
||||
|
||||
package com.arm.wa.uiauto.vellamo;
|
||||
|
||||
import android.os.Bundle;
|
||||
import android.support.test.runner.AndroidJUnit4;
|
||||
import android.support.test.uiautomator.UiObject;
|
||||
import android.support.test.uiautomator.UiObjectNotFoundException;
|
||||
import android.support.test.uiautomator.UiSelector;
|
||||
import android.support.test.uiautomator.UiWatcher;
|
||||
import android.util.Log;
|
||||
|
||||
import com.arm.wa.uiauto.BaseUiAutomation;
|
||||
import com.arm.wa.uiauto.UxPerfUiAutomation;
|
||||
|
||||
import org.junit.Test;
|
||||
import org.junit.Before;
|
||||
import org.junit.runner.RunWith;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
@RunWith(AndroidJUnit4.class)
|
||||
public class UiAutomation extends BaseUiAutomation {
|
||||
|
||||
public static String TAG = "vellamo";
|
||||
public static ArrayList<String> scores = new ArrayList();
|
||||
public static Boolean wasError = false;
|
||||
|
||||
protected UxPerfUiAutomation uxPerf;
|
||||
|
||||
Bundle parameters;
|
||||
String version;
|
||||
Boolean browser;
|
||||
Boolean metal;
|
||||
Boolean multicore;
|
||||
Integer browserToUse;
|
||||
|
||||
@Before
|
||||
public void initialize(){
|
||||
uxPerf = new UxPerfUiAutomation();
|
||||
parameters = getParams();
|
||||
version = parameters.getString("version");
|
||||
browser = parameters.getBoolean("browser");
|
||||
metal = parameters.getBoolean("metal");
|
||||
multicore = parameters.getBoolean("multicore");
|
||||
browserToUse = parameters.getInt("browserToUse") - 1;
|
||||
}
|
||||
|
||||
@Test
|
||||
public void setup() throws Exception {
|
||||
dismissEULA();
|
||||
if (version.equals("2.0.3")) {
|
||||
dissmissWelcomebanner();
|
||||
} else {
|
||||
dismissLetsRoll();
|
||||
if (version.equals("3.2.4")) {
|
||||
dismissArrow();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void runWorkload() throws Exception {
|
||||
if (version.equals("2.0.3")) {
|
||||
startTest();
|
||||
dismissNetworkConnectionDialogIfNecessary();
|
||||
dismissExplanationDialogIfNecessary();
|
||||
waitForTestCompletion(15 * 60, "com.quicinc.vellamo:id/act_ba_results_btn_no");
|
||||
} else {
|
||||
if (browser) {
|
||||
startBrowserTest(browserToUse, version);
|
||||
proccessTest("Browser");
|
||||
}
|
||||
if (multicore) {
|
||||
startTestV3(1, version);
|
||||
proccessTest("Multicore");
|
||||
}
|
||||
if (metal) {
|
||||
startTestV3(2, version);
|
||||
proccessTest("Metal");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void extractResults() throws Exception {
|
||||
for(String result : scores){
|
||||
Log.v(TAG, String.format("VELLAMO RESULT: %s", result));
|
||||
}
|
||||
if (wasError) Log.v("vellamoWatcher", "VELLAMO ERROR: Something crashed while running browser benchmark");
|
||||
}
|
||||
|
||||
public void startTest() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject runButton = mDevice.findObject(selector.textContains("Run All Chapters"));
|
||||
|
||||
if (!runButton.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
UiObject pager = mDevice.findObject(selector.className("android.support.v4.view.ViewPager"));
|
||||
pager.swipeLeft(2);
|
||||
if (!runButton.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find \"Run All Chapters\" button.");
|
||||
}
|
||||
}
|
||||
runButton.click();
|
||||
}
|
||||
|
||||
public void startBrowserTest(int browserToUse, String version) throws Exception {
|
||||
//Ensure chrome is selected as "browser" fails to run the benchmark
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject browserToUseButton = mDevice.findObject(selector.className("android.widget.ImageButton")
|
||||
.longClickable(true).instance(browserToUse));
|
||||
UiObject browserButton = mDevice.findObject(selector.className("android.widget.ImageButton")
|
||||
.longClickable(true).selected(true));
|
||||
//Disable browsers
|
||||
while(browserButton.exists()) browserButton.click();
|
||||
if (browserToUseButton.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (browserToUseButton.exists()) {
|
||||
browserToUseButton.click();
|
||||
}
|
||||
}
|
||||
|
||||
//enable a watcher to dismiss browser dialogs
|
||||
UiWatcher stoppedWorkingDialogWatcher = new UiWatcher() {
|
||||
@Override
|
||||
public boolean checkForCondition() {
|
||||
UiObject stoppedWorkingDialog = mDevice.findObject(new UiSelector().textStartsWith("Unfortunately"));
|
||||
if(stoppedWorkingDialog.exists()){
|
||||
wasError = true;
|
||||
UiObject okButton = mDevice.findObject(new UiSelector().className("android.widget.Button").text("OK"));
|
||||
try {
|
||||
okButton.click();
|
||||
} catch (UiObjectNotFoundException e) {
|
||||
// TODO Auto-generated catch block
|
||||
e.printStackTrace();
|
||||
}
|
||||
return (stoppedWorkingDialog.waitUntilGone(25000));
|
||||
}
|
||||
return false;
|
||||
}
|
||||
};
|
||||
// Register watcher
|
||||
mDevice.registerWatcher("stoppedWorkingDialogWatcher", stoppedWorkingDialogWatcher);
|
||||
|
||||
// Run watcher
|
||||
mDevice.runWatchers();
|
||||
|
||||
startTestV3(0, version);
|
||||
}
|
||||
|
||||
public void startTestV3(int run, String version) throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
|
||||
UiObject thirdRunButton = mDevice.findObject(selector.resourceId("com.quicinc.vellamo:id/card_launcher_run_button").instance(2));
|
||||
if (!thirdRunButton.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!thirdRunButton.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find three \"Run\" buttons.");
|
||||
}
|
||||
}
|
||||
|
||||
//Run benchmarks
|
||||
UiObject runButton = mDevice.findObject(selector.resourceId("com.quicinc.vellamo:id/card_launcher_run_button").instance(run));
|
||||
if (!runButton.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!runButton.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find correct \"Run\" button.");
|
||||
}
|
||||
}
|
||||
runButton.click();
|
||||
|
||||
//Skip tutorial screen
|
||||
if (version.equals("3.2.4")) {
|
||||
UiObject gotItButton = mDevice.findObject(selector.textContains("Got it"));
|
||||
if (!gotItButton.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!gotItButton.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find correct \"GOT IT\" button.");
|
||||
}
|
||||
}
|
||||
gotItButton.click();
|
||||
}
|
||||
|
||||
else {
|
||||
UiObject swipeScreen = mDevice.findObject(selector.textContains("Swipe left to continue"));
|
||||
if (!swipeScreen.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!swipeScreen.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find \"Swipe screen\".");
|
||||
}
|
||||
}
|
||||
sleep(1);
|
||||
swipeScreen.swipeLeft(2);
|
||||
sleep(1);
|
||||
swipeScreen.swipeLeft(2);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public void proccessTest(String metric) throws Exception{
|
||||
waitForTestCompletion(15 * 60, "com.quicinc.vellamo:id/button_no");
|
||||
|
||||
//Remove watcher
|
||||
mDevice.removeWatcher("stoppedWorkingDialogWatcher");
|
||||
|
||||
getScore(metric, "com.quicinc.vellamo:id/card_score_score");
|
||||
mDevice.pressBack();
|
||||
mDevice.pressBack();
|
||||
mDevice.pressBack();
|
||||
}
|
||||
|
||||
public void getScore(String metric, String resourceID) throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject score = mDevice.findObject(selector.resourceId(resourceID));
|
||||
if (!score.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!score.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find score on screen.");
|
||||
}
|
||||
}
|
||||
scores.add(metric + " " + score.getText().trim());
|
||||
}
|
||||
|
||||
public void waitForTestCompletion(int timeout, String resourceID) throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject resultsNoButton = mDevice.findObject(selector.resourceId(resourceID));
|
||||
if (!resultsNoButton.waitForExists(TimeUnit.SECONDS.toMillis(timeout))) {
|
||||
throw new UiObjectNotFoundException("Did not see results screen.");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public void dismissEULA() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
waitText("Vellamo EULA");
|
||||
UiObject acceptButton = mDevice.findObject(selector.textMatches("Accept|ACCEPT")
|
||||
.className("android.widget.Button"));
|
||||
if (acceptButton.exists()) {
|
||||
acceptButton.click();
|
||||
}
|
||||
}
|
||||
|
||||
public void dissmissWelcomebanner() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject welcomeBanner = mDevice.findObject(selector.textContains("WELCOME"));
|
||||
if (welcomeBanner.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
UiObject pager = mDevice.findObject(selector.className("android.support.v4.view.ViewPager"));
|
||||
pager.swipeLeft(2);
|
||||
pager.swipeLeft(2);
|
||||
}
|
||||
}
|
||||
|
||||
public void dismissLetsRoll() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject letsRollButton = mDevice.findObject(selector.className("android.widget.Button")
|
||||
.textContains("LET'S ROLL"));
|
||||
if (!letsRollButton.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!letsRollButton.exists()) {
|
||||
// As a fall-back look for the old capitalization
|
||||
letsRollButton = mDevice.findObject(selector.className("android.widget.Button")
|
||||
.textContains("Let's Roll"));
|
||||
if (!letsRollButton.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find \"Let's Roll\" button.");
|
||||
}
|
||||
}
|
||||
}
|
||||
letsRollButton.click();
|
||||
}
|
||||
|
||||
public void dismissArrow() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject cardContainer = mDevice.findObject(selector.resourceId("com.quicinc.vellamo:id/cards_container")) ;
|
||||
if (!cardContainer.waitForExists(TimeUnit.SECONDS.toMillis(5))) {
|
||||
if (!cardContainer.exists()) {
|
||||
throw new UiObjectNotFoundException("Could not find vellamo main screen");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void dismissNetworkConnectionDialogIfNecessary() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject dialog = mDevice.findObject(selector.className("android.widget.TextView")
|
||||
.textContains("No Network Connection"));
|
||||
if (dialog.exists()) {
|
||||
UiObject yesButton = mDevice.findObject(selector.className("android.widget.Button")
|
||||
.text("Yes"));
|
||||
yesButton.click();
|
||||
}
|
||||
}
|
||||
|
||||
public void dismissExplanationDialogIfNecessary() throws Exception {
|
||||
UiSelector selector = new UiSelector();
|
||||
UiObject dialog = mDevice.findObject(selector.className("android.widget.TextView")
|
||||
.textContains("Benchmarks Explanation"));
|
||||
if (dialog.exists()) {
|
||||
UiObject noButton = mDevice.findObject(selector.className("android.widget.Button")
|
||||
.text("No"));
|
||||
noButton.click();
|
||||
}
|
||||
}
|
||||
}
|
23
wa/workloads/vellamo/uiauto/build.gradle
Normal file
23
wa/workloads/vellamo/uiauto/build.gradle
Normal file
@ -0,0 +1,23 @@
|
||||
// Top-level build file where you can add configuration options common to all sub-projects/modules.
|
||||
|
||||
buildscript {
|
||||
repositories {
|
||||
jcenter()
|
||||
}
|
||||
dependencies {
|
||||
classpath 'com.android.tools.build:gradle:2.3.2'
|
||||
|
||||
// NOTE: Do not place your application dependencies here; they belong
|
||||
// in the individual module build.gradle files
|
||||
}
|
||||
}
|
||||
|
||||
allprojects {
|
||||
repositories {
|
||||
jcenter()
|
||||
}
|
||||
}
|
||||
|
||||
task clean(type: Delete) {
|
||||
delete rootProject.buildDir
|
||||
}
|
55
wa/workloads/vellamo/uiauto/build.sh
Executable file
55
wa/workloads/vellamo/uiauto/build.sh
Executable file
@ -0,0 +1,55 @@
|
||||
#!/bin/bash
|
||||
# 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.
|
||||
#
|
||||
set -e
|
||||
|
||||
# CD into build dir if possible - allows building from any directory
|
||||
script_path='.'
|
||||
if `readlink -f $0 &>/dev/null`; then
|
||||
script_path=`readlink -f $0 2>/dev/null`
|
||||
fi
|
||||
script_dir=`dirname $script_path`
|
||||
cd $script_dir
|
||||
|
||||
# Ensure gradelw exists before starting
|
||||
if [[ ! -f gradlew ]]; then
|
||||
echo 'gradlew file not found! Check that you are in the right directory.'
|
||||
exit 9
|
||||
fi
|
||||
|
||||
# Copy base class library from wa dist
|
||||
libs_dir=app/libs
|
||||
base_class=`python -c "import os, wa; print os.path.join(os.path.dirname(wa.__file__), 'framework', 'uiauto', 'uiauto.aar')"`
|
||||
mkdir -p $libs_dir
|
||||
cp $base_class $libs_dir
|
||||
|
||||
# Build and return appropriate exit code if failed
|
||||
# gradle build
|
||||
./gradlew clean :app:assembleDebug
|
||||
exit_code=$?
|
||||
if [[ $exit_code -ne 0 ]]; then
|
||||
echo "ERROR: 'gradle build' exited with code $exit_code"
|
||||
exit $exit_code
|
||||
fi
|
||||
|
||||
# If successful move APK file to workload folder (overwrite previous)
|
||||
package=com.arm.wa.uiauto.vellamo
|
||||
rm -f ../$package
|
||||
if [[ -f app/build/apk/$package.apk ]]; then
|
||||
cp app/build/apk/$package.apk ../$package.apk
|
||||
else
|
||||
echo 'ERROR: UiAutomator apk could not be found!'
|
||||
exit 9
|
||||
fi
|
BIN
wa/workloads/vellamo/uiauto/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
wa/workloads/vellamo/uiauto/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
6
wa/workloads/vellamo/uiauto/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
6
wa/workloads/vellamo/uiauto/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,6 @@
|
||||
#Wed May 03 15:42:44 BST 2017
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-3.3-all.zip
|
160
wa/workloads/vellamo/uiauto/gradlew
vendored
Executable file
160
wa/workloads/vellamo/uiauto/gradlew
vendored
Executable file
@ -0,0 +1,160 @@
|
||||
#!/usr/bin/env bash
|
||||
|
||||
##############################################################################
|
||||
##
|
||||
## Gradle start up script for UN*X
|
||||
##
|
||||
##############################################################################
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS=""
|
||||
|
||||
APP_NAME="Gradle"
|
||||
APP_BASE_NAME=`basename "$0"`
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD="maximum"
|
||||
|
||||
warn ( ) {
|
||||
echo "$*"
|
||||
}
|
||||
|
||||
die ( ) {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
}
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
case "`uname`" in
|
||||
CYGWIN* )
|
||||
cygwin=true
|
||||
;;
|
||||
Darwin* )
|
||||
darwin=true
|
||||
;;
|
||||
MINGW* )
|
||||
msys=true
|
||||
;;
|
||||
esac
|
||||
|
||||
# Attempt to set APP_HOME
|
||||
# Resolve links: $0 may be a link
|
||||
PRG="$0"
|
||||
# Need this for relative symlinks.
|
||||
while [ -h "$PRG" ] ; do
|
||||
ls=`ls -ld "$PRG"`
|
||||
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||
if expr "$link" : '/.*' > /dev/null; then
|
||||
PRG="$link"
|
||||
else
|
||||
PRG=`dirname "$PRG"`"/$link"
|
||||
fi
|
||||
done
|
||||
SAVED="`pwd`"
|
||||
cd "`dirname \"$PRG\"`/" >/dev/null
|
||||
APP_HOME="`pwd -P`"
|
||||
cd "$SAVED" >/dev/null
|
||||
|
||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||
|
||||
# Determine the Java command to use to start the JVM.
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||
else
|
||||
JAVACMD="$JAVA_HOME/bin/java"
|
||||
fi
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
else
|
||||
JAVACMD="java"
|
||||
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
|
||||
# Increase the maximum file descriptors if we can.
|
||||
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
|
||||
MAX_FD_LIMIT=`ulimit -H -n`
|
||||
if [ $? -eq 0 ] ; then
|
||||
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||
MAX_FD="$MAX_FD_LIMIT"
|
||||
fi
|
||||
ulimit -n $MAX_FD
|
||||
if [ $? -ne 0 ] ; then
|
||||
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||
fi
|
||||
else
|
||||
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||
fi
|
||||
fi
|
||||
|
||||
# For Darwin, add options to specify how the application appears in the dock
|
||||
if $darwin; then
|
||||
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||
fi
|
||||
|
||||
# For Cygwin, switch paths to Windows format before running java
|
||||
if $cygwin ; then
|
||||
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||
|
||||
# We build the pattern for arguments to be converted via cygpath
|
||||
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||
SEP=""
|
||||
for dir in $ROOTDIRSRAW ; do
|
||||
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||
SEP="|"
|
||||
done
|
||||
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||
# Add a user-defined pattern to the cygpath arguments
|
||||
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||
fi
|
||||
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||
i=0
|
||||
for arg in "$@" ; do
|
||||
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||
|
||||
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||
else
|
||||
eval `echo args$i`="\"$arg\""
|
||||
fi
|
||||
i=$((i+1))
|
||||
done
|
||||
case $i in
|
||||
(0) set -- ;;
|
||||
(1) set -- "$args0" ;;
|
||||
(2) set -- "$args0" "$args1" ;;
|
||||
(3) set -- "$args0" "$args1" "$args2" ;;
|
||||
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||
esac
|
||||
fi
|
||||
|
||||
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
|
||||
function splitJvmOpts() {
|
||||
JVM_OPTS=("$@")
|
||||
}
|
||||
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
|
||||
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
|
||||
|
||||
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
|
90
wa/workloads/vellamo/uiauto/gradlew.bat
vendored
Normal file
90
wa/workloads/vellamo/uiauto/gradlew.bat
vendored
Normal file
@ -0,0 +1,90 @@
|
||||
@if "%DEBUG%" == "" @echo off
|
||||
@rem ##########################################################################
|
||||
@rem
|
||||
@rem Gradle startup script for Windows
|
||||
@rem
|
||||
@rem ##########################################################################
|
||||
|
||||
@rem Set local scope for the variables with windows NT shell
|
||||
if "%OS%"=="Windows_NT" setlocal
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS=
|
||||
|
||||
set DIRNAME=%~dp0
|
||||
if "%DIRNAME%" == "" set DIRNAME=.
|
||||
set APP_BASE_NAME=%~n0
|
||||
set APP_HOME=%DIRNAME%
|
||||
|
||||
@rem Find java.exe
|
||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||
|
||||
set JAVA_EXE=java.exe
|
||||
%JAVA_EXE% -version >NUL 2>&1
|
||||
if "%ERRORLEVEL%" == "0" goto init
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:findJavaFromJavaHome
|
||||
set JAVA_HOME=%JAVA_HOME:"=%
|
||||
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||
|
||||
if exist "%JAVA_EXE%" goto init
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:init
|
||||
@rem Get command-line arguments, handling Windowz variants
|
||||
|
||||
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||
if "%@eval[2+2]" == "4" goto 4NT_args
|
||||
|
||||
:win9xME_args
|
||||
@rem Slurp the command line arguments.
|
||||
set CMD_LINE_ARGS=
|
||||
set _SKIP=2
|
||||
|
||||
:win9xME_args_slurp
|
||||
if "x%~1" == "x" goto execute
|
||||
|
||||
set CMD_LINE_ARGS=%*
|
||||
goto execute
|
||||
|
||||
:4NT_args
|
||||
@rem Get arguments from the 4NT Shell from JP Software
|
||||
set CMD_LINE_ARGS=%$
|
||||
|
||||
:execute
|
||||
@rem Setup the command line
|
||||
|
||||
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||
|
||||
@rem Execute Gradle
|
||||
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
||||
|
||||
:end
|
||||
@rem End local scope for the variables with windows NT shell
|
||||
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||
|
||||
:fail
|
||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||
rem the _cmd.exe /c_ return code!
|
||||
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||
exit /b 1
|
||||
|
||||
:mainEnd
|
||||
if "%OS%"=="Windows_NT" endlocal
|
||||
|
||||
:omega
|
1
wa/workloads/vellamo/uiauto/settings.gradle
Normal file
1
wa/workloads/vellamo/uiauto/settings.gradle
Normal file
@ -0,0 +1 @@
|
||||
include ':app'
|
89
wa/workloads/youtube/__init__.py
Executable file
89
wa/workloads/youtube/__init__.py
Executable file
@ -0,0 +1,89 @@
|
||||
# Copyright 2014-2016 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.
|
||||
#
|
||||
|
||||
from wa import Parameter, ApkUiautoWorkload
|
||||
from wa.framework.exception import ConfigError
|
||||
|
||||
|
||||
class Youtube(ApkUiautoWorkload):
|
||||
|
||||
name = 'youtube'
|
||||
description = '''
|
||||
A workload to perform standard productivity tasks within YouTube.
|
||||
|
||||
The workload plays a video from the app, determined by the ``video_source`` parameter.
|
||||
While the video is playing, a some common actions are done such as video seeking, pausing
|
||||
playback and navigating the comments section.
|
||||
|
||||
Test description:
|
||||
The ``video_source`` parameter determines where the video to be played will be found
|
||||
in the app. Possible values are ``search``, ``home``, ``my_videos``, and ``trending``.
|
||||
|
||||
-A. search - Goes to the search view, does a search for the given term, and plays the
|
||||
first video in the results. The parameter ``search_term`` must also be provided
|
||||
in the agenda for this to work. This is the default mode.
|
||||
-B. home - Scrolls down once on the app's home page to avoid ads (if present, would be
|
||||
first video), then select and plays the video that appears at the top of the list.
|
||||
-C. my_videos - Goes to the 'My Videos' section of the user's account page and plays a
|
||||
video from there. The user must have at least one uploaded video for this to work.
|
||||
-D. trending - Goes to the 'Trending Videos' section of the app, and plays the first
|
||||
video in the trending videos list.
|
||||
|
||||
For the selected video source, the following test steps are performed:
|
||||
|
||||
1. Navigate to the general app settings page to disable autoplay. This improves test
|
||||
stability and predictability by preventing screen transition to load a new video
|
||||
while in the middle of the test.
|
||||
2. Select the video from the source specified above, and dismiss any potential embedded
|
||||
advert that may pop-up before the actual video.
|
||||
3. Let the video play for a few seconds, pause it, then resume.
|
||||
4. Expand the info card that shows video metadata, then collapse it again.
|
||||
5. Scroll down to the end of related videos and comments under the info card, and then
|
||||
back up to the start. A maximum of 5 swipe actions is performed in either direction.
|
||||
|
||||
Known working APK version: 11.19.56
|
||||
'''
|
||||
package_names = ['com.google.android.youtube']
|
||||
|
||||
parameters = [
|
||||
Parameter('video_source', kind=str, default='search',
|
||||
allowed_values=['home', 'my_videos', 'search', 'trending'],
|
||||
description='''
|
||||
Determines where to play the video from. This can either be from the
|
||||
YouTube home, my videos section, trending videos or found in search.
|
||||
'''),
|
||||
Parameter('search_term', kind=str,
|
||||
default='Big Buck Bunny 60fps 4K - Official Blender Foundation Short Film',
|
||||
description='''
|
||||
The search term to use when ``video_source`` is set to ``search``.
|
||||
Ignored otherwise.
|
||||
'''),
|
||||
]
|
||||
|
||||
# This workload relies on the internet so check that there is a working
|
||||
# internet connection
|
||||
requires_network = True
|
||||
|
||||
def __init__(self, device, **kwargs):
|
||||
super(Youtube, self).__init__(device, **kwargs)
|
||||
self.run_timeout = 300
|
||||
|
||||
def validate(self):
|
||||
super(Youtube, self).validate()
|
||||
self.gui.uiauto_params['video_source'] = self.video_source
|
||||
self.gui.uiauto_params['search_term'] = self.search_term
|
||||
# Make sure search term is set if video source is 'search'
|
||||
if (self.video_source == 'search') and not self.search_term:
|
||||
raise ConfigError("Param 'search_term' must be specified when video source is 'search'")
|
BIN
wa/workloads/youtube/com.arm.wa.uiauto.youtube.apk
Normal file
BIN
wa/workloads/youtube/com.arm.wa.uiauto.youtube.apk
Normal file
Binary file not shown.
35
wa/workloads/youtube/uiauto/app/build.gradle
Normal file
35
wa/workloads/youtube/uiauto/app/build.gradle
Normal file
@ -0,0 +1,35 @@
|
||||
apply plugin: 'com.android.application'
|
||||
|
||||
def packageName = "com.arm.wa.uiauto.youtube"
|
||||
|
||||
android {
|
||||
compileSdkVersion 25
|
||||
buildToolsVersion "25.0.3"
|
||||
defaultConfig {
|
||||
applicationId "${packageName}"
|
||||
minSdkVersion 18
|
||||
targetSdkVersion 25
|
||||
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
||||
}
|
||||
buildTypes {
|
||||
applicationVariants.all { variant ->
|
||||
variant.outputs.each { output ->
|
||||
output.outputFile = file("$project.buildDir/apk/${packageName}.apk")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
dependencies {
|
||||
compile fileTree(dir: 'libs', include: ['*.jar'])
|
||||
compile 'com.android.support.test:runner:0.5'
|
||||
compile 'com.android.support.test:rules:0.5'
|
||||
compile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2'
|
||||
compile(name: 'uiauto', ext:'aar')
|
||||
}
|
||||
|
||||
repositories {
|
||||
flatDir {
|
||||
dirs 'libs'
|
||||
}
|
||||
}
|
13
wa/workloads/youtube/uiauto/app/src/main/AndroidManifest.xml
Normal file
13
wa/workloads/youtube/uiauto/app/src/main/AndroidManifest.xml
Normal file
@ -0,0 +1,13 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
package="com.arm.wa.uiauto.youtube"
|
||||
android:versionCode="1"
|
||||
android:versionName="1.0">
|
||||
|
||||
|
||||
<instrumentation
|
||||
android:name="android.support.test.runner.AndroidJUnitRunner"
|
||||
android:targetPackage="${applicationId}"/>
|
||||
|
||||
</manifest>
|
||||
|
@ -0,0 +1,268 @@
|
||||
/* Copyright 2014-2016 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.
|
||||
*/
|
||||
|
||||
package com.arm.wa.uiauto.youtube;
|
||||
|
||||
import android.os.Bundle;
|
||||
import android.os.SystemClock;
|
||||
import android.support.test.internal.runner.ClassPathScanner;
|
||||
import android.support.test.runner.AndroidJUnit4;
|
||||
import android.support.test.uiautomator.UiObject;
|
||||
import android.support.test.uiautomator.UiScrollable;
|
||||
import android.support.test.uiautomator.UiSelector;
|
||||
|
||||
import com.arm.wa.uiauto.BaseUiAutomation;
|
||||
import com.arm.wa.uiauto.ActionLogger;
|
||||
import com.arm.wa.uiauto.UxPerfUiAutomation;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
import org.junit.runner.RunWith;
|
||||
|
||||
import static com.arm.wa.uiauto.BaseUiAutomation.FindByCriteria.BY_DESC;
|
||||
import static com.arm.wa.uiauto.BaseUiAutomation.FindByCriteria.BY_ID;
|
||||
import static com.arm.wa.uiauto.BaseUiAutomation.FindByCriteria.BY_TEXT;
|
||||
|
||||
@RunWith(AndroidJUnit4.class)
|
||||
public class UiAutomation extends BaseUiAutomation {
|
||||
|
||||
public static final String SOURCE_MY_VIDEOS = "my_videos";
|
||||
public static final String SOURCE_SEARCH = "search";
|
||||
public static final String SOURCE_TRENDING = "trending";
|
||||
|
||||
public static final int WAIT_TIMEOUT_1SEC = 1000;
|
||||
public static final int VIDEO_SLEEP_SECONDS = 3;
|
||||
public static final int LIST_SWIPE_COUNT = 5;
|
||||
|
||||
protected UxPerfUiAutomation uxPerf;
|
||||
|
||||
protected Bundle parameters;
|
||||
protected String packageID;
|
||||
|
||||
@Before
|
||||
public void initilize() {
|
||||
uxPerf = new UxPerfUiAutomation();
|
||||
parameters = getParams();
|
||||
packageID = uxPerf.getPackageID(parameters);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void setup() throws Exception {
|
||||
mDevice.setOrientationNatural();
|
||||
runApplicationInitialization();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void runWorkload() throws Exception {
|
||||
String videoSource = parameters.getString("video_source");
|
||||
String searchTerm = parameters.getString("search_term");
|
||||
testPlayVideo(videoSource, searchTerm);
|
||||
dismissAdvert();
|
||||
checkPlayerError();
|
||||
pausePlayVideo();
|
||||
checkVideoInfo();
|
||||
scrollRelated();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void teardown() throws Exception {
|
||||
mDevice.unfreezeRotation();
|
||||
}
|
||||
|
||||
// Get application parameters and clear the initial run dialogues of the application launch.
|
||||
public void runApplicationInitialization() throws Exception {
|
||||
clearFirstRunDialogues();
|
||||
disableAutoplay();
|
||||
}
|
||||
|
||||
// Sets the UiObject that marks the end of the application launch.
|
||||
public UiObject getLaunchEndObject() {
|
||||
UiObject launchEndObject = mDevice.findObject(new UiSelector()
|
||||
.resourceId(packageID + "menu_search"));
|
||||
return launchEndObject;
|
||||
}
|
||||
|
||||
public void clearFirstRunDialogues() throws Exception {
|
||||
UiObject laterButton =
|
||||
mDevice.findObject(new UiSelector().textContains("Later")
|
||||
.className("android.widget.TextView"));
|
||||
if (laterButton.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
laterButton.click();
|
||||
}
|
||||
|
||||
UiObject cancelButton =
|
||||
mDevice.findObject(new UiSelector().textContains("Cancel")
|
||||
.className("android.widget.Button"));
|
||||
if (cancelButton.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
cancelButton.click();
|
||||
}
|
||||
|
||||
UiObject skipButton =
|
||||
mDevice.findObject(new UiSelector().textContains("Skip")
|
||||
.className("android.widget.TextView"));
|
||||
if (skipButton.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
skipButton.click();
|
||||
}
|
||||
|
||||
UiObject gotItButton =
|
||||
mDevice.findObject(new UiSelector().textContains("Got it")
|
||||
.className("android.widget.Button"));
|
||||
if (gotItButton.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
gotItButton.click();
|
||||
}
|
||||
}
|
||||
|
||||
public void disableAutoplay() throws Exception {
|
||||
clickUiObject(BY_DESC, "More options");
|
||||
clickUiObject(BY_TEXT, "Settings", true);
|
||||
clickUiObject(BY_TEXT, "General", true);
|
||||
|
||||
// Don't fail fatally if autoplay toggle cannot be found
|
||||
UiObject autoplayToggle =
|
||||
mDevice.findObject(new UiSelector().textContains("Autoplay"));
|
||||
if (autoplayToggle.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
autoplayToggle.click();
|
||||
}
|
||||
mDevice.pressBack();
|
||||
|
||||
// Tablet devices use a split with General in the left pane and Autoplay in the right so no
|
||||
// need to click back twice
|
||||
UiObject generalButton =
|
||||
mDevice.findObject(new UiSelector().textContains("General")
|
||||
.className("android.widget.TextView"));
|
||||
if (generalButton.exists()) {
|
||||
mDevice.pressBack();
|
||||
}
|
||||
}
|
||||
|
||||
public void testPlayVideo(String source, String searchTerm) throws Exception {
|
||||
String testTag = "play";
|
||||
ActionLogger logger = new ActionLogger(testTag + "_" + source, parameters);
|
||||
|
||||
if (SOURCE_SEARCH.equalsIgnoreCase(source)) {
|
||||
clickUiObject(BY_DESC, "Search");
|
||||
UiObject textField = getUiObjectByResourceId(packageID + "search_edit_text");
|
||||
textField.setText(searchTerm);
|
||||
mDevice.pressEnter();
|
||||
// If a video exists whose title contains the exact search term, then play it
|
||||
// Otherwise click the first video in the search results
|
||||
UiObject thumbnail =
|
||||
mDevice.findObject(new UiSelector().resourceId(packageID + "thumbnail"));
|
||||
UiObject matchedVideo =
|
||||
thumbnail.getFromParent(new UiSelector().textContains(searchTerm));
|
||||
|
||||
logger.start();
|
||||
if (matchedVideo.exists()) {
|
||||
matchedVideo.clickAndWaitForNewWindow();
|
||||
} else {
|
||||
thumbnail.clickAndWaitForNewWindow();
|
||||
}
|
||||
logger.stop();
|
||||
|
||||
} else if (SOURCE_MY_VIDEOS.equalsIgnoreCase(source)) {
|
||||
clickUiObject(BY_DESC, "Account");
|
||||
clickUiObject(BY_TEXT, "My Videos", true);
|
||||
|
||||
logger.start();
|
||||
clickUiObject(BY_ID, packageID + "thumbnail", true);
|
||||
logger.stop();
|
||||
|
||||
} else if (SOURCE_TRENDING.equalsIgnoreCase(source)) {
|
||||
clickUiObject(BY_DESC, "Trending");
|
||||
|
||||
logger.start();
|
||||
clickUiObject(BY_ID, packageID + "thumbnail", true);
|
||||
logger.stop();
|
||||
|
||||
} else { // homepage videos
|
||||
UiScrollable list =
|
||||
new UiScrollable(new UiSelector().resourceId(packageID + "results"));
|
||||
if (list.exists()) {
|
||||
list.scrollForward();
|
||||
}
|
||||
|
||||
logger.start();
|
||||
clickUiObject(BY_ID, packageID + "thumbnail", true);
|
||||
logger.stop();
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
public void dismissAdvert() throws Exception {
|
||||
UiObject advert =
|
||||
mDevice.findObject(new UiSelector().textContains("Visit advertiser"));
|
||||
if (advert.exists()) {
|
||||
UiObject skip =
|
||||
mDevice.findObject(new UiSelector().textContains("Skip ad"));
|
||||
if (skip.waitForExists(WAIT_TIMEOUT_1SEC*5)) {
|
||||
skip.click();
|
||||
sleep(VIDEO_SLEEP_SECONDS);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void checkPlayerError() throws Exception {
|
||||
UiObject playerError =
|
||||
mDevice.findObject(new UiSelector().resourceId(packageID + "player_error_view"));
|
||||
UiObject tapToRetry =
|
||||
mDevice.findObject(new UiSelector().textContains("Tap to retry"));
|
||||
if (playerError.waitForExists(WAIT_TIMEOUT_1SEC) || tapToRetry.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
throw new RuntimeException("Video player encountered an error and cannot continue.");
|
||||
}
|
||||
}
|
||||
|
||||
public void pausePlayVideo() throws Exception {
|
||||
UiObject player = getUiObjectByResourceId(packageID + "player_fragment_container");
|
||||
sleep(VIDEO_SLEEP_SECONDS);
|
||||
repeatClickUiObject(player, 2, 100);
|
||||
sleep(1); // pause the video momentarily
|
||||
player.click();
|
||||
sleep(VIDEO_SLEEP_SECONDS);
|
||||
}
|
||||
|
||||
public void checkVideoInfo() throws Exception {
|
||||
UiObject expandButton =
|
||||
mDevice.findObject(new UiSelector().resourceId(packageID + "expand_button"));
|
||||
if (!expandButton.waitForExists(WAIT_TIMEOUT_1SEC)) {
|
||||
return;
|
||||
}
|
||||
// Expand video info
|
||||
expandButton.click();
|
||||
sleep(1); // short delay to simulate user action
|
||||
expandButton.click();
|
||||
}
|
||||
|
||||
public void scrollRelated() throws Exception {
|
||||
String testTag = "scroll";
|
||||
|
||||
// ListView of related videos and (maybe) comments
|
||||
UiScrollable list =
|
||||
new UiScrollable(new UiSelector().resourceId(packageID + "watch_list"));
|
||||
if (list.isScrollable()) {
|
||||
ActionLogger logger = new ActionLogger(testTag + "_down", parameters);
|
||||
logger.start();
|
||||
list.flingToEnd(LIST_SWIPE_COUNT);
|
||||
logger.stop();
|
||||
|
||||
logger = new ActionLogger(testTag + "_up", parameters);
|
||||
logger.start();
|
||||
list.flingToBeginning(LIST_SWIPE_COUNT);
|
||||
logger.stop();
|
||||
}
|
||||
// After flinging, give the window enough time to settle down before
|
||||
// the next step, or else UiAutomator fails to find views in time
|
||||
sleep(VIDEO_SLEEP_SECONDS);
|
||||
}
|
||||
}
|
23
wa/workloads/youtube/uiauto/build.gradle
Normal file
23
wa/workloads/youtube/uiauto/build.gradle
Normal file
@ -0,0 +1,23 @@
|
||||
// Top-level build file where you can add configuration options common to all sub-projects/modules.
|
||||
|
||||
buildscript {
|
||||
repositories {
|
||||
jcenter()
|
||||
}
|
||||
dependencies {
|
||||
classpath 'com.android.tools.build:gradle:2.3.2'
|
||||
|
||||
// NOTE: Do not place your application dependencies here; they belong
|
||||
// in the individual module build.gradle files
|
||||
}
|
||||
}
|
||||
|
||||
allprojects {
|
||||
repositories {
|
||||
jcenter()
|
||||
}
|
||||
}
|
||||
|
||||
task clean(type: Delete) {
|
||||
delete rootProject.buildDir
|
||||
}
|
55
wa/workloads/youtube/uiauto/build.sh
Executable file
55
wa/workloads/youtube/uiauto/build.sh
Executable file
@ -0,0 +1,55 @@
|
||||
#!/bin/bash
|
||||
# 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.
|
||||
#
|
||||
set -e
|
||||
|
||||
# CD into build dir if possible - allows building from any directory
|
||||
script_path='.'
|
||||
if `readlink -f $0 &>/dev/null`; then
|
||||
script_path=`readlink -f $0 2>/dev/null`
|
||||
fi
|
||||
script_dir=`dirname $script_path`
|
||||
cd $script_dir
|
||||
|
||||
# Ensure gradelw exists before starting
|
||||
if [[ ! -f gradlew ]]; then
|
||||
echo 'gradlew file not found! Check that you are in the right directory.'
|
||||
exit 9
|
||||
fi
|
||||
|
||||
# Copy base class library from wa dist
|
||||
libs_dir=app/libs
|
||||
base_class=`python -c "import os, wa; print os.path.join(os.path.dirname(wa.__file__), 'framework', 'uiauto', 'uiauto.aar')"`
|
||||
mkdir -p $libs_dir
|
||||
cp $base_class $libs_dir
|
||||
|
||||
# Build and return appropriate exit code if failed
|
||||
# gradle build
|
||||
./gradlew clean :app:assembleDebug
|
||||
exit_code=$?
|
||||
if [[ $exit_code -ne 0 ]]; then
|
||||
echo "ERROR: 'gradle build' exited with code $exit_code"
|
||||
exit $exit_code
|
||||
fi
|
||||
|
||||
# If successful move APK file to workload folder (overwrite previous)
|
||||
package=com.arm.wa.uiauto.youtube
|
||||
rm -f ../$package
|
||||
if [[ -f app/build/apk/$package.apk ]]; then
|
||||
cp app/build/apk/$package.apk ../$package.apk
|
||||
else
|
||||
echo 'ERROR: UiAutomator apk could not be found!'
|
||||
exit 9
|
||||
fi
|
BIN
wa/workloads/youtube/uiauto/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
wa/workloads/youtube/uiauto/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
6
wa/workloads/youtube/uiauto/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
6
wa/workloads/youtube/uiauto/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,6 @@
|
||||
#Wed May 03 15:42:44 BST 2017
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-3.3-all.zip
|
160
wa/workloads/youtube/uiauto/gradlew
vendored
Executable file
160
wa/workloads/youtube/uiauto/gradlew
vendored
Executable file
@ -0,0 +1,160 @@
|
||||
#!/usr/bin/env bash
|
||||
|
||||
##############################################################################
|
||||
##
|
||||
## Gradle start up script for UN*X
|
||||
##
|
||||
##############################################################################
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS=""
|
||||
|
||||
APP_NAME="Gradle"
|
||||
APP_BASE_NAME=`basename "$0"`
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD="maximum"
|
||||
|
||||
warn ( ) {
|
||||
echo "$*"
|
||||
}
|
||||
|
||||
die ( ) {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
}
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
case "`uname`" in
|
||||
CYGWIN* )
|
||||
cygwin=true
|
||||
;;
|
||||
Darwin* )
|
||||
darwin=true
|
||||
;;
|
||||
MINGW* )
|
||||
msys=true
|
||||
;;
|
||||
esac
|
||||
|
||||
# Attempt to set APP_HOME
|
||||
# Resolve links: $0 may be a link
|
||||
PRG="$0"
|
||||
# Need this for relative symlinks.
|
||||
while [ -h "$PRG" ] ; do
|
||||
ls=`ls -ld "$PRG"`
|
||||
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||
if expr "$link" : '/.*' > /dev/null; then
|
||||
PRG="$link"
|
||||
else
|
||||
PRG=`dirname "$PRG"`"/$link"
|
||||
fi
|
||||
done
|
||||
SAVED="`pwd`"
|
||||
cd "`dirname \"$PRG\"`/" >/dev/null
|
||||
APP_HOME="`pwd -P`"
|
||||
cd "$SAVED" >/dev/null
|
||||
|
||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||
|
||||
# Determine the Java command to use to start the JVM.
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||
else
|
||||
JAVACMD="$JAVA_HOME/bin/java"
|
||||
fi
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
else
|
||||
JAVACMD="java"
|
||||
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
|
||||
# Increase the maximum file descriptors if we can.
|
||||
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
|
||||
MAX_FD_LIMIT=`ulimit -H -n`
|
||||
if [ $? -eq 0 ] ; then
|
||||
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||
MAX_FD="$MAX_FD_LIMIT"
|
||||
fi
|
||||
ulimit -n $MAX_FD
|
||||
if [ $? -ne 0 ] ; then
|
||||
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||
fi
|
||||
else
|
||||
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||
fi
|
||||
fi
|
||||
|
||||
# For Darwin, add options to specify how the application appears in the dock
|
||||
if $darwin; then
|
||||
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||
fi
|
||||
|
||||
# For Cygwin, switch paths to Windows format before running java
|
||||
if $cygwin ; then
|
||||
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||
|
||||
# We build the pattern for arguments to be converted via cygpath
|
||||
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||
SEP=""
|
||||
for dir in $ROOTDIRSRAW ; do
|
||||
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||
SEP="|"
|
||||
done
|
||||
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||
# Add a user-defined pattern to the cygpath arguments
|
||||
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||
fi
|
||||
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||
i=0
|
||||
for arg in "$@" ; do
|
||||
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||
|
||||
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||
else
|
||||
eval `echo args$i`="\"$arg\""
|
||||
fi
|
||||
i=$((i+1))
|
||||
done
|
||||
case $i in
|
||||
(0) set -- ;;
|
||||
(1) set -- "$args0" ;;
|
||||
(2) set -- "$args0" "$args1" ;;
|
||||
(3) set -- "$args0" "$args1" "$args2" ;;
|
||||
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||
esac
|
||||
fi
|
||||
|
||||
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
|
||||
function splitJvmOpts() {
|
||||
JVM_OPTS=("$@")
|
||||
}
|
||||
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
|
||||
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
|
||||
|
||||
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
|
90
wa/workloads/youtube/uiauto/gradlew.bat
vendored
Normal file
90
wa/workloads/youtube/uiauto/gradlew.bat
vendored
Normal file
@ -0,0 +1,90 @@
|
||||
@if "%DEBUG%" == "" @echo off
|
||||
@rem ##########################################################################
|
||||
@rem
|
||||
@rem Gradle startup script for Windows
|
||||
@rem
|
||||
@rem ##########################################################################
|
||||
|
||||
@rem Set local scope for the variables with windows NT shell
|
||||
if "%OS%"=="Windows_NT" setlocal
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS=
|
||||
|
||||
set DIRNAME=%~dp0
|
||||
if "%DIRNAME%" == "" set DIRNAME=.
|
||||
set APP_BASE_NAME=%~n0
|
||||
set APP_HOME=%DIRNAME%
|
||||
|
||||
@rem Find java.exe
|
||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||
|
||||
set JAVA_EXE=java.exe
|
||||
%JAVA_EXE% -version >NUL 2>&1
|
||||
if "%ERRORLEVEL%" == "0" goto init
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:findJavaFromJavaHome
|
||||
set JAVA_HOME=%JAVA_HOME:"=%
|
||||
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||
|
||||
if exist "%JAVA_EXE%" goto init
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:init
|
||||
@rem Get command-line arguments, handling Windowz variants
|
||||
|
||||
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||
if "%@eval[2+2]" == "4" goto 4NT_args
|
||||
|
||||
:win9xME_args
|
||||
@rem Slurp the command line arguments.
|
||||
set CMD_LINE_ARGS=
|
||||
set _SKIP=2
|
||||
|
||||
:win9xME_args_slurp
|
||||
if "x%~1" == "x" goto execute
|
||||
|
||||
set CMD_LINE_ARGS=%*
|
||||
goto execute
|
||||
|
||||
:4NT_args
|
||||
@rem Get arguments from the 4NT Shell from JP Software
|
||||
set CMD_LINE_ARGS=%$
|
||||
|
||||
:execute
|
||||
@rem Setup the command line
|
||||
|
||||
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||
|
||||
@rem Execute Gradle
|
||||
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
||||
|
||||
:end
|
||||
@rem End local scope for the variables with windows NT shell
|
||||
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||
|
||||
:fail
|
||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||
rem the _cmd.exe /c_ return code!
|
||||
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||
exit /b 1
|
||||
|
||||
:mainEnd
|
||||
if "%OS%"=="Windows_NT" endlocal
|
||||
|
||||
:omega
|
1
wa/workloads/youtube/uiauto/settings.gradle
Normal file
1
wa/workloads/youtube/uiauto/settings.gradle
Normal file
@ -0,0 +1 @@
|
||||
include ':app'
|
Loading…
x
Reference in New Issue
Block a user