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