mirror of
https://github.com/ARM-software/workload-automation.git
synced 2025-04-15 07:10:56 +01:00
Gmail productivity workload
A new workload to perform standard productivity tasks within Gmail. Each user event/step is timed and reported back as a metric.
This commit is contained in:
parent
d01f338a1d
commit
cc2280a317
78
wlauto/workloads/gmail/__init__.py
Executable file
78
wlauto/workloads/gmail/__init__.py
Executable file
@ -0,0 +1,78 @@
|
|||||||
|
import os
|
||||||
|
import logging
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
|
||||||
|
from wlauto import AndroidUiAutoBenchmark, Parameter
|
||||||
|
|
||||||
|
|
||||||
|
class Gmail(AndroidUiAutoBenchmark):
|
||||||
|
|
||||||
|
name = 'gmail'
|
||||||
|
package = 'com.google.android.gm'
|
||||||
|
activity = ''
|
||||||
|
view = [package+'/com.google.android.gm.ConversationListActivityGmail',
|
||||||
|
package+'/com.google.android.gm.ComposeActivityGmail']
|
||||||
|
description = """
|
||||||
|
A workload to perform standard productivity tasks within Gmail.
|
||||||
|
|
||||||
|
The workload carries out various tasks, such as creatign new emails and
|
||||||
|
sending them, whilst also producing metrics for action completion times.
|
||||||
|
"""
|
||||||
|
|
||||||
|
regex = re.compile(r'uxperf_gmail.*: (?P<key>\w+) (?P<value>\d+)')
|
||||||
|
|
||||||
|
parameters = [
|
||||||
|
Parameter('recipient', default='armuxperf@gmail.com', mandatory=False,
|
||||||
|
description=""""
|
||||||
|
The email address of the recipient. Setting a void address
|
||||||
|
will stop any mesage failures clogging up your device inbox
|
||||||
|
"""),
|
||||||
|
Parameter('dumpsys_enabled', kind=bool, default=True,
|
||||||
|
description="""
|
||||||
|
If ``True``, dumpsys captures will be carried out during the
|
||||||
|
test run. The output is piped to log files which are then
|
||||||
|
pulled from the phone.
|
||||||
|
"""),
|
||||||
|
]
|
||||||
|
|
||||||
|
instrumentation_log = ''.join([name, '_instrumentation.log'])
|
||||||
|
|
||||||
|
def __init__(self, device, **kwargs):
|
||||||
|
super(Gmail, self).__init__(device, **kwargs)
|
||||||
|
self.uiauto_params['recipient'] = self.recipient
|
||||||
|
|
||||||
|
def validate(self):
|
||||||
|
super(Gmail, self).validate()
|
||||||
|
self.output_file = os.path.join(self.device.working_directory, self.instrumentation_log)
|
||||||
|
self.uiauto_params['package'] = self.package
|
||||||
|
self.uiauto_params['output_dir'] = self.device.working_directory
|
||||||
|
self.uiauto_params['output_file'] = self.output_file
|
||||||
|
self.uiauto_params['dumpsys_enabled'] = self.dumpsys_enabled
|
||||||
|
|
||||||
|
def update_result(self, context):
|
||||||
|
super(Gmail, self).update_result(context)
|
||||||
|
|
||||||
|
if self.dumpsys_enabled:
|
||||||
|
self.device.pull_file(self.output_file, context.output_directory)
|
||||||
|
result_file = os.path.join(context.output_directory, self.instrumentation_log)
|
||||||
|
|
||||||
|
with open(result_file, 'r') as wfh:
|
||||||
|
regex = re.compile(r'(?P<key>\w+)\s+(?P<value1>\d+)\s+(?P<value2>\d+)\s+(?P<value3>\d+)')
|
||||||
|
for line in wfh:
|
||||||
|
match = regex.search(line)
|
||||||
|
if match:
|
||||||
|
context.result.add_metric((match.group('key') + "_start"),
|
||||||
|
match.group('value1'))
|
||||||
|
context.result.add_metric((match.group('key') + "_finish"),
|
||||||
|
match.group('value2'))
|
||||||
|
context.result.add_metric((match.group('key') + "_duration"),
|
||||||
|
match.group('value3'))
|
||||||
|
|
||||||
|
def teardown(self, context):
|
||||||
|
super(Gmail, self).teardown(context)
|
||||||
|
|
||||||
|
for file in self.device.listdir(self.device.working_directory):
|
||||||
|
if file.startswith (self.name) and file.endswith(".log"):
|
||||||
|
self.device.pull_file(os.path.join(self.device.working_directory, file), context.output_directory)
|
||||||
|
self.device.delete_file(os.path.join(self.device.working_directory, file))
|
BIN
wlauto/workloads/gmail/com.arm.wlauto.uiauto.gmail.jar
Normal file
BIN
wlauto/workloads/gmail/com.arm.wlauto.uiauto.gmail.jar
Normal file
Binary file not shown.
12
wlauto/workloads/gmail/uiauto/build.sh
Executable file
12
wlauto/workloads/gmail/uiauto/build.sh
Executable file
@ -0,0 +1,12 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
class_dir=bin/classes/com/arm/wlauto/uiauto
|
||||||
|
base_classes=`python -c "import os, wlauto; print os.path.join(os.path.dirname(wlauto.__file__), 'common', 'android', '*.class')"`
|
||||||
|
mkdir -p $class_dir
|
||||||
|
cp $base_classes $class_dir
|
||||||
|
|
||||||
|
ant build
|
||||||
|
|
||||||
|
if [[ -f bin/com.arm.wlauto.uiauto.gmail.jar ]]; then
|
||||||
|
cp bin/com.arm.wlauto.uiauto.gmail.jar ..
|
||||||
|
fi
|
92
wlauto/workloads/gmail/uiauto/build.xml
Normal file
92
wlauto/workloads/gmail/uiauto/build.xml
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project name="com.arm.wlauto.uiauto.gmail" default="help">
|
||||||
|
|
||||||
|
<!-- The local.properties file is created and updated by the 'android' tool.
|
||||||
|
It contains the path to the SDK. It should *NOT* be checked into
|
||||||
|
Version Control Systems. -->
|
||||||
|
<property file="local.properties" />
|
||||||
|
|
||||||
|
<!-- The ant.properties file can be created by you. It is only edited by the
|
||||||
|
'android' tool to add properties to it.
|
||||||
|
This is the place to change some Ant specific build properties.
|
||||||
|
Here are some properties you may want to change/update:
|
||||||
|
|
||||||
|
source.dir
|
||||||
|
The name of the source directory. Default is 'src'.
|
||||||
|
out.dir
|
||||||
|
The name of the output directory. Default is 'bin'.
|
||||||
|
|
||||||
|
For other overridable properties, look at the beginning of the rules
|
||||||
|
files in the SDK, at tools/ant/build.xml
|
||||||
|
|
||||||
|
Properties related to the SDK location or the project target should
|
||||||
|
be updated using the 'android' tool with the 'update' action.
|
||||||
|
|
||||||
|
This file is an integral part of the build system for your
|
||||||
|
application and should be checked into Version Control Systems.
|
||||||
|
|
||||||
|
-->
|
||||||
|
<property file="ant.properties" />
|
||||||
|
|
||||||
|
<!-- if sdk.dir was not set from one of the property file, then
|
||||||
|
get it from the ANDROID_HOME env var.
|
||||||
|
This must be done before we load project.properties since
|
||||||
|
the proguard config can use sdk.dir -->
|
||||||
|
<property environment="env" />
|
||||||
|
<condition property="sdk.dir" value="${env.ANDROID_HOME}">
|
||||||
|
<isset property="env.ANDROID_HOME" />
|
||||||
|
</condition>
|
||||||
|
|
||||||
|
<!-- The project.properties file is created and updated by the 'android'
|
||||||
|
tool, as well as ADT.
|
||||||
|
|
||||||
|
This contains project specific properties such as project target, and library
|
||||||
|
dependencies. Lower level build properties are stored in ant.properties
|
||||||
|
(or in .classpath for Eclipse projects).
|
||||||
|
|
||||||
|
This file is an integral part of the build system for your
|
||||||
|
application and should be checked into Version Control Systems. -->
|
||||||
|
<loadproperties srcFile="project.properties" />
|
||||||
|
|
||||||
|
<!-- quick check on sdk.dir -->
|
||||||
|
<fail
|
||||||
|
message="sdk.dir is missing. Make sure to generate local.properties using 'android update project' or to inject it through the ANDROID_HOME environment variable."
|
||||||
|
unless="sdk.dir"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
Import per project custom build rules if present at the root of the project.
|
||||||
|
This is the place to put custom intermediary targets such as:
|
||||||
|
-pre-build
|
||||||
|
-pre-compile
|
||||||
|
-post-compile (This is typically used for code obfuscation.
|
||||||
|
Compiled code location: ${out.classes.absolute.dir}
|
||||||
|
If this is not done in place, override ${out.dex.input.absolute.dir})
|
||||||
|
-post-package
|
||||||
|
-post-build
|
||||||
|
-pre-clean
|
||||||
|
-->
|
||||||
|
<import file="custom_rules.xml" optional="true" />
|
||||||
|
|
||||||
|
<!-- Import the actual build file.
|
||||||
|
|
||||||
|
To customize existing targets, there are two options:
|
||||||
|
- Customize only one target:
|
||||||
|
- copy/paste the target into this file, *before* the
|
||||||
|
<import> task.
|
||||||
|
- customize it to your needs.
|
||||||
|
- Customize the whole content of build.xml
|
||||||
|
- copy/paste the content of the rules files (minus the top node)
|
||||||
|
into this file, replacing the <import> task.
|
||||||
|
- customize to your needs.
|
||||||
|
|
||||||
|
***********************
|
||||||
|
****** IMPORTANT ******
|
||||||
|
***********************
|
||||||
|
In all cases you must update the value of version-tag below to read 'custom' instead of an integer,
|
||||||
|
in order to avoid having your file be overridden by tools such as "android update project"
|
||||||
|
-->
|
||||||
|
<!-- version-tag: VERSION_TAG -->
|
||||||
|
<import file="${sdk.dir}/tools/ant/uibuild.xml" />
|
||||||
|
|
||||||
|
</project>
|
14
wlauto/workloads/gmail/uiauto/project.properties
Normal file
14
wlauto/workloads/gmail/uiauto/project.properties
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# This file is automatically generated by Android Tools.
|
||||||
|
# Do not modify this file -- YOUR CHANGES WILL BE ERASED!
|
||||||
|
#
|
||||||
|
# This file must be checked in Version Control Systems.
|
||||||
|
#
|
||||||
|
# To customize properties used by the Ant build system edit
|
||||||
|
# "ant.properties", and override values to adapt the script to your
|
||||||
|
# project structure.
|
||||||
|
#
|
||||||
|
# To enable ProGuard to shrink and obfuscate your code, uncomment this (available properties: sdk.dir, user.home):
|
||||||
|
#proguard.config=${sdk.dir}/tools/proguard/proguard-android.txt:proguard-project.txt
|
||||||
|
|
||||||
|
# Project target.
|
||||||
|
target=android-18
|
@ -0,0 +1,123 @@
|
|||||||
|
package com.arm.wlauto.uiauto.gmail;
|
||||||
|
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.os.SystemClock;
|
||||||
|
|
||||||
|
// Import the uiautomator libraries
|
||||||
|
import com.android.uiautomator.core.UiObject;
|
||||||
|
import com.android.uiautomator.core.UiObjectNotFoundException;
|
||||||
|
import com.android.uiautomator.core.UiScrollable;
|
||||||
|
import com.android.uiautomator.core.UiSelector;
|
||||||
|
import com.android.uiautomator.testrunner.UiAutomatorTestCase;
|
||||||
|
|
||||||
|
import com.arm.wlauto.uiauto.UxPerfUiAutomation;
|
||||||
|
|
||||||
|
import java.io.BufferedWriter;
|
||||||
|
import java.io.FileWriter;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Map.Entry;
|
||||||
|
|
||||||
|
public class UiAutomation extends UxPerfUiAutomation {
|
||||||
|
|
||||||
|
public static String TAG = "uxperf_gmail";
|
||||||
|
|
||||||
|
private Bundle parameters;
|
||||||
|
private LinkedHashMap<String, Timer> timingResults = new LinkedHashMap<String, Timer>();
|
||||||
|
|
||||||
|
public void runUiAutomation() throws Exception {
|
||||||
|
parameters = getParams();
|
||||||
|
|
||||||
|
Timer result = new Timer();
|
||||||
|
result.start();
|
||||||
|
|
||||||
|
clearFirstRunDialogues();
|
||||||
|
|
||||||
|
clickNewMail();
|
||||||
|
setToField();
|
||||||
|
setSubjectField();
|
||||||
|
setComposeField();
|
||||||
|
clickSendButton();
|
||||||
|
|
||||||
|
result.end();
|
||||||
|
timingResults.put("Total", result);
|
||||||
|
|
||||||
|
writeResultsToFile(timingResults, parameters.getString("output_file"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clearFirstRunDialogues () throws Exception {
|
||||||
|
// Enter search text into the file searchBox. This will automatically filter the list.
|
||||||
|
UiObject gotItBox = getUiObjectByResourceId("com.google.android.gm:id/welcome_tour_got_it",
|
||||||
|
"android.widget.TextView");
|
||||||
|
gotItBox.clickAndWaitForNewWindow();
|
||||||
|
UiObject takeMeToBox = getUiObjectByText("Take me to Gmail", "android.widget.TextView");
|
||||||
|
takeMeToBox.clickAndWaitForNewWindow();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clickNewMail() throws Exception {
|
||||||
|
Timer result = new Timer();
|
||||||
|
UiObject newMailButton = getUiObjectByDescription("Compose", "android.widget.ImageButton");
|
||||||
|
result.start();
|
||||||
|
newMailButton.clickAndWaitForNewWindow(timeout);
|
||||||
|
result.end();
|
||||||
|
timingResults.put("newMail", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setToField() throws Exception {
|
||||||
|
Timer result = new Timer();
|
||||||
|
UiObject toField = getUiObjectByDescription("To", "android.widget.TextView");
|
||||||
|
String recipient = parameters.getString("recipient").replace('_', ' ');
|
||||||
|
result.start();
|
||||||
|
toField.setText(recipient);
|
||||||
|
getUiDevice().pressEnter();
|
||||||
|
result.end();
|
||||||
|
timingResults.put("To", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSubjectField() throws Exception {
|
||||||
|
Timer result = new Timer();
|
||||||
|
UiObject subjectField = getUiObjectByText("Subject", "android.widget.EditText");
|
||||||
|
result.start();
|
||||||
|
subjectField.setText("This is a test message");
|
||||||
|
getUiDevice().pressEnter();
|
||||||
|
result.end();
|
||||||
|
timingResults.put("Subject", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setComposeField() throws Exception {
|
||||||
|
Timer result = new Timer();
|
||||||
|
UiObject composeField = getUiObjectByText("Compose email", "android.widget.EditText");
|
||||||
|
result.start();
|
||||||
|
composeField.setText("This is a test composition");
|
||||||
|
getUiDevice().pressEnter();
|
||||||
|
result.end();
|
||||||
|
timingResults.put("Compose", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clickSendButton() throws Exception {
|
||||||
|
Timer result = new Timer();
|
||||||
|
UiObject sendButton = getUiObjectByDescription("Send", "android.widget.TextView");
|
||||||
|
result.start();
|
||||||
|
sendButton.clickAndWaitForNewWindow(timeout);
|
||||||
|
result.end();
|
||||||
|
timingResults.put("Send", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void writeResultsToFile(LinkedHashMap timingResults, String file) throws Exception {
|
||||||
|
// Write out the key/value pairs to the instrumentation log file
|
||||||
|
FileWriter fstream = new FileWriter(file);
|
||||||
|
BufferedWriter out = new BufferedWriter(fstream);
|
||||||
|
Iterator<Entry<String, Timer>> it = timingResults.entrySet().iterator();
|
||||||
|
|
||||||
|
while (it.hasNext()) {
|
||||||
|
Map.Entry<String, Timer> pairs = it.next();
|
||||||
|
Timer results = pairs.getValue();
|
||||||
|
long start = results.getStart();
|
||||||
|
long finish = results.getFinish();
|
||||||
|
long duration = results.getDuration();
|
||||||
|
out.write(String.format(pairs.getKey() + " " + start + " " + finish + " " + duration + "\n"));
|
||||||
|
}
|
||||||
|
out.close();
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user