Home | History | Annotate | Download | only in tradefed
      1 page.title=End-to-End Test Example
      2 @jd:body
      3 
      4 <!--
      5     Copyright 2010 The Android Open Source Project
      6 
      7     Licensed under the Apache License, Version 2.0 (the "License");
      8     you may not use this file except in compliance with the License.
      9     You may obtain a copy of the License at
     10 
     11         http://www.apache.org/licenses/LICENSE-2.0
     12 
     13     Unless required by applicable law or agreed to in writing, software
     14     distributed under the License is distributed on an "AS IS" BASIS,
     15     WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     16     See the License for the specific language governing permissions and
     17     limitations under the License.
     18 -->
     19 
     20 <p>This tutorial guides you through the construction of a "hello world" Trade Federation test
     21 configuration, and gives you a hands-on introduction to the Trade Federation framework.  Starting
     22 from the TF development environment, it guides you through the process of creating a simple Trade
     23 Federation config and gradually adding more features to it.</p>
     24 
     25 <p>The tutorial presents the TF test development process as a set of exercises, each consisting of
     26 several steps.  The exercises demonstrate how to gradually build and refine your configuration, and
     27 provide all the sample code you need to complete the test configuration.  The title of each
     28 exercise is annotated with a letter describing which roles are involved in that step: <b>D</b> for
     29 Developer, <b>I</b> for Integrator, and/or <b>R</b> for Test Runner.</p>
     30 
     31 <p>When you are finished with the tutorial, you will have created a functioning TF configuration and
     32 will have learned many of the most important concepts in the TF framework.</p>
     33 
     34 <h2>Set up TradeFederation development environment</h2>
     35 <p>See the <a href="/devices/tech/test_infra/tradefed/fundamentals/machine_setup.html"
     36 >Machine Setup</a> page for how to setup the development environment. The rest of this tutorial
     37 assumes you have a shell open that has been initialized to the Trade Federation environment.</p>
     38 
     39 <p>For simplicity, this tutorial will illustrate adding a configuration and its classes to the
     40 Trade Federation framework core library.  This can be extended to developing modules outside the
     41 source tree by simply compiling the tradefed JAR, and compiling your modules against that JAR.</p>
     42 
     43 <h2>Creating a test class (D)</h2>
     44 <p>Lets create a hello world test that just dumps a message to stdout. A tradefed test will
     45 generally implement the <a href="/reference/com/android/tradefed/testtype/IRemoteTest.html"
     46 >IRemoteTest</a> interface.</p>
     47 
     48 <p>Here's an implementation for the HelloWorldTest:</p>
     49 <pre><code>package com.android.tradefed.example;
     50 
     51 import com.android.tradefed.device.DeviceNotAvailableException;
     52 import com.android.tradefed.result.ITestInvocationListener;
     53 import com.android.tradefed.testtype.IRemoteTest;
     54 
     55 public class HelloWorldTest implements IRemoteTest {
     56     &#64;Override
     57     public void run(ITestInvocationListener listener) throws DeviceNotAvailableException {
     58         System.out.println("Hello, TF World!");
     59     }
     60 }
     61 </code></pre>
     62 
     63 <p>Save this sample code to
     64 <code>&lt;tree&gt;/tools/tradefederation/prod-tests/src/com/android/tradefed/example/HelloWorldTest.java</code>
     65 and rebuild tradefed from your shell:</p>
     66 <pre><code>m -jN</code></pre>
     67 
     68 <p>If the build does not succeed, consult the
     69 <a href="/devices/tech/test_infra/tradefed/fundamentals/machine_setup.html">Machine Setup</a> page
     70 to ensure that you didn't miss any steps.</p>
     71 
     72 <h2>Creating a Configuration (I)</h2>
     73 <p>Trade Federation tests are made executable by creating a <b>Configuration</b>, which is an XML file
     74 that instructs tradefed on which test (or tests) to run, as well as which other modules to
     75 execute, and in what order.</p>
     76 
     77 <p>Lets create a new Configuration for our HelloWorldTest:</p>
     78 <pre><code>&lt;configuration description="Runs the hello world test"&gt;
     79     &lt;test class="com.android.tradefed.example.HelloWorldTest" /&gt;
     80 &lt;/configuration&gt;</code></pre>
     81 
     82 <p>TF will parse the Configuration XML file (aka <b>config</b>), load the specified class using
     83 reflection, instantiate it, cast it to a <code>IRemoteTest</code>, and call its <code>run</code>
     84 method.</p>
     85 
     86 <p>Note that we've specified the full class name of the HelloWorldTest. Save this data to a
     87 <code>helloworld.xml</code> file anywhere on your local filesystem (eg <code>/tmp/helloworld.xml</code>).</p>
     88 
     89 <h2>Running the config (R)</h2>
     90 <p>From your shell, launch the tradefed console</p>
     91 <pre><code>$ tradefed.sh
     92 </code></pre>
     93 
     94 <p>Ensure that a device is connected to the host machine and is visible to tradefed</p>
     95 <pre><code>tf &gt;list devices
     96 Serial            State      Product   Variant   Build   Battery  
     97 004ad9880810a548  Available  mako      mako      JDQ39   100
     98 </code></pre>
     99 
    100 <p>Configurations can be executed using the <code>run &lt;config&gt;</code> console command.  Try this:</p>
    101 <p>FIXME: redo this</p>
    102 <pre><code>tf&gt; run /tmp/helloworld.xml
    103 05-12 13:19:36 I/TestInvocation: Starting invocation for target stub on build 0 on device 004ad9880810a548
    104 Hello, TF World!
    105 </code></pre>
    106 <p>You should see "Hello, TF World!" outputted on the terminal.</p>
    107 
    108 <h2>Adding the config to the Classpath (D, I, R)</h2>
    109 <p>For convenience of deployment, you can also bundle configs into the tradefed jars
    110 themselves. Tradefed will automatically recognize all configurations placed in 'config' folders on
    111 the classpath.</p>
    112 
    113 <p>Lets illustrate this now by moving the helloworld.xml into the tradefed core library.</p>
    114 <p>Move the <code>helloworld.xml</code> file into 
    115 <code>&lt;tree&gt;/tools/tradefederation/prod-tests/res/config/example/helloworld.xml</code>.</p>
    116 <p>Rebuild tradefed, and restart the tradefed console. </p>
    117 <p>Ask tradefed to display the list of configurations from the classpath:</p>
    118 <pre><code>tf&gt; list configs
    119 []
    120 example/helloworld: Runs the hello world test
    121 </code></pre>
    122 
    123 <p>You can now run the helloworld config via the following command</p>
    124 <pre><code>tf &gt;run example/helloworld
    125 05-12 13:21:21 I/TestInvocation: Starting invocation for target stub on build 0 on device 004ad9880810a548
    126 Hello, TF World!
    127 </code></pre>
    128 
    129 <h2>Interacting with a Device (D, R)</h2>
    130 <p>So far our hello world test isn't doing anything interesting. Tradefed's specialty is running
    131 tests using Android devices, so lets add an Android device to the test.</p>
    132 
    133 <p>Tests can get a reference to an Android device by implementing the
    134 <a href="/reference/com/android/tradefed/testtype/IDeviceTest.html">IDeviceTest</a> interface.</p>
    135 
    136 <p>Here's a sample implementation of what this looks like:</p>
    137 <pre><code>public class HelloWorldTest implements IRemoteTest, IDeviceTest {
    138     private ITestDevice mDevice;
    139     &#64;Override
    140     public void setDevice(ITestDevice device) {
    141         mDevice = device;
    142     }
    143 
    144     &#64;Override
    145     public ITestDevice getDevice() {
    146         return mDevice;
    147     }
    148 
    149 }
    150 </code></pre>
    151 
    152 <p>The Trade Federation framework will inject the <code>ITestDevice</code> reference into your
    153 test via the <code>IDeviceTest#setDevice</code> method, before the <code>IRemoteTest#run</code>
    154 method is called.</p>
    155 
    156 <p>Let's modify the HelloWorldTest print message to display the serial number of the device.</p>
    157 <pre><code>&#64;Override
    158 public void run(ITestInvocationListener listener) throws DeviceNotAvailableException {
    159     System.out.println("Hello, TF World! I have device " + getDevice().getSerialNumber());
    160 }
    161 </code></pre>
    162 
    163 <p>Now rebuild tradefed, and check the list of devices:</p>
    164 <pre><code>$ tradefed.sh
    165 tf &gt;list devices
    166 Serial            State      Product   Variant   Build   Battery  
    167 004ad9880810a548  Available  mako      mako      JDQ39   100
    168 </code></pre>
    169 
    170 <p>Take note of the serial number listed as Available above. That is the device that should be
    171 allocated to HelloWorld.</p>
    172 <pre><code>tf &gt;run example/helloworld
    173 05-12 13:26:18 I/TestInvocation: Starting invocation for target stub on build 0 on device 004ad9880810a548
    174 Hello, TF World! I have device 004ad9880810a548
    175 </code></pre>
    176 
    177 <p>You should see the new print message displaying the serial number of the device.</p>
    178 
    179 <h2>Sending Test Results (D)</h2>
    180 <p><code>IRemoteTest</code>s report results by calling methods on the
    181 <a href="/reference/com/android/tradefed/result/ITestInvocationListener.html"
    182 >ITestInvocationListener</a> instance provided to their <code>#run</code> method.  Note that the
    183 TF framework itself is responsible for reporting the start and end of each Invocation, (via
    184 the <a href="/reference/com/android/tradefed/result/ITestInvocationListener.html#invocationStarted(com.android.tradefed.build.IBuildInfo)"
    185 >ITestInvocationListener#invocationStarted</a> and
    186 <a href="/reference/com/android/tradefed/result/ITestInvocationListener.html#invocationEnded(long)"
    187 >ITestInvocationListener#invocationEnded</a> methods, respectively).</p>
    188 
    189 <p>A <b>test run</b> is a logical collection of tests. To report test results,
    190 <code>IRemoteTest</code>s are responsible
    191 for reporting the start of a test run, the start and end of each test, and the end of the test run.</p>
    192 
    193 <p>Here's what the HelloWorldTest implementation might look like with a single failed test result.</p>
    194 <pre><code>&#64;Override
    195 public void run(ITestInvocationListener listener) throws DeviceNotAvailableException {
    196     System.out.println("Hello, TF World! I have device " + getDevice().getSerialNumber());
    197 
    198     TestIdentifier testId = new TestIdentifier("com.example.TestClassName", "sampleTest");
    199     listener.testRunStarted("helloworldrun", 1);
    200     listener.testStarted(testId);
    201     listener.testFailed(TestFailure.FAILURE, testId, "oh noes, test failed");
    202     listener.testEnded(testId, Collections.emptyMap());
    203     listener.testRunEnded(0, Collections.emptyMap());
    204 }</code></pre>
    205 
    206 <p>Note that Trade Federation also includes several <code>IRemoteTest</code> implementations that
    207 you can reuse instead of writing your own from scratch.  These include, for instance,
    208 <a href="/reference/com/android/tradefed/testtype/InstrumentationTest.html"
    209 >InstrumentationTest</a>, which can run an Android application's tests remotely on an Android
    210 device, parse the results, and forward them to the <code>ITestInvocationListener</code>). See the
    211 <a href="/reference/com/android/tradefed/testtype/package-summary.html">Test Types
    212 documentation</a> for more details.</p>
    213 
    214 <h2>Storing Test Results (I)</h2>
    215 <p>By default, a TF config will use the
    216 <a href="/reference/com/android/tradefed/result/TextResultReporter.html">TextResultReporter</a> as
    217 the test listener implementation.  <code>TextResultReporter</code> will dump the results of an
    218 invocation to stdout. To illustrate, try running the hello-world config from the previous
    219 section:</p>
    220 <pre><code>$ ./tradefed.sh
    221 tf &gt;run example/helloworld
    222 05-16 20:03:15 I/TestInvocation: Starting invocation for target stub on build 0 on device 004ad9880810a548
    223 Hello, TF World! I have device 004ad9880810a548
    224 05-16 20:03:15 I/InvocationToJUnitResultForwarder: run helloworldrun started: 1 tests
    225 Test FAILURE: com.example.TestClassName#sampleTest 
    226  stack: oh noes, test failed 
    227 05-16 20:03:15 I/InvocationToJUnitResultForwarder: run ended 0 ms
    228 </code></pre>
    229 
    230 <p>If you want to store the results of an invocation elsewhere, such as in a file, you need to
    231 specify a custom <code>ITestInvocationListener</code> implementation by using the
    232 <code>result_reporter</code> tag in your configuration.</p>
    233 
    234 <p>Trade Federation includes the
    235 <a href="/reference/com/android/tradefed/result/XmlResultReporter.html">XmlResultReporter</a>
    236 listener, which will write test results to an XML file, in a format similar to that used by the
    237 <em>ant</em> JUnit XML writer.</p>
    238 
    239 <p>Let's specify the result_reporter in the configuration now. Edit the
    240 <code>/res/config/example/helloworld.xml</code> config like this:</p>
    241 <pre><code>&lt;configuration description="Runs the hello world test"&gt;
    242     &lt;test class="com.android.tradefed.example.HelloWorldTest" /&gt;
    243     &lt;result_reporter class="com.android.tradefed.result.XmlResultReporter" /&gt;
    244 &lt;/configuration&gt;
    245 </code></pre>
    246 
    247 <p>Now rebuild tradefed and re-run the hello world sample:</p>
    248 <pre><code>tf &gt;run example/helloworld
    249 05-16 21:07:07 I/TestInvocation: Starting invocation for target stub on build 0 on device 004ad9880810a548
    250 Hello, TF World! I have device 004ad9880810a548
    251 05-16 21:07:07 I/XmlResultReporter: Saved device_logcat log to /tmp/0/inv_2991649128735283633/device_logcat_6999997036887173857.txt
    252 05-16 21:07:07 I/XmlResultReporter: Saved host_log log to /tmp/0/inv_2991649128735283633/host_log_6307746032218561704.txt
    253 05-16 21:07:07 I/XmlResultReporter: XML test result file generated at /tmp/0/inv_2991649128735283633/test_result_536358148261684076.xml. Total tests 1, Failed 1, Error 0
    254 </code></pre>
    255 
    256 <p>Notice the log message stating that an XML file has been generated. The generated file should look like this:</p>
    257 <pre><code>&lt;?xml version='1.0' encoding='UTF-8' ?&gt;
    258 &lt;testsuite name="stub" tests="1" failures="1" errors="0" time="9" timestamp="2011-05-17T04:07:07" hostname="localhost"&gt;
    259   &lt;properties /&gt;
    260   &lt;testcase name="sampleTest" classname="com.example.TestClassName" time="0"&gt;
    261     &lt;failure&gt;oh noes, test failed
    262     &lt;/failure&gt;
    263   &lt;/testcase&gt;
    264 &lt;/testsuite&gt;
    265 </code></pre>
    266 
    267 <p>You can also write your own custom invocation listeners. It just needs to implement the
    268 <a href="/reference/com/android/tradefed/result/ITestInvocationListener.html"
    269 >ITestInvocationListener</a> interface.</p>
    270 
    271 <p>Also note that tradefed supports multiple invocation listeners, meaning that you can send test
    272 results to multiple independent destinations. Just specify multiple
    273 <code>&lt;result_reporter&gt;</code> tags in your config to do this.</p>
    274 
    275 <h2>Logging (D, I, R)</h2>
    276 <p>TradeFederation includes two logging facilities:</p>
    277 <ol>
    278 <li>ability to capture logs from the device (aka device logcat)</li>
    279 <li>ability to record logs from the TradeFederation framework running on the host machine (aka the
    280     host log)</li>
    281 </ol>
    282 <p>Lets focus on #2 for now. Trade Federation's host logs are reported using the
    283 <a href="/reference/com/android/tradefed/log/LogUtil.CLog.html">CLog wrapper</a> for the
    284 ddmlib Log class.</p>
    285 
    286 <p>Let's convert the previous <code>System.out.println</code> call in HelloWorldTest to a
    287 <code>CLog</code> call:</p>
    288 <pre><code>&#64;Override
    289 public void run(ITestInvocationListener listener) throws DeviceNotAvailableException {
    290     CLog.i("Hello, TF World! I have device %s", getDevice().getSerialNumber());
    291 </code></pre>
    292 
    293 <p>Note that <code>CLog</code> handles string interpolation directly, akin to
    294 <code>String.format</code>.  At this point, when you rebuild and rerun TF, you should see the
    295 log message on stdout.</p>
    296 <pre><code>tf&gt; run example/helloworld
    297 
    298 05-16 21:30:46 I/HelloWorldTest: Hello, TF World! I have device 004ad9880810a548
    299 
    300 </code></pre>
    301 
    302 <p>By default, tradefed will
    303 <a href"/reference/com/android/tradefed/log/StdoutLogger.html">output host log messages to
    304 stdout</a>. TF also includes a log implementation that will write messages to a file:
    305 <a href="/reference/com/android/tradefed/log/FileLogger.html">FileLogger</a>. To add file logging,
    306 add a <code>logger</code> tag to the config, specifying the full class name of
    307 <code>FileLogger</code>.</p>
    308 <pre><code>&lt;configuration description="Runs the hello world test"&gt;
    309     &lt;test class="com.android.tradefed.example.HelloWorldTest" /&gt;
    310     &lt;result_reporter class="com.android.tradefed.result.XmlResultReporter" /&gt;
    311     &lt;logger class="com.android.tradefed.log.FileLogger" /&gt;
    312 &lt;/configuration&gt;
    313 </code></pre>
    314 
    315 <p>Now rebuild and run the helloworld example again.</p>
    316 <pre><code>tf &gt;run example/helloworld 
    317 
    318 05-16 21:38:21 I/XmlResultReporter: Saved device_logcat log to /tmp/0/inv_6390011618174565918/device_logcat_1302097394309452308.txt
    319 05-16 21:38:21 I/XmlResultReporter: Saved host_log log to /tmp/0/inv_6390011618174565918/host_log_4255420317120216614.txt
    320 
    321 </code></pre>
    322 <p>Note the log message indicating the path of the host log. View the contents of that file, and you
    323 should see your HelloWorldTest log message</p>
    324 <pre><code>$ more /tmp/0/inv_6390011618174565918/host_log_4255420317120216614.txt
    325 
    326 05-16 21:38:21 I/HelloWorldTest: Hello, TF World! I have device 004ad9880810a548
    327 </code></pre>
    328 
    329 <p>The TradeFederation framework will also automatically capture the logcat from the allocated device,
    330 and send it the invocation listener for processing. <code>XmlResultReporter</code> will save the
    331 captured device logcat as a file.</p>
    332 
    333 <h2>Option Handling (D, I, R)</h2>
    334 <p>Objects loaded from a Trade Federation Configuration (aka <b>Configuration objects</b>) also have the
    335 ability to receive data from command line arguments.</p>
    336 <p>This is accomplished via the <code>@Option</code> annotation. To participate, a Configuration object class
    337 would apply the <code>@Option</code> annotation to a member field, and provide it a unique name. This would
    338 allow that member field's value to be populated via a command line option, and would also
    339 automatically add that option to the configuration help system (Note: not all field types are
    340 supported: see the
    341 <a href="/reference/com/android/tradefed/config/OptionSetter.html">OptionSetter javadoc</a> for a
    342 description of supported types).</p>
    343 
    344 <p>Let's add an <code>@Option</code> to the HelloWorldTest:</p>
    345 <pre><code>@Option(name="my_option",
    346         shortName='m',
    347         description="this is the option's help text",
    348         // always display this option in the default help text
    349         importance=Importance.ALWAYS)
    350 private String mMyOption = "thisisthedefault";
    351 </code></pre>
    352 
    353 <p>And let's add a log message to display the value of the option in HelloWorldTest, so we can
    354 demonstrate that it was received correctly.</p>
    355 <pre><code>&#64;Override
    356 public void run(ITestInvocationListener listener) throws DeviceNotAvailableException {
    357     
    358     CLog.logAndDisplay(LogLevel.INFO, "I received option '%s'", mMyOption);
    359 </code></pre>
    360 
    361 <p>Rebuild TF and run helloworld; you should see a log message with <code>my_option</code>'s
    362 default value.</p>
    363 <pre><code>tf&gt; run example/helloworld
    364 
    365 05-24 18:30:05 I/HelloWorldTest: I received option 'thisisthedefault'
    366 </code></pre>
    367 
    368 <h3>Passing Values from the Command Line</h3>
    369 <p>Now pass in a value for my_option: you should see my_option getting populated with that value</p>
    370 <pre><code>tf&gt; run example/helloworld --my_option foo
    371 
    372 05-24 18:33:44 I/HelloWorldTest: I received option 'foo'
    373 </code></pre>
    374 
    375 <p>TF configurations also include a help system, which automatically displays help text for
    376 <code>@Option</code> fields. Try it now, and you should see the help text for
    377 <code>my_option</code>:</p>
    378 <pre><code>tf&gt; run --help example/helloworld
    379 Printing help for only the important options. To see help for all options, use the --help-all flag
    380 
    381   cmd_options options:
    382     --[no-]help          display the help text for the most important/critical options. Default: false.
    383     --[no-]help-all      display the full help text for all options. Default: false.
    384     --[no-]loop          keep running continuously. Default: false.
    385 
    386   test options:
    387     -m, --my_option      this is the option's help text Default: thisisthedefault.
    388 
    389   'file' logger options:
    390     --log-level-display  the minimum log level to display on stdout. Must be one of verbose, debug, info, warn, error, assert. Default: error.
    391 </code></pre>
    392 
    393 <p>Note the message at the top about "printing only the important options." To reduce option help
    394 clutter, TF uses the <code>Option#importance</code> attribute to determine whether to show a
    395 particular <code>@Option</code> field's help text
    396 when <code>--help</code> is specified. <code>--help-all</code> will always show help for all
    397 <code>@Option</code> fields, regardless of importance. See the
    398 <a href="/reference/com/android/tradefed/config/Option.Importance.html"
    399 >Option.Importance javadoc</a> for details.</p>
    400 
    401 <h3>Passing Values from a Configuration</h3>
    402 <p>You can also specify an Option's value within the config by adding a
    403 <code>&lt;option name="" value=""&gt;</code> element. Let's see how this looks in
    404 <code>helloworld.xml</code>:</p>
    405 <pre><code>&lt;test class="com.android.tradefed.example.HelloWorldTest" &gt;
    406     &lt;option name="my_option" value="fromxml" /&gt;
    407 &lt;/test&gt;
    408 </code></pre>
    409 
    410 <p>Re-building and running helloworld should now produce this output:</p>
    411 <pre><code>05-24 20:38:25 I/HelloWorldTest: I received option 'fromxml'
    412 </code></pre>
    413 
    414 <p>The configuration help should also be updated to indicate my_option's new default value:</p>
    415 <pre><code>tf&gt; run --help example/helloworld
    416   test options:
    417     -m, --my_option      this is the option's help text Default: fromxml.
    418 </code></pre>
    419 <p>Also note that other configuration objects included in the helloworld config, such as
    420 <code>FileLogger</code>, also accept options. The option <code>--log-level-display</code> is
    421 interesting because it filters the logs that show up on stdout. You may have noticed from earlier
    422 in the tutorial that the "Hello, TF World! I have device ' log message stopped being displayed
    423 on stdout once we switched to using <code>FileLogger</code>. You can increase the verbosity of
    424 logging to stdout by passing in the <code>--log-level-display</code> arg.</p>
    425 
    426 <p>Try this now, and you should see the 'I have device' log message reappear on stdout, in
    427 addition to being logged to a file.</p>
    428 <pre><code>tf &gt;run --log-level-display info example/helloworld
    429 
    430 05-24 18:53:50 I/HelloWorldTest: Hello, TF World! I have device 004ad9880810a548
    431 </code></pre>
    432 
    433 <h2>That's All, Folks!</h2>
    434 <p>As a reminder, if you're stuck on something, the
    435 <a href="https://android.googlesource.com/platform/tools/tradefederation/+/master"
    436 >Trade Federation source code</a> has a lot of useful information that isn't
    437 exposed in the documentation.  And if all else fails, try asking on the
    438 <a href="/source/community/index.html">android-platform</a> Google Group, with "Trade Federation"
    439 in the message subject.</p>
    440 
    441