DCD-545: Use TestInfra helpers to query docker image.

This commit is contained in:
Steve Smith 2019-08-06 14:57:51 +10:00
parent 00d24164db
commit 4c01753833
3 changed files with 60 additions and 36 deletions

View File

@ -18,7 +18,6 @@ def docker_cli():
for tag in container.image.tags: for tag in container.image.tags:
if tag.startswith(IMAGE_NAME): if tag.startswith(IMAGE_NAME):
container.remove(force=True) container.remove(force=True)
@pytest.fixture(scope='module', params=DOCKERFILES) @pytest.fixture(scope='module', params=DOCKERFILES)
@ -32,4 +31,4 @@ def image(request):
buildargs=buildargs, buildargs=buildargs,
dockerfile=request.param, dockerfile=request.param,
rm=True)[0] rm=True)[0]
return image return image

View File

@ -17,3 +17,4 @@ urllib3==1.25.3
wcwidth==0.1.7 wcwidth==0.1.7
websocket-client==0.56.0 websocket-client==0.56.0
zipp==0.5.2 zipp==0.5.2
testinfra==3.0.6

View File

@ -2,12 +2,19 @@ import pytest
import io import io
import tarfile import tarfile
import testinfra
import time import time
import xml.etree.ElementTree as etree import xml.etree.ElementTree as etree
import requests import requests
# Run an image and wrap it in a TestInfra host for convenience.
# FIXME: There's probably a way to turn this into a fixture with parameters.
def run_image(docker_cli, image, environment={}, ports={}):
container = docker_cli.containers.run(image, environment=environment, ports=ports, detach=True)
return testinfra.get_host("docker://"+container.id)
# Helper function to get a file-like object from an image # Helper function to get a file-like object from an image
def get_fileobj_from_container(container, filepath): def get_fileobj_from_container(container, filepath):
time.sleep(0.5) # Give container a moment if just started time.sleep(0.5) # Give container a moment if just started
@ -21,9 +28,10 @@ def get_fileobj_from_container(container, filepath):
file = tar.extractfile(filename) file = tar.extractfile(filename)
return file return file
# TestInfra's process command doesn't seem to work for arg matching
def get_procs(container): def get_procs(container):
ps = container.exec_run('ps aux') ps = container.run('ps -axo args')
return ps.output.decode().split('\n') return ps.stdout.split('\n')
def wait_for_proc(container, proc_str, max_wait=10): def wait_for_proc(container, proc_str, max_wait=10):
waited = 0 waited = 0
@ -34,7 +42,54 @@ def wait_for_proc(container, proc_str, max_wait=10):
time.sleep(0.1) time.sleep(0.1)
waited += 0.1 waited += 0.1
raise RuntimeError("Failed to find target process") raise TimeoutError("Failed to find target process")
######################################################################
# Tests
def test_jvm_args(docker_cli, image):
environment = {
'JVM_MINIMUM_MEMORY': '383m',
'JVM_MAXIMUM_MEMORY': '2047m',
'JVM_SUPPORT_RECOMMENDED_ARGS': '-verbose:gc',
}
container = run_image(docker_cli, image, environment=environment)
jvm = wait_for_proc(container, "org.apache.catalina.startup.Bootstrap")
assert f'-Xms{environment.get("JVM_MINIMUM_MEMORY")}' in jvm
assert f'-Xmx{environment.get("JVM_MAXIMUM_MEMORY")}' in jvm
assert environment.get('JVM_SUPPORT_RECOMMENDED_ARGS') in jvm
def test_install_permissions(docker_cli, image):
container = run_image(docker_cli, image)
assert container.file('/opt/atlassian/confluence/conf/server.xml').user == 'root'
for d in ['logs', 'work', 'temp']:
path = '/opt/atlassian/confluence/{}/'.format(d)
assert container.file(path).user == 'confluence'
def test_first_run_state(docker_cli, image):
PORT = 8090
container = run_image(docker_cli, image, ports={PORT: PORT})
jvm = wait_for_proc(container, "org.apache.catalina.startup.Bootstrap")
for i in range(20):
try:
r = requests.get(f'http://localhost:{PORT}/status')
except requests.exceptions.ConnectionError:
pass
else:
if r.status_code == 200:
state = r.json().get('state')
assert state in ('STARTING', 'FIRST_RUN')
return
time.sleep(1)
raise TimeoutError
# def test_server_xml_defaults(docker_cli, image): # def test_server_xml_defaults(docker_cli, image):
@ -113,34 +168,3 @@ def wait_for_proc(container, proc_str, max_wait=10):
# container = docker_cli.containers.run(image, environment=environment, detach=True) # container = docker_cli.containers.run(image, environment=environment, detach=True)
# confluence_cfg_xml = get_fileobj_from_container(container, '/var/atlassian/application-data/confluence/confluence.cfg.xml') # confluence_cfg_xml = get_fileobj_from_container(container, '/var/atlassian/application-data/confluence/confluence.cfg.xml')
# xml = etree.parse(confluence_cfg_xml) # xml = etree.parse(confluence_cfg_xml)
def test_jvm_args(docker_cli, image):
environment = {
'JVM_MINIMUM_MEMORY': '383m',
'JVM_MAXIMUM_MEMORY': '2047m',
'JVM_SUPPORT_RECOMMENDED_ARGS': '-verbose:gc',
}
container = docker_cli.containers.run(image, environment=environment, detach=True)
jvm = wait_for_proc(container, "org.apache.catalina.startup.Bootstrap")
assert f'-Xms{environment.get("JVM_MINIMUM_MEMORY")}' in jvm
assert f'-Xmx{environment.get("JVM_MAXIMUM_MEMORY")}' in jvm
assert environment.get('JVM_SUPPORT_RECOMMENDED_ARGS') in jvm
def test_first_run_state(docker_cli, image):
PORT = 8090
container = docker_cli.containers.run(image, ports={PORT: PORT}, detach=True)
for i in range(20):
try:
r = requests.get(f'http://localhost:{PORT}/status')
except requests.exceptions.ConnectionError:
pass
else:
if r.status_code == 200:
state = r.json().get('state')
assert state in ('STARTING', 'FIRST_RUN')
return
time.sleep(1)
raise TimeoutError