Backport changes for Cockpit CI

Related: rhbz#1698366
This commit is contained in:
Alexander Todorov 2019-06-17 15:35:31 +03:00 committed by Alexander Todorov
parent 3951363660
commit 8734f662e9
22 changed files with 547 additions and 52 deletions

View File

@ -9,6 +9,12 @@ TAG = lorax-$(VERSION)-$(RELEASE)
IMAGE_RELEASE = $(shell awk -F: '/FROM/ { print $$2}' Dockerfile.test) IMAGE_RELEASE = $(shell awk -F: '/FROM/ { print $$2}' Dockerfile.test)
ifeq ($(TEST_OS),)
TEST_OS = rhel-7-7
endif
export TEST_OS
VM_IMAGE=$(CURDIR)/test/images/$(TEST_OS)
default: all default: all
src/composer/version.py: src/composer/version.py:
@ -43,8 +49,6 @@ test: docs
coverage report -m coverage report -m
[ -f "/usr/bin/coveralls" ] && [ -n "$(COVERALLS_REPO_TOKEN)" ] && coveralls || echo [ -f "/usr/bin/coveralls" ] && [ -n "$(COVERALLS_REPO_TOKEN)" ] && coveralls || echo
./tests/test_cli.sh
# need `losetup`, which needs Docker to be in privileged mode (--privileged) # need `losetup`, which needs Docker to be in privileged mode (--privileged)
# but even so fails in Travis CI # but even so fails in Travis CI
@ -80,9 +84,15 @@ docs:
archive: tag archive: tag
@git archive --format=tar --prefix=$(PKGNAME)-$(VERSION)/ $(TAG) > $(PKGNAME)-$(VERSION).tar @git archive --format=tar --prefix=$(PKGNAME)-$(VERSION)/ $(TAG) > $(PKGNAME)-$(VERSION).tar
@gzip $(PKGNAME)-$(VERSION).tar @gzip -f $(PKGNAME)-$(VERSION).tar
@echo "The archive is in $(PKGNAME)-$(VERSION).tar.gz" @echo "The archive is in $(PKGNAME)-$(VERSION).tar.gz"
srpm: archive $(PKGNAME).spec
rpmbuild -bs \
--define "_sourcedir $(CURDIR)" \
--define "_srcrpmdir $(CURDIR)" \
lorax-composer.spec
local: local:
@rm -rf $(PKGNAME)-$(VERSION).tar.gz @rm -rf $(PKGNAME)-$(VERSION).tar.gz
@rm -rf /var/tmp/$(PKGNAME)-$(VERSION) @rm -rf /var/tmp/$(PKGNAME)-$(VERSION)
@ -106,6 +116,35 @@ docs-in-docker:
ci: check test ci: check test
$(VM_IMAGE): TAG=HEAD
$(VM_IMAGE): srpm bots
srpm=$(shell rpm --qf '%{Name}-%{Version}-%{Release}.src.rpm\n' -q --specfile lorax-composer.spec | head -n1) ; \
bots/image-customize -v \
--resize 20G \
--upload $$srpm:/var/tmp \
--upload $(CURDIR)/test/vm.install:/var/tmp/vm.install \
--upload $(realpath tests):/ \
--run-command "chmod +x /var/tmp/vm.install" \
--run-command "cd /var/tmp; /var/tmp/vm.install $$srpm" \
$(TEST_OS)
[ -f ~/.config/lorax-test-env ] && bots/image-customize \
--upload ~/.config/lorax-test-env:/var/tmp/lorax-test-env \
$(TEST_OS) || echo
# convenience target for the above
vm: $(VM_IMAGE)
echo $(VM_IMAGE)
vm-reset:
rm -f $(VM_IMAGE) $(VM_IMAGE).qcow2
# checkout Cockpit's bots/ directory for standard test VM images and API to launch them
# must be from cockpit's master, as only that has current and existing images; but testvm.py API is stable
bots:
git fetch --depth=1 https://github.com/cockpit-project/cockpit.git
git checkout --force FETCH_HEAD -- bots/
git reset bots
.PHONY: all install check test clean tag docs archive local .PHONY: all install check test clean tag docs archive local
.PHONY: ci_after_success .PHONY: ci_after_success

60
test/README.md Normal file
View File

@ -0,0 +1,60 @@
# Integration Tests
lorax uses Cockpit's integration test framework and infrastructure. To do this,
we're checking out Cockpit's `bots/` subdirectory. It contains links to test
images and tools to manipulate and start virtual machines from them.
Each test is run on a new instance of a virtual machine.
Branch/test matrix is configured in `bots/tests-scan` inside the
[cockpit repository](https://github.com/cockpit-project/cockpit).
## Dependencies
These dependencies are needed on Fedora to run tests locally:
$ sudo dnf install curl expect \
libvirt libvirt-client libvirt-daemon libvirt-python \
python python-libguestfs python-lxml libguestfs-xfs \
python3 libvirt-python3 \
libguestfs-tools qemu qemu-kvm rpm-build rsync xz
## Building a test VM
To build a test VM, run
$ make vm
This downloads a base image from Cockpit's infrastructure. You can control
which image is downloaded with the `TEST_OS` environment variable. Cockpit's
[documentation](https://github.com/cockpit-project/cockpit/blob/master/test/README.md#test-configuration)
lists accepted values. It then creates a new image based on that (a qemu
snapshot) in `tests/images`, which contain the current `tests/` directory and
have newly built rpms from the current checkout installed.
To delete the generated image, run
$ make vm-reset
Base images are stored in `bots/images`. Set `TEST_DATA` to override this
directory.
## Running tests
After building a test image, run
$ ./test/check-cli [TESTNAME]
or any of the other `check-*` scripts. To debug a test failure, pass `--sit`.
This will keep the test machine running after the first failure and print an
ssh line to connect to it.
Run `make vm` after changing tests or lorax source to recreate the test
machine. It is usually not necessary to reset the VM.
## Updating images
The `bots/` directory is checked out from Cockpit when `make vm` is first run.
To get the latest images you need to update it manually (in order not to poll
GitHub every time):
$ make -B bots

75
test/check-api Executable file
View File

@ -0,0 +1,75 @@
#!/usr/bin/python3
from time import sleep
import composertest
import requests
import subprocess
class TestApi(composertest.ComposerTestCase):
"""Test Composer HTTP API"""
def setUp(self):
super(TestApi, self).setUp()
# Forward /run/weldr/api.socket to a port on the host
# Set ExitOnForwardFailure so that ssh blocks until the forward is set
# up before going to the background (-f), which it closes stdout. We
# wait for that by calling read() on it.
self.composer_port = self.network._lock(8080)
forwarder_command = self.ssh_command[:]
forwarder_command.extend(["-fNT",
"-o", "ExitOnForwardFailure=yes",
"-L", "localhost:%d:/run/weldr/api.socket" % self.composer_port])
self.forwarder_proc = subprocess.Popen(forwarder_command, stdout=subprocess.PIPE)
self.forwarder_proc.stdout.read()
def tearDown(self):
self.forwarder_proc.terminate()
sleep(1) # wait and check for timeout
if self.forwarder_proc.poll() is None:
self.forwarder_proc.kill()
super(TestApi, self).tearDown()
def request(self, method, path, check=True):
self.assertEqual(path[0], "/")
url = "http://localhost:%d%s" % (self.composer_port, path)
r = requests.request(method, url, timeout=30)
if check:
r.raise_for_status()
return r
def test_basic(self):
"""Basic checks for the API"""
#
# API status without depsolve errors
#
r = self.request("GET", "/api/status")
self.assertEqual(r.status_code, 200)
status = r.json()
self.assertEqual(status.keys(), { "build", "api", "db_version", "schema_version", "db_supported", "backend", "msgs" })
self.assertEqual(status["msgs"], [])
self.assertEqual(r.headers.keys(), { "Content-Type", "Content-Length", "Date" })
#
# HTTP errors should return json responses
#
r = self.request("GET", "/marmalade", check=False)
self.assertEqual(r.status_code, 404)
self.assertEqual(r.json(), {
"status": False,
"errors": [{ "id": "HTTPError", "code": 404, "msg": "Not Found" }]
})
r = self.request("POST", "/api/status", check=False)
self.assertEqual(r.status_code, 405)
self.assertEqual(r.json(), {
"status": False,
"errors": [{ "id": "HTTPError", "code": 405, "msg": "Method Not Allowed" }]
})
if __name__ == '__main__':
composertest.main()

43
test/check-cli Executable file
View File

@ -0,0 +1,43 @@
#!/usr/bin/python3
import composertest
class TestImages(composertest.ComposerTestCase):
"""
This is the "entry-point" to the test suite when
executed in Cockpit CI. If $TEST_SCENARIO=="" or
$TEST_SCENARIO="images" we end up here.
New test methods should be added here first!
When this target becomes too slow we split out into
separate scenarios!
"""
def test_blueprint_sanity(self):
self.runCliTest("/tests/cli/test_blueprints_sanity.sh")
def test_compose_sanity(self):
self.runCliTest("/tests/cli/test_compose_sanity.sh")
def test_ext4_filesystem(self):
self.runCliTest("/tests/cli/test_compose_ext4-filesystem.sh")
def test_partitioned_disk(self):
self.runCliTest("/tests/cli/test_compose_partitioned-disk.sh")
def test_tar(self):
self.runCliTest("/tests/cli/test_compose_tar.sh")
class TestQcow2(composertest.ComposerTestCase):
def test_qcow2(self):
self.runCliTest("/tests/cli/test_compose_qcow2.sh")
class TestLiveIso(composertest.ComposerTestCase):
def test_live_iso(self):
self.runCliTest("/tests/cli/test_compose_live-iso.sh")
if __name__ == '__main__':
composertest.main()

21
test/check-cloud Executable file
View File

@ -0,0 +1,21 @@
#!/usr/bin/python3
import composertest
class TestCloud(composertest.ComposerTestCase):
def test_aws(self):
self.runCliTest("/tests/cli/test_build_and_deploy_aws.sh")
def test_azure(self):
self.runCliTest("/tests/cli/test_build_and_deploy_azure.sh")
def test_openstack(self):
self.runCliTest("/tests/cli/test_build_and_deploy_openstack.sh")
def test_vmware(self):
self.runCliTest("/tests/cli/test_build_and_deploy_vmware.sh")
if __name__ == '__main__':
composertest.main()

135
test/composertest.py Executable file
View File

@ -0,0 +1,135 @@
#!/usr/bin/python3
from __future__ import print_function
import argparse
import os
import subprocess
import sys
import unittest
# import Cockpit's machinery for test VMs and its browser test API
sys.path.append(os.path.join(os.path.dirname(__file__), "../bots/machine"))
import testvm # pylint: disable=import-error
def print_exception(etype, value, tb):
import traceback
# only include relevant lines
limit = 0
while tb and '__unittest' in tb.tb_frame.f_globals:
limit += 1
tb = tb.tb_next
traceback.print_exception(etype, value, tb, limit=limit)
class ComposerTestCase(unittest.TestCase):
image = testvm.DEFAULT_IMAGE
sit = False
def __init__(self, methodName='runTest'):
super(ComposerTestCase, self).__init__(methodName=methodName)
# by default run() does this and defaultTestResult()
# always creates new object which is local for the .run() method
self.ci_result = self.defaultTestResult()
def run(self, result=None):
# so we override run() and use an object attribute which we can
# reference later in tearDown() and extract the errors from
super(ComposerTestCase, self).run(result=self.ci_result)
def setUp(self):
self.network = testvm.VirtNetwork(0)
self.machine = testvm.VirtMachine(self.image, networking=self.network.host(), memory_mb=2048)
print("Starting virtual machine '%s'" % self.image)
self.machine.start()
self.machine.wait_boot()
# run a command to force starting the SSH master
self.machine.execute("uptime")
self.ssh_command = ["ssh", "-o", "ControlPath=" + self.machine.ssh_master,
"-p", self.machine.ssh_port,
self.machine.ssh_user + "@" + self.machine.ssh_address]
print("Machine is up. Connect to it via:")
print(" ".join(self.ssh_command))
print()
print("Waiting for lorax-composer to become ready...")
curl_command = ["curl", "--max-time", "360",
"--silent",
"--unix-socket", "/run/weldr/api.socket",
"http://localhost/api/status"]
r = subprocess.call(self.ssh_command + curl_command, stdout=open(os.devnull, 'w'))
self.assertEqual(r.returncode, 0)
def tearDown(self):
# `errors` is a list of tuples (method, error)
errors = list(e[1] for e in self.ci_result.errors if e[1])
if errors and self.sit:
for e in errors:
print_exception(*e)
print()
print(" ".join(self.ssh_command))
input("Press RETURN to continue...")
self.machine.stop()
def execute(self, command):
"""Execute a command on the test machine."""
return subprocess.call(self.ssh_command + command)
def runCliTest(self, script):
execute_params = ["CLI=/usr/bin/composer-cli",
"TEST=" + self.id(),
"PACKAGE=composer-cli",
"/tests/test_cli.sh", script]
if self.sit:
execute_params.insert(0, "COMPOSER_TEST_FAIL_FAST=1")
r = self.execute(execute_params)
self.assertEqual(r.returncode, 0)
def print_tests(tests):
for test in tests:
if isinstance(test, unittest.TestSuite):
print_tests(test)
# I don't know how this is used when running the tests
# (maybe not used from what it looks like) so not sure how to refactor it
# elif isinstance(test, unittest.loader._FailedTest):
# name = test.id().replace("unittest.loader._FailedTest.", "")
# print("Error: '%s' does not match a test" % name, file=sys.stderr)
else:
print(test.id().replace("__main__.", ""))
def main():
parser = argparse.ArgumentParser()
parser.add_argument("tests", nargs="*", help="List of tests modules, classes, and methods")
parser.add_argument("-l", "--list", action="store_true", help="Print the list of tests that would be executed")
parser.add_argument("-s", "--sit", action="store_true", help="Halt test execution (but keep VM running) when a test fails")
args = parser.parse_args()
ComposerTestCase.sit = args.sit
module = __import__("__main__")
if args.tests:
tests = unittest.defaultTestLoader.loadTestsFromNames(args.tests, module)
else:
tests = unittest.defaultTestLoader.loadTestsFromModule(module)
if args.list:
print_tests(tests)
return 0
runner = unittest.TextTestRunner(verbosity=2, failfast=args.sit)
result = runner.run(tests)
sys.exit(not result.wasSuccessful())

18
test/run Executable file
View File

@ -0,0 +1,18 @@
#!/bin/sh -e
# This is the expected entry point for Cockpit CI; will be called without
# arguments but with an appropriate $TEST_OS, and optionally $TEST_SCENARIO
make vm
if [ -n "$TEST_SCENARIO" ]; then
if [ "$TEST_SCENARIO" == "live-iso" ]; then
test/check-cli TestLiveIso
elif [ "$TEST_SCENARIO" == "qcow2" ]; then
test/check-cli TestQcow2
else
test/check-cloud TestCloud.test_$TEST_SCENARIO
fi
else
test/check-cli TestImages
test/check-api
fi

35
test/vm.install Executable file
View File

@ -0,0 +1,35 @@
#!/bin/sh -eux
SRPM="$1"
if ! rpm -q beakerlib; then
if [ $(. /etc/os-release && echo $ID) = "rhel" ]; then
(cd /etc/yum.repos.d; curl -O -L http://download.devel.redhat.com/beakerrepos/beaker-client-RedHatEnterpriseLinux.repo)
# The beaker repository doesn't include repos for minor releases
VERSION=$(. /etc/os-release && echo ${VERSION_ID%.*})
yum install -y --releasever=$VERSION --setopt=sslverify=0 beakerlib
# prevent yum from trying to sync the cache again later (it fails without sslverify=0)
rm /etc/yum.repos.d/beaker-client-RedHatEnterpriseLinux.repo
else
yum install -y beakerlib
fi
fi
# Grow root partition to make room for images. This only works on Fedora right now.
parted --script /dev/vda resizepart 2 100%
partprobe
pvs --noheadings -opv_name | xargs pvresize
rootlv=$(findmnt --noheadings -oSOURCE /)
lvresize $rootlv -l+100%FREE -r
rm -rf build-results
su builder -c "/usr/bin/mock --no-clean --resultdir build-results --rebuild $SRPM"
packages=$(find build-results -name '*.rpm' -not -name '*.src.rpm')
rpm -e --verbose $(basename -a ${packages[@]} | sed 's/-[0-9].*.rpm$//') || true
yum install -y $packages
systemctl enable lorax-composer.socket
systemctl enable docker.service

View File

@ -1,5 +1,21 @@
#!/usr/bin/env bash #!/usr/bin/env bash
# Monkey-patch beakerlib to exit on first failure if COMPOSER_TEST_FAIL_FAST is
# set. https://github.com/beakerlib/beakerlib/issues/42
if [ "$COMPOSER_TEST_FAIL_FAST" == "1" ]; then
eval "original$(declare -f __INTERNAL_LogAndJournalFail)"
__INTERNAL_LogAndJournalFail () {
original__INTERNAL_LogAndJournalFail
# end test somewhat cleanly so that beakerlib logs the FAIL correctly
rlPhaseEnd
rlJournalEnd
exit 1
}
fi
# a generic helper function unifying the specific checks executed on a running # a generic helper function unifying the specific checks executed on a running
# image instance # image instance
verify_image() { verify_image() {
@ -31,7 +47,7 @@ check_root_account() {
# ssh returns 255 in case of any ssh error, so it's better to grep the specific error message # ssh returns 255 in case of any ssh error, so it's better to grep the specific error message
rlRun -t -c "ssh $SSH_OPTS -o PubkeyAuthentication=no root@${SSH_MACHINE} 2>&1 | grep -i 'permission denied ('" \ rlRun -t -c "ssh $SSH_OPTS -o PubkeyAuthentication=no root@${SSH_MACHINE} 2>&1 | grep -i 'permission denied ('" \
0 "Can't ssh to '$SSH_MACHINE' as root using password-based auth" 0 "Can't ssh to '$SSH_MACHINE' as root using password-based auth"
rlRun -t -c "ssh $SSH_OPTS ${SSH_USER}@${SSH_MACHINE} \"sudo grep -E '^root:(\*LOCK\*|!)' /etc/shadow\"" \ rlRun -t -c "ssh $SSH_OPTS ${SSH_USER}@${SSH_MACHINE} \"sudo grep -E '^root:(\*LOCK\*|!):' /etc/shadow\"" \
0 "root account is disabled in /etc/shadow" 0 "root account is disabled in /etc/shadow"
rlRun -t -c "ssh $SSH_OPTS ${SSH_USER}@${SSH_MACHINE} \"sudo grep 'USER_LOGIN.*acct=\\\"root\\\".*terminal=ssh.*res=failed' /var/log/audit/audit.log\"" \ rlRun -t -c "ssh $SSH_OPTS ${SSH_USER}@${SSH_MACHINE} \"sudo grep 'USER_LOGIN.*acct=\\\"root\\\".*terminal=ssh.*res=failed' /var/log/audit/audit.log\"" \
0 "audit.log contains entry about unsuccessful root login" 0 "audit.log contains entry about unsuccessful root login"

View File

@ -1,7 +1,10 @@
#!/bin/bash #!/bin/bash
# Note: execute this file from the project root directory # Note: execute this file from the project root directory
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"

View File

@ -7,8 +7,10 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. ./tests/cli/lib/lib.sh . $(dirname 0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -82,7 +84,7 @@ __EOF__
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
sleep 30 sleep 30
done; done;

View File

@ -7,8 +7,10 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. ./tests/cli/lib/lib.sh . $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -70,7 +72,7 @@ rlJournalStart
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
sleep 30 sleep 30
done; done;

View File

@ -7,9 +7,10 @@
# #
##### #####
. /usr/share/beakerlib/beakerlib.sh set -e
. ./tests/cli/lib/lib.sh
. /usr/share/beakerlib/beakerlib.sh
. $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
VENV=`mktemp -d /tmp/ansible.venv.XXX` VENV=`mktemp -d /tmp/ansible.venv.XXX`
@ -29,8 +30,8 @@ rlJournalStart
rlLogInfo "OS_USERNAME=$OS_USERNAME" rlLogInfo "OS_USERNAME=$OS_USERNAME"
fi fi
export OS_TENANT_NAME="${OS_TENANT_NAME:-$OS_USERNAME}" export OS_PROJECT_NAME="${OS_PROJECT_NAME:-$OS_USERNAME}"
rlLogInfo "OS_TENANT_NAME=$OS_TENANT_NAME" rlLogInfo "OS_PROJECT_NAME=$OS_PROJECT_NAME"
if [ -z "$OS_PASSWORD" ]; then if [ -z "$OS_PASSWORD" ]; then
rlFail "OS_PASSWORD is empty!" rlFail "OS_PASSWORD is empty!"
@ -81,7 +82,7 @@ __EOF__
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
sleep 30 sleep 30
done; done;

View File

@ -1,5 +1,5 @@
#!/bin/bash #!/bin/bash
# Note: Execute this file from the project root directory # Note: execute this file from the project root directory
##### #####
# #
@ -7,8 +7,10 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. ./tests/cli/lib/lib.sh . $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -65,7 +67,7 @@ rlJournalStart
SAMPLES="$SAMPLES/samples" SAMPLES="$SAMPLES/samples"
rlPhaseEnd rlPhaseEnd
rlPhaseStartTest "Compose start" rlPhaseStartTest "compose start"
rlAssertEquals "SELinux operates in enforcing mode" "$(getenforce)" "Enforcing" rlAssertEquals "SELinux operates in enforcing mode" "$(getenforce)" "Enforcing"
SSH_KEY_DIR=`mktemp -d /tmp/composer-ssh-keys.XXXXXX` SSH_KEY_DIR=`mktemp -d /tmp/composer-ssh-keys.XXXXXX`
rlRun -t -c "ssh-keygen -t rsa -N '' -f $SSH_KEY_DIR/id_rsa" rlRun -t -c "ssh-keygen -t rsa -N '' -f $SSH_KEY_DIR/id_rsa"
@ -96,9 +98,9 @@ __EOF__
UUID=`echo $UUID | cut -f 2 -d' '` UUID=`echo $UUID | cut -f 2 -d' '`
rlPhaseEnd rlPhaseEnd
rlPhaseStartTest "Compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
sleep 30 sleep 30
done; done;
@ -113,7 +115,7 @@ __EOF__
python $SAMPLES/upload_file_to_datastore.py -S -s $V_HOST -u $V_USERNAME -p $V_PASSWORD \ python $SAMPLES/upload_file_to_datastore.py -S -s $V_HOST -u $V_USERNAME -p $V_PASSWORD \
-d $V_DATASTORE -l `readlink -f $IMAGE` -r $IMAGE -d $V_DATASTORE -l `readlink -f $IMAGE` -r $IMAGE
rlAssert0 "Image upload successful" $? rlAssert0 "Image upload successfull" $?
rlPhaseEnd rlPhaseEnd
rlPhaseStartTest "Start VM instance" rlPhaseStartTest "Start VM instance"
@ -129,13 +131,13 @@ __EOF__
rlLogInfo "INSTANCE_UUID=$INSTANCE_UUID" rlLogInfo "INSTANCE_UUID=$INSTANCE_UUID"
fi fi
# Wait for instance to become running and had assigned a public IP # wait for instance to become running and had assigned a public IP
IP_ADDRESS="None" IP_ADDRESS="None"
while [ "$IP_ADDRESS" == "None" ]; do while [ "$IP_ADDRESS" == "None" ]; do
rlLogInfo "IP_ADDRESS is not assigned yet ..." rlLogInfo "IP_ADDRESS is not assigned yet ..."
sleep 30 sleep 30
IP_ADDRESS=`python $SAMPLES/find_by_uuid.py -S -s $V_HOST -u $V_USERNAME -p $V_PASSWORD \ IP_ADDRESS=`python $SAMPLES/find_by_uuid.py -S -s $V_HOST -u $V_USERNAME -p $V_PASSWORD \
--uuid $INSTANCE_UUID | grep 'ip address' | tr -d ' ' | cut -f2 -d:` --uuid $INSTANCE_UUID | grep 'ip address' | tr -d ' ' | cut -f2- -d:`
done done
rlLogInfo "Running instance IP_ADDRESS=$IP_ADDRESS" rlLogInfo "Running instance IP_ADDRESS=$IP_ADDRESS"
@ -150,7 +152,7 @@ __EOF__
rlPhaseEnd rlPhaseEnd
rlPhaseStartCleanup rlPhaseStartCleanup
# note: VMDK disk is removed when destroying the VM # note: vmdk disk is removed when destroying the VM
python $SAMPLES/destroy_vm.py -S -s $V_HOST -u $V_USERNAME -p $V_PASSWORD --uuid $INSTANCE_UUID python $SAMPLES/destroy_vm.py -S -s $V_HOST -u $V_USERNAME -p $V_PASSWORD --uuid $INSTANCE_UUID
rlAssert0 "VM destroyed" $? rlAssert0 "VM destroyed" $?
rlRun -t -c "$CLI compose delete $UUID" rlRun -t -c "$CLI compose delete $UUID"

View File

@ -9,7 +9,10 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -25,7 +28,7 @@ rlJournalStart
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
sleep 60 sleep 60
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
done; done;

View File

@ -7,15 +7,18 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. ./tests/cli/lib/lib.sh . $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
QEMU="/usr/libexec/qemu-kvm" QEMU_BIN="/usr/bin/qemu-system-$(uname -m)"
QEMU="$QEMU_BIN -machine accel=kvm:tcg"
rlJournalStart rlJournalStart
rlPhaseStartSetup rlPhaseStartSetup
rlAssertExists $QEMU rlAssertExists $QEMU_BIN
rlPhaseEnd rlPhaseEnd
rlPhaseStartTest "compose start" rlPhaseStartTest "compose start"
@ -30,7 +33,7 @@ rlJournalStart
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
sleep 20 sleep 20
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
done; done;
@ -56,7 +59,7 @@ rlJournalStart
rlPhaseEnd rlPhaseEnd
rlPhaseStartCleanup rlPhaseStartCleanup
rlRun -t -c "killall -9 $(basename $QEMU)" rlRun -t -c "killall -9 $(basename $QEMU_BIN)"
rlRun -t -c "$CLI compose delete $UUID" rlRun -t -c "$CLI compose delete $UUID"
rlRun -t -c "rm -rf $IMAGE" rlRun -t -c "rm -rf $IMAGE"
rlPhaseEnd rlPhaseEnd

View File

@ -9,7 +9,10 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -25,7 +28,7 @@ rlJournalStart
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
sleep 60 sleep 60
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
done; done;

View File

@ -7,15 +7,18 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. ./tests/cli/lib/lib.sh . $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
QEMU="/usr/libexec/qemu-kvm" QEMU_BIN="/usr/bin/qemu-system-$(uname -m)"
QEMU="$QEMU_BIN -machine accel=kvm:tcg"
rlJournalStart rlJournalStart
rlPhaseStartSetup rlPhaseStartSetup
rlAssertExists $QEMU rlAssertExists $QEMU_BIN
rlPhaseEnd rlPhaseEnd
rlPhaseStartTest "compose start" rlPhaseStartTest "compose start"
@ -58,7 +61,7 @@ __EOF__
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
sleep 20 sleep 20
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
done; done;
@ -82,7 +85,7 @@ __EOF__
rlPhaseEnd rlPhaseEnd
rlPhaseStartCleanup rlPhaseStartCleanup
rlRun -t -c "killall -9 $(basename $QEMU)" rlRun -t -c "killall -9 $(basename $QEMU_BIN)"
rlRun -t -c "$CLI compose delete $UUID" rlRun -t -c "$CLI compose delete $UUID"
rlRun -t -c "rm -rf $IMAGE $TMP_DIR $SSH_KEY_DIR" rlRun -t -c "rm -rf $IMAGE $TMP_DIR $SSH_KEY_DIR"
rlPhaseEnd rlPhaseEnd

View File

@ -1,7 +1,10 @@
#!/bin/bash #!/bin/bash
# Note: execute this file from the project root directory # Note: execute this file from the project root directory
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -29,7 +32,7 @@ rlJournalStart
rlPhaseStartTest "compose image" rlPhaseStartTest "compose image"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
sleep 60 sleep 60
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
done; done;

View File

@ -7,7 +7,10 @@
# #
##### #####
set -e
. /usr/share/beakerlib/beakerlib.sh . /usr/share/beakerlib/beakerlib.sh
. $(dirname $0)/lib/lib.sh
CLI="${CLI:-./src/bin/composer-cli}" CLI="${CLI:-./src/bin/composer-cli}"
@ -28,7 +31,7 @@ rlJournalStart
rlPhaseStartTest "compose finished" rlPhaseStartTest "compose finished"
if [ -n "$UUID" ]; then if [ -n "$UUID" ]; then
until $CLI compose details $UUID | grep FINISHED; do until $CLI compose details $UUID | grep 'FINISHED\|FAILED'; do
sleep 10 sleep 10
rlLogInfo "Waiting for compose to finish ..." rlLogInfo "Waiting for compose to finish ..."
done; done;
@ -36,6 +39,13 @@ rlJournalStart
rlFail "Compose UUID is empty!" rlFail "Compose UUID is empty!"
fi fi
# Running a compose can lead to a different selinux policy in the
# kernel, which may break docker. Reload the policy from the host and
# restart docker as a workaround.
# See https://bugzilla.redhat.com/show_bug.cgi?id=1711813
semodule -R
systemctl restart docker
rlRun -t -c "$CLI compose image $UUID" rlRun -t -c "$CLI compose image $UUID"
IMAGE="$UUID-root.tar.xz" IMAGE="$UUID-root.tar.xz"
rlPhaseEnd rlPhaseEnd
@ -49,8 +59,8 @@ rlJournalStart
rlPhaseStartTest "Verify tar image with systemd-nspawn" rlPhaseStartTest "Verify tar image with systemd-nspawn"
if [ -f /usr/bin/systemd-nspawn ]; then if [ -f /usr/bin/systemd-nspawn ]; then
NSPAWN_DIR=`mktemp -d /tmp/nspawn.XXXX` NSPAWN_DIR=`mktemp -d /var/tmp/nspawn.XXXX`
rlRun -t -c "tar -xJvf $IMAGE -C $NSPAWN_DIR" rlRun -t -c "tar -xJf $IMAGE -C $NSPAWN_DIR"
# verify we can run a container with this image # verify we can run a container with this image
rlRun -t -c "systemd-nspawn -D $NSPAWN_DIR cat /etc/redhat-release" rlRun -t -c "systemd-nspawn -D $NSPAWN_DIR cat /etc/redhat-release"

View File

@ -1,8 +1,10 @@
#!/bin/bash #!/bin/bash
# Note: Execute this file from the project root directory # Note: execute this file from the project root directory
# setup set -eu
rm -rf /var/tmp/beakerlib-*/
export BEAKERLIB_DIR=$(mktemp -d /tmp/composer-test.XXXXXX)
CLI="${CLI:-}"
function setup_tests { function setup_tests {
local share_dir=$1 local share_dir=$1
@ -43,6 +45,15 @@ function teardown_tests {
mv ${blueprints_dir}.orig $blueprints_dir mv ${blueprints_dir}.orig $blueprints_dir
} }
# cloud credentials
if [ -f "~/.config/lorax-test-env" ]; then
. ~/.config/lorax-test-env
fi
if [ -f "/var/tmp/lorax-test-env" ]; then
. /var/tmp/lorax-test-env
fi
if [ -z "$CLI" ]; then if [ -z "$CLI" ]; then
export top_srcdir=`pwd` export top_srcdir=`pwd`
. ./tests/testenv.sh . ./tests/testenv.sh
@ -55,7 +66,7 @@ if [ -z "$CLI" ]; then
chmod a+rx -R $SHARE_DIR chmod a+rx -R $SHARE_DIR
setup_tests $SHARE_DIR $BLUEPRINTS_DIR setup_tests $SHARE_DIR $BLUEPRINTS_DIR
# Start the lorax-composer daemon # start the lorax-composer daemon
./src/sbin/lorax-composer --sharedir $SHARE_DIR $BLUEPRINTS_DIR & ./src/sbin/lorax-composer --sharedir $SHARE_DIR $BLUEPRINTS_DIR &
else else
export PACKAGE="composer-cli" export PACKAGE="composer-cli"
@ -65,14 +76,14 @@ else
fi fi
# Wait for the backend to become ready # wait for the backend to become ready
tries=0 tries=0
until curl -m 15 --unix-socket /run/weldr/api.socket http://localhost:4000/api/status | grep 'db_supported.*true'; do until curl -m 15 --unix-socket /run/weldr/api.socket http://localhost:4000/api/status | grep 'db_supported.*true'; do
tries=$((tries + 1)) tries=$((tries + 1))
if [ $tries -gt 20 ]; then if [ $tries -gt 50 ]; then
exit 1 exit 1
fi fi
sleep 2 sleep 5
echo "DEBUG: Waiting for backend API to become ready before testing ..." echo "DEBUG: Waiting for backend API to become ready before testing ..."
done; done;
@ -80,14 +91,14 @@ done;
export BEAKERLIB_JOURNAL=0 export BEAKERLIB_JOURNAL=0
export PATH="/usr/local/bin:$PATH" export PATH="/usr/local/bin:$PATH"
if [ -z "$*" ]; then if [ -z "$*" ]; then
# Invoke cli/ tests which can be executed without special preparation # invoke cli/ tests which can be executed without special preparation
./tests/cli/test_blueprints_sanity.sh ./tests/cli/test_blueprints_sanity.sh
./tests/cli/test_compose_sanity.sh ./tests/cli/test_compose_sanity.sh
else else
# Execute other cli tests which need more adjustments in the calling environment # execute other cli tests which need more adjustments in the calling environment
# or can't be executed inside Travis CI # or can't be executed inside Travis CI
for TEST in "$@"; do for TEST in "$@"; do
./$TEST $TEST
done done
fi fi
@ -106,8 +117,11 @@ else
systemctl start lorax-composer systemctl start lorax-composer
fi fi
# Look for failures . $BEAKERLIB_DIR/TestResults
grep RESULT_STRING /var/tmp/beakerlib-*/TestResults | grep -v PASS && exit 1
# Explicit return code for Makefile if [ $TESTRESULT_RESULT_ECODE != 0 ]; then
exit 0 echo "Test failed. Leaving log in $BEAKERLIB_DIR"
exit $TESTRESULT_RESULT_ECODE
fi
rm -rf $BEAKERLIB_DIR

View File

@ -1,5 +1,9 @@
#!/bin/sh #!/bin/sh
top_srcdir="${top_srcdir:-}"
top_buildir="${top_builddir:-}"
PYTHONPATH="${PYTHONPATH:-}"
if [ -z "$top_srcdir" ]; then if [ -z "$top_srcdir" ]; then
echo "*** top_srcdir must be set" echo "*** top_srcdir must be set"
exit 1 exit 1