Build testing using Autotest

From Yocto Project
Revision as of 16:35, 6 March 2013 by Stefans (talk | contribs)
(diff) ← Older revision | Latest revision (diff) | Newer revision → (diff)
Jump to navigationJump to search

DRAFT DRAFT DRAFT

What is Autotest ?

Autotest is a framework for fully automated testing. It is designed primarily to test the Linux kernel, though it can serve many other purposes such as qualifying new hardware, virtualization testing and other general user space program testing under linux platforms.

If you want to know more about Autotest itself see the Autotest wiki. [Fedora's AutoQA] is one example of a project that uses autotest as their test automation framework.

Because of the increasing need to automate our build tests the Yocto QA team decided to use Autotest. Currently there are 12 test cases that we run using Autotest.

Before using any of the test cases or writing new tests you must understand how things work, and in order to do that you should read the Autotest documentation:

https://github.com/autotest/autotest/wiki/ClientQuickStart

https://github.com/autotest/autotest/wiki/ControlRequirements

https://github.com/autotest/autotest/wiki/ControlHowto

https://github.com/autotest/autotest/wiki/AutotestApi

https://github.com/autotest/autotest/wiki/ResultsSpecification

How to run the tests using the standalone autotest client on your machine

Types of tests and paths

If you have read the autotest docs you should know that there are more types of tests:

  • client-side control files, which can be
    • manually deployed and run (what this section is about)
    • deployed by autotest server and run automatically on client machines (next section)
  • server-side control files (when jobs need to run stuff both on the server and multiple autotest clients) - we don't care about these.

Have a look here https://github.com/autotest/autotest/wiki/DirectoryStructure to learn more about the directory structure of Autotest.


Get Autotest

  • clone the autotest repo
$ git clone https://github.com/autotest/autotest 
$ cd autotest/client/
  • get our tests (don't mind the path, just clone it to tmp/site_tests)
$ git clone -b stefans/tests git://git.yoctoproject.org/poky-contrib tmp/site_tests

Ok, so now you should have autotest installed and our tests in ~/autotest/client/tmp/site_tests.

You'll see a bunch of control.* files and a heater.py if you "ls -l ~/autotest/client/tmp/site_tests/heater"

heater.py it's the actual test code (so we only wrote one test which we reuse in different ways by running different control files).

  • Let's see the available tests:
$ cd ~/autotest/client
$ ./autotest list
Remotely fetched tests (/home/stefans/autotest/client/tmp/site_tests)
[Control]                                          [Description]
heater/control.172-sstate-sato-satosdk             172 - Build core-image-sato and core-image-sato-sdk
heater/control.292-lib64lsbsdk                     292 - Build lib64-core-image-lsb-sdk
heater/control.minimal                             Build core-image-minimal with a clean builddir
heater/control.280-lib32connman                    280 - Build core-image-sato with multilib and lib32-connman-gnome installed
heater/control.169-remake                          169 - Build remake and remake-native
heater/control.290-lib64sato                       290 - Build lib64-core-image-sato
heater/control.291-lib64satosdk                    291 - Build lib64-core-image-sato-sdk
heater/control.311-buildapp                        322 - Bitbake build-appliance-image
heater/control.options                             This is just a dumb control file used for global options
heater/control.296-perf                            296 - Build perf recipe
heater/control.281x-minx32                         almost 281 - Build core-image-minimal with x32 tune
heater/control.325-emgd                            325 - Build emgd-driver-bin
  • Here is what autotest shows for heater/control.minimal (add --verbose more noise and bitbake's output)
[stefans@firebird client]$ ./autotest run heater/control.minimal
18:14:52 INFO | Writing results to /home/stefans/autotest/client/results/default
18:14:52 INFO | Could not symlink init scripts (lack of permissions)
18:14:52 INFO | Not logging /proc/slabinfo (lack of permissions)
18:14:52 INFO | START	----	----	timestamp=1362413692	localtime=Mar 04 18:14:52	
18:14:52 INFO | 	START	heater.minimal	heater.minimal	timestamp=1362413692	localtime=Mar 04 18:14:52	
18:14:52 INFO | Not logging /proc/slabinfo (lack of permissions)
18:14:52 INFO | Removing clone directory if it exists...
18:14:52 INFO | Getting repo...
18:14:52 INFO | Fetching git [REP 'git://git.yoctoproject.org/poky' BRANCH 'master'] -> /home/stefans/autotest/client/tmp/heater/src/poky
18:15:08 INFO | git commit ID is 93ec7b4d1550e07caec86e2998d0f94a01c7e785 (tag 1.4_M4.rc1-142-g93ec7b4)
18:15:08 INFO | The repo dir used is: /home/stefans/autotest/client/tmp/heater/src/poky
18:15:08 INFO | The build dir used is: /home/stefans/autotest/client/tmp/heater/src/build
18:15:08 INFO | Removing build directory if it exists...
18:15:08 INFO | Config file is:
BB_NUMBER_THREADS = "8"
PARALLEL_MAKE = "-j 8"
MACHINE ??= "qemux86-64"
DISTRO ?= "poky"
PACKAGE_CLASSES ?= "package_rpm"
EXTRA_IMAGE_FEATURES = "debug-tweaks"
USER_CLASSES ?= "buildstats image-mklibs image-prelink"
PATCHRESOLVE = "noop"
CONF_VERSION = "1"
BB_DISKMON_DIRS = "\
            STOPTASKS,${TMPDIR},1G,100K \
            STOPTASKS,${DL_DIR},1G,100K \
            STOPTASKS,${SSTATE_DIR},1G,100K \
            ABORT,${TMPDIR},100M,1K \
            ABORT,${DL_DIR},100M,1K \
            ABORT,${SSTATE_DIR},100M,1K" 

SSTATE_DIR ?= "/data/yocto/sstate-cache"
DL_DIR ?= "/data/yocto/downloads"

18:15:08 INFO | Command issued is: source ./oe-init-build-env /home/stefans/autotest/client/tmp/heater/src/build; export CLONEDIR=/home/stefans/autotest/client/tmp/heater/src/poky; bitbake core-image-minimal
18:46:52 INFO | Command returned zero exit status... That's GOOD!
18:46:52 INFO | Not logging /proc/slabinfo (lack of permissions)
18:46:52 INFO | Not logging /var/log/messages (lack of permissions)
18:46:52 INFO | 		GOOD	heater.minimal	heater.minimal	timestamp=1362415612	localtime=Mar 04 18:46:52	completed successfully
18:46:52 INFO | 	END GOOD	heater.minimal	heater.minimal	timestamp=1362415612	localtime=Mar 04 18:46:52	
18:46:52 INFO | END GOOD	----	----	timestamp=1362415612	localtime=Mar 04 18:46:52	

How to write a new test aka control file

What's a control file?

The key defining component of a job is its control file; this file defines all aspects of a jobs life cycle. The control file is a python script which directly drives execution of the tests in question.

Autotest supplies you with a job object which drives the job and supplies services to the control file. A control file can be as simple as: job.run_test('test')

Control files should always start with control.XXXXX. XXXXX is up to you, but we use something like control.<testopia-case-id>-<short descriptive name>

Stuff to know
  • heater.py is the actual code
  • control files call heater with different args
  • heater.py will take care of cloning poky and setting up local.conf and running a build
  • all tests have a default location for the git clone and build dir (which is shared among tests).
  • the clone dir and build dir can be overwritten by the control files themselves with the right args, but we usually don't want to do that.
  • there is a default local.conf used which can pe appended

In the future there will be more classes which can be used for different tests but for the most basic ones what we have now should suffice.

Examples

One of the simplest example is control.minimal which does a bitbake core-image-minimal with a clean build dir.

[stefans@firebird client]$ cat tmp/site_tests/heater/control.minimal 
AUTHOR = "Stefan Stanacar"
NAME = "Build core-image-minimal with a clean builddir"
TIME = "MEDIUM"
TEST_CATEGORY = "Functional"
TEST_CLASS = "General"
TEST_TYPE = "client"

DOC = """
Build core-image-minimal
"""

job.run_test('heater', tag='minimal', cmd='bitbake core-image-minimal', clean=True)

Notes:

What happens behind the scene on my machine (step by step):

  • the clone dir used is: /home/stefans/autotest/client/tmp/heater/src/poky
  • the build dir used is: /home/stefans/autotest/client/tmp/heater/src/build
  • the clone dir is removed if it exists (because of clean=True)
  • git init/fetch/checkout master for git://git.yoctoproject.org/poky (because no branch or commit args were passed to the call it points to master head)
  • the build dir is removed if it exists (becasue of clean=True)
  • a local.conf will be created from a default one (hardcoded in heater.py) which can pe appended or overwritten (not the case here)
  • from inside the clone dir a command is passed to the shell: source ./oe-init-build-env /home/stefans/autotest/client/tmp/heater/src/build; export CLONEDIR=/home/stefans/autotest/client/tmp/heater/src/poky; bitbake core-image-minimal


Another example.

Let's say we want to automate this test case https://bugzilla.yoctoproject.org/tr_show_case.cgi?case_id=325

In this case we would write something like this:

[stefans@firebird ~]$ cd ~/autotest/client/tmp/site_tests/heater/
[stefans@firebird heater]$ cat control.325-emgd 
AUTHOR = "Stefan Stanacar"
NAME = "325 - Build emgd-driver-bin"
TIME = "MEDIUM"
TEST_CATEGORY = "Functional"
TEST_CLASS = "General"
TEST_TYPE = "client"

DOC = """
Set LICENSE_FLAGS_WHITELIST, add meta-intel layer and build emgd-driver-bin for crownbay
"""

appendconf ="""LICENSE_FLAGS_WHITELIST = "license_emgd-driver-bin_1.14"
MACHINE = "crownbay"
"""


if job.run_test('heater', tag='emgd01', gitsrc="git://git.yoctoproject.org/meta-intel", meta=True):
    job.run_test('heater', tag='emgd02', cmd='bitbake emgd-driver-bin', appendconf=appendconf, addlayers="meta-intel meta-intel/meta-crownbay")

Steps:

  • git init or reset /fetch/checkout master for git://git.yoctoproject.org/meta-intel (because no branch or commit args were passed to the call it points to master head)
  • a local.conf will be created from a default one (hardcoded in heater.py) and appendconf is appended to it

..... TBD


Installing Autotest server with Web frontend

Intro

Tests can be run on standalone machines, or in a server-client mode. While running the standalone autotest client it's a convient way to run a single test the machine where you've clone autotest, it's not efficient. You might want to run multiple instances of autotest running jobs consisting of many tests. (a pool of vms running different distros for example)

The server interaction is simple:

  • Copy the control file across
  • Execute the control file repeatedly until it completes
  • Client notifies server of any reboot for monitoring
  • Upon completion of control script, server pulls results back (not client push)

The server consists of three main parts:

  • A mysql database that holds information on all jobs, clients (test machines), users and tests.
  • The dispatcher (monitor_db) - chooses the jobs from the database to run. It's input is the database, pretty much all it does is start autoserv processes to service requests.
    • There is normally one dispatcher process per machine
    • Client side jobs are run asyncronously (as client machines become available)
    • Server side jobs are run syncronously
  • Autoserv: the server manages clients via autoserv processes - there will be one autoserv process per running job?. Each autoserv process:
    • controls and monitors one or more clients
    • verifies clients are working properly, and if it fails verification, attempts to repair it
    • manages the execution of a job
    • updates the autotest software on each client before commencing work.

Installing and configuring

On https://github.com/autotest/autotest/wiki/SysAdmin you can find documentation on how to setup an autotest server infrastructure on your distro. They provide a script for Fedora/Ubuntu which automates the installation. However I recommend you follow the manual steps to get a better understanding of what's happening behind the scenes.

Here is a step by step install on my Fedora 18 machine, which replicates pretty close the docs (or script) from the autotest wiki with some small changes.


Installing autotest server on a Fedora 18 host:
Note: currently autotest has a lot of problems if SELinux is enabled and enforcing (so disable it before anything else, sorry)

# yum install git make wget python-devel unzip
# yum install httpd mod_wsgi mysql-server MySQL-python gnuplot python-crypto python-paramiko java-1.7.0-openjdk-devel python-httplib2
# yum install numpy python-matplotlib libpng-devel freetype-devel python-imaging

- enable and start MySQL
# systemctl enable mysqld.service
# systemctl start mysqld.service
- set a MySQL pass if you haven't done it yet
# mysqladmin -u root password MYSQL_ROOT_PASS

- clone autotest in /usr/local/autotest (if you want to change that be prepared to edit the apache config file and others)
# cd /usr/local
# git clone git://github.com/autotest/autotest.git

- set apache configs
# ln -s /usr/local/autotest/apache/conf /etc/httpd/autotest.d
# ln -s /usr/local/autotest/apache/apache-conf /etc/httpd/conf.d/z_autotest.conf
# ln -s /usr/local/autotest/apache/apache-web-conf /etc/httpd/conf.d/z_autotest-web.conf

- create a user for autotest install (you'll want to use /usr/local/autotest as home dir, otherwise you'll have edit a lot of config files)
# useradd -d /usr/local/autotest autotest
# passwd autotest
# chown -R autotest:autotest /usr/local/autotest
# su - autotest

- edit ~autotest/.bashrc and add your export {http,ftp,http}_proxy=http://proxy/ there if you need a proxy
- edit ~autotest/.gitconfig and add gitproxy config fragment if you need a proxy

- set autotest db (replace MYSQL_ROOT_PASS with the pasword you set earlier and MYSQL_AUTOTEST_PASS wit a passor for the autotest db
$ ./installation_support/autotest-database-turnkey -s --root-password MYSQL_ROOT_PASS -p MYSQL_AUTOTEST_PASS
- as autotest user
$ ./utils/build_externals.py
- compile frontend
$ ./utils/compile_gwt_clients.py -a

- Now you need to edit global_config.ini and set the password usef for autotest db (earlier MYSQL_AUTOTEST_PASS)
- and set a path for autotest clients install (where autotest server will deploy it's code and tests on the machines/VMs)
- Make sure that path it's on a large partition because that's where the builds take place
$ vi global_config.ini
---
password: please_set_this_password

[scroll down]

 [AUTOSERV]
 # Autotest potential install paths
client_autodir_paths: /data/autotest

----

$ exit
# systemctl enable httpd.service
# systemctl start httpd.service
# cp /usr/local/autotest/utils/autotestd.service /etc/systemd/system
# systemctl enable autotestd.service
# systemctl start autotestd.service

That's it, now go to http://localhost/afe and you should see a web interface.

Adding the tests

# su - autotest
$ cd client
$ mv site_tests site_tests.orig
$ git clone -b stefans/tests git://git.yoctoproject.org/poky-contrib site_tests
$ ~/utils/test_importer.py

Each time you want to update the tests you need to go to /usr/local/autotest/client/site_tests and do a git pull and then run /usr/local/autotest/utils/test_importer.py (so they show up in the web interface)

Add clients

Setup password-less ssh connection from the server to the client machine (which can be a VM running on the same host as the autotest server)

  • As autotest user, on the server, create a RSA key
ssh-keygen -t rsa
  • Then, still on the server, and as autotest, copy it to the host:
ssh-copy-id root@<client machine>

From the web interface go to http://localhost/afe/server/admin and add some hosts. Then they would show up when you create a job.