mirror of
https://github.com/ARM-software/workload-automation.git
synced 2025-02-22 04:49:00 +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):
|
def __init__(self):
|
||||||
super(SubClass, self).__init__()
|
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):
|
class SubSubClass(SubClass):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super(SubSubClass, self).__init__()
|
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):
|
class AnotherClass(object):
|
||||||
|
|
||||||
@ -262,8 +292,8 @@ class OncePerInstanceEnvironmentTest(TestCase):
|
|||||||
|
|
||||||
sc.initilize_once_per_instance()
|
sc.initilize_once_per_instance()
|
||||||
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()
|
||||||
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)
|
cfg_point.set_value(final_config, value, check_mandatory=False)
|
||||||
|
|
||||||
if state.generic_config[source]:
|
if state.generic_config[source]:
|
||||||
msg = 'Unexected values for {}: {}'
|
msg = 'Unexpected values for {}: {}'
|
||||||
raise ConfigError(msg.format(state.generic_name,
|
raise ConfigError(msg.format(state.generic_name,
|
||||||
state.generic_config[source]))
|
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)
|
cfg_point.set_value(final_config, value, check_mandatory=False)
|
||||||
|
|
||||||
if state.specific_config[source]:
|
if state.specific_config[source]:
|
||||||
msg = 'Unexected values for {}: {}'
|
msg = 'Unexpected values for {}: {}'
|
||||||
raise ConfigError(msg.format(state.specific_name,
|
raise ConfigError(msg.format(state.specific_name,
|
||||||
state.specific_config[source]))
|
state.specific_config[source]))
|
||||||
|
@ -6,6 +6,8 @@ from wa.framework.configuration.core import Status
|
|||||||
|
|
||||||
class Job(object):
|
class Job(object):
|
||||||
|
|
||||||
|
_workload_cache = {}
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def id(self):
|
def id(self):
|
||||||
return self.spec.id
|
return self.spec.id
|
||||||
@ -40,11 +42,15 @@ class Job(object):
|
|||||||
|
|
||||||
def load(self, target, loader=pluginloader):
|
def load(self, target, loader=pluginloader):
|
||||||
self.logger.info('Loading job {}'.format(self.id))
|
self.logger.info('Loading job {}'.format(self.id))
|
||||||
self.workload = loader.get_workload(self.spec.workload_name,
|
if self.iteration == 1:
|
||||||
target,
|
self.workload = loader.get_workload(self.spec.workload_name,
|
||||||
**self.spec.workload_parameters)
|
target,
|
||||||
self.workload.init_resources(self.context)
|
**self.spec.workload_parameters)
|
||||||
self.workload.validate()
|
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):
|
def initialize(self, context):
|
||||||
self.logger.info('Initializing job {}'.format(self.id))
|
self.logger.info('Initializing job {}'.format(self.id))
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
<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"/>
|
<uses-permission android:name="android.permission.READ_LOGS"/>
|
||||||
|
|
||||||
|
@ -38,6 +38,14 @@ import java.util.concurrent.TimeoutException;
|
|||||||
|
|
||||||
public class BaseUiAutomation {
|
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 Instrumentation mInstrumentation;
|
||||||
public Context mContext;
|
public Context mContext;
|
||||||
public UiDevice mDevice;
|
public UiDevice mDevice;
|
||||||
@ -140,4 +148,218 @@ public class BaseUiAutomation {
|
|||||||
throw new TimeoutException("Timed out waiting for Logcat text \"%s\"".format(searchText));
|
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,
|
from wa.framework.resource import (ApkFile, JarFile, ReventFile, NO_ONE,
|
||||||
Executable, File)
|
Executable, File)
|
||||||
from wa.framework.exception import WorkloadError
|
from wa.framework.exception import WorkloadError
|
||||||
|
from wa.utils.types import ParameterDict
|
||||||
from wa.utils.revent import ReventRecorder
|
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.utils.android import ApkInfo
|
||||||
from devlib.exception import TargetError
|
from devlib.exception import TargetError
|
||||||
@ -160,6 +161,7 @@ class ApkWorkload(Workload):
|
|||||||
def init_resources(self, context):
|
def init_resources(self, context):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@once_per_instance
|
||||||
def initialize(self, context):
|
def initialize(self, context):
|
||||||
self.apk.initialize(context)
|
self.apk.initialize(context)
|
||||||
|
|
||||||
@ -176,7 +178,7 @@ class ApkWorkload(Workload):
|
|||||||
def teardown(self, context):
|
def teardown(self, context):
|
||||||
self.apk.teardown()
|
self.apk.teardown()
|
||||||
|
|
||||||
@once
|
@once_per_instance
|
||||||
def finalize(self, context):
|
def finalize(self, context):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@ -190,14 +192,14 @@ class ApkUIWorkload(ApkWorkload):
|
|||||||
def init_resources(self, context):
|
def init_resources(self, context):
|
||||||
super(ApkUIWorkload, self).init_resources(context)
|
super(ApkUIWorkload, self).init_resources(context)
|
||||||
self.gui.init_resources(context.resolver)
|
self.gui.init_resources(context.resolver)
|
||||||
self.gui.init_commands()
|
|
||||||
|
|
||||||
|
@once_per_instance
|
||||||
def initialize(self, context):
|
def initialize(self, context):
|
||||||
super(ApkUIWorkload, self).initialize(context)
|
super(ApkUIWorkload, self).initialize(context)
|
||||||
self.gui.deploy()
|
|
||||||
|
|
||||||
def setup(self, context):
|
def setup(self, context):
|
||||||
super(ApkUIWorkload, self).setup(context)
|
super(ApkUIWorkload, self).setup(context)
|
||||||
|
self.gui.deploy()
|
||||||
self.gui.setup()
|
self.gui.setup()
|
||||||
|
|
||||||
def run(self, context):
|
def run(self, context):
|
||||||
@ -212,7 +214,7 @@ class ApkUIWorkload(ApkWorkload):
|
|||||||
self.gui.teardown()
|
self.gui.teardown()
|
||||||
super(ApkUIWorkload, self).teardown(context)
|
super(ApkUIWorkload, self).teardown(context)
|
||||||
|
|
||||||
@once
|
@once_per_instance
|
||||||
def finalize(self, context):
|
def finalize(self, context):
|
||||||
super(ApkUIWorkload, self).finalize(context)
|
super(ApkUIWorkload, self).finalize(context)
|
||||||
self.gui.remove()
|
self.gui.remove()
|
||||||
@ -226,6 +228,11 @@ class ApkUiautoWorkload(ApkUIWorkload):
|
|||||||
super(ApkUiautoWorkload, self).__init__(target, **kwargs)
|
super(ApkUiautoWorkload, self).__init__(target, **kwargs)
|
||||||
self.gui = UiAutomatorGUI(self)
|
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):
|
class ReventWorkload(ApkUIWorkload):
|
||||||
|
|
||||||
@ -261,7 +268,7 @@ class UiAutomatorGUI(object):
|
|||||||
self.logger = logging.getLogger('gui')
|
self.logger = logging.getLogger('gui')
|
||||||
self.uiauto_file = None
|
self.uiauto_file = None
|
||||||
self.commands = {}
|
self.commands = {}
|
||||||
self.uiauto_params = {}
|
self.uiauto_params = ParameterDict()
|
||||||
|
|
||||||
def init_resources(self, resolver):
|
def init_resources(self, resolver):
|
||||||
self.uiauto_file = resolver.get(ApkFile(self.owner, uiauto=True))
|
self.uiauto_file = resolver.get(ApkFile(self.owner, uiauto=True))
|
||||||
@ -273,7 +280,7 @@ class UiAutomatorGUI(object):
|
|||||||
params_dict = self.uiauto_params
|
params_dict = self.uiauto_params
|
||||||
params_dict['workdir'] = self.target.working_directory
|
params_dict['workdir'] = self.target.working_directory
|
||||||
params = ''
|
params = ''
|
||||||
for k, v in self.uiauto_params.iteritems():
|
for k, v in params_dict.iter_encoded_items():
|
||||||
params += ' -e {} {}'.format(k, v)
|
params += ' -e {} {}'.format(k, v)
|
||||||
|
|
||||||
for stage in self.stages:
|
for stage in self.stages:
|
||||||
@ -286,7 +293,9 @@ class UiAutomatorGUI(object):
|
|||||||
instrumentation_string)
|
instrumentation_string)
|
||||||
|
|
||||||
def deploy(self):
|
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):
|
def set(self, name, value):
|
||||||
self.uiauto_params[name] = value
|
self.uiauto_params[name] = value
|
||||||
@ -395,9 +404,6 @@ class ReventGUI(object):
|
|||||||
def remove(self):
|
def remove(self):
|
||||||
self.revent_recorder.remove()
|
self.revent_recorder.remove()
|
||||||
|
|
||||||
def init_commands(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _check_revent_files(self):
|
def _check_revent_files(self):
|
||||||
if not self.revent_run_file:
|
if not self.revent_run_file:
|
||||||
# pylint: disable=too-few-format-args
|
# pylint: disable=too-few-format-args
|
||||||
@ -435,9 +441,9 @@ class PackageHandler(object):
|
|||||||
|
|
||||||
def initialize(self, context):
|
def initialize(self, context):
|
||||||
self.resolve_package(context)
|
self.resolve_package(context)
|
||||||
self.initialize_package(context)
|
|
||||||
|
|
||||||
def setup(self, context):
|
def setup(self, context):
|
||||||
|
self.initialize_package(context)
|
||||||
self.start_activity()
|
self.start_activity()
|
||||||
self.target.execute('am kill-all') # kill all *background* activities
|
self.target.execute('am kill-all') # kill all *background* activities
|
||||||
self.target.clear_logcat()
|
self.target.clear_logcat()
|
||||||
@ -514,9 +520,12 @@ class PackageHandler(object):
|
|||||||
self.apk_version = host_version
|
self.apk_version = host_version
|
||||||
|
|
||||||
def start_activity(self):
|
def start_activity(self):
|
||||||
cmd = 'am start -W -n {}/{}'
|
if not self.apk_info.activity:
|
||||||
output = self.target.execute(cmd.format(self.apk_info.package,
|
cmd = 'am start -W {}'.format(self.apk_info.package)
|
||||||
self.apk_info.activity))
|
else:
|
||||||
|
cmd = 'am start -W -n {}/{}'.format(self.apk_info.package,
|
||||||
|
self.apk_info.activity)
|
||||||
|
output = self.target.execute(cmd)
|
||||||
if 'Error:' in output:
|
if 'Error:' in output:
|
||||||
# this will dismiss any error dialogs
|
# this will dismiss any error dialogs
|
||||||
self.target.execute('am force-stop {}'.format(self.apk_info.package))
|
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))
|
self.target.execute('pm clear {}'.format(self.apk_info.package))
|
||||||
|
|
||||||
def install_apk(self, context):
|
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 'Failure' in output:
|
||||||
if 'ALREADY_EXISTS' in output:
|
if 'ALREADY_EXISTS' in output:
|
||||||
msg = 'Using already installed APK (did not unistall properly?)'
|
msg = 'Using already installed APK (did not unistall properly?)'
|
||||||
|
@ -59,7 +59,7 @@ def once_per_instance(method):
|
|||||||
def wrapper(*args, **kwargs):
|
def wrapper(*args, **kwargs):
|
||||||
if __active_environment is None:
|
if __active_environment is None:
|
||||||
activate_environment('default')
|
activate_environment('default')
|
||||||
func_id = repr(args[0])
|
func_id = repr(method.__hash__()) + repr(args[0])
|
||||||
if func_id in __environments[__active_environment]:
|
if func_id in __environments[__active_environment]:
|
||||||
return
|
return
|
||||||
else:
|
else:
|
||||||
|
@ -32,6 +32,7 @@ import numbers
|
|||||||
import shlex
|
import shlex
|
||||||
import string
|
import string
|
||||||
from bisect import insort
|
from bisect import insort
|
||||||
|
from urllib import quote, unquote
|
||||||
from collections import defaultdict, MutableMapping
|
from collections import defaultdict, MutableMapping
|
||||||
from copy import copy
|
from copy import copy
|
||||||
|
|
||||||
@ -586,3 +587,118 @@ def enum(args, start=0, step=1):
|
|||||||
|
|
||||||
return Enum
|
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"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
<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:versionCode="1"
|
||||||
android:versionName="1.0">
|
android:versionName="1.0">
|
||||||
|
|
||||||
|
@ -33,6 +33,9 @@ public class UiAutomation extends BaseUiAutomation {
|
|||||||
|
|
||||||
public static String TAG = "benchmarkpi";
|
public static String TAG = "benchmarkpi";
|
||||||
|
|
||||||
|
public Bundle parameters;
|
||||||
|
public String packageID;
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void runWorkload() throws Exception {
|
public void runWorkload() throws Exception {
|
||||||
startTest();
|
startTest();
|
||||||
@ -43,7 +46,7 @@ public class UiAutomation extends BaseUiAutomation {
|
|||||||
public void extractResults() throws Exception {
|
public void extractResults() throws Exception {
|
||||||
UiSelector selector = new UiSelector();
|
UiSelector selector = new UiSelector();
|
||||||
UiObject resultsText = mDevice.findObject(selector.textContains("You calculated Pi in")
|
UiObject resultsText = mDevice.findObject(selector.textContains("You calculated Pi in")
|
||||||
.className("android.widget.TextView"));
|
.className("android.widget.TextView"));
|
||||||
Log.v(TAG, resultsText.getText());
|
Log.v(TAG, resultsText.getText());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@ if [[ ! -f gradlew ]]; then
|
|||||||
exit 9
|
exit 9
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Copy base class library from wlauto dist
|
# Copy base class library from wa dist
|
||||||
libs_dir=app/libs
|
libs_dir=app/libs
|
||||||
base_class=`python -c "import os, wa; print os.path.join(os.path.dirname(wa.__file__), 'framework', 'uiauto', 'uiauto.aar')"`
|
base_class=`python -c "import os, wa; print os.path.join(os.path.dirname(wa.__file__), 'framework', 'uiauto', 'uiauto.aar')"`
|
||||||
mkdir -p $libs_dir
|
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