Skip to content

Commit d6dc361

Browse files
committed
still in progress. renamed and broadened focus to hopefully work for
other types of containers (i.e., not just canvas).
1 parent 071d073 commit d6dc361

File tree

7 files changed

+151
-131
lines changed

7 files changed

+151
-131
lines changed

HISTORY.md

-32
This file was deleted.

README.md

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
1-
## canvas-docker-test
1+
## python-docker-test
22

3-
A unittest.TestCase mixin for executing integration/acceptance tests against a Docker-ized Canvas instance.
3+
A unittest.TestCase mixin for executing integration/acceptance tests against
4+
containerized services.
45

56
### Running your app server
67

7-
In order for your app to visible from the containerized Canvas instance you must bind the app server to Docker's gateway ip (default: 172.17.42.1).
8+
In order for your app to visible from the container you must bind the app server
9+
to it's gateway ip (default: 172.17.42.1).
810

911
In django:
1012

canvas_docker_test/__init__.py

-3
This file was deleted.

canvas_docker_test/mixin.py

-87
This file was deleted.

python_docker_test/__init__.py

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
__version__ = '0.1.0'
2+
3+
from mixin import *

python_docker_test/mixin.py

+137
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,137 @@
1+
# -*- coding: utf-8 -*-
2+
3+
import sys
4+
import threading
5+
from time import sleep
6+
import docker
7+
from docker.errors import APIError
8+
from requests import ConnectionError
9+
10+
__all__ = ['PythonDockerTestMixin', 'ConfigurationError', 'ContainerNotReady']
11+
12+
DEFAULT_READY_TRIES = 10
13+
DEFAULT_READY_SLEEP = 3
14+
15+
class ConfigurationError(Exception):
16+
pass
17+
18+
class ContainerNotReady(Exception):
19+
pass
20+
21+
class ContainerStartThread(threading.Thread):
22+
23+
def __init__(self, image, ready_callback, ready_tries, ready_sleep):
24+
self.is_ready = threading.Event()
25+
self.error = None
26+
self.image = image
27+
self.ready_tries = ready_tries
28+
self.ready_sleep = ready_sleep
29+
self.ready_callback = ready_callback
30+
super(ContainerStartThread, self).__init__()
31+
32+
def run(self):
33+
34+
try:
35+
try:
36+
self.client = docker.Client(version='auto')
37+
self.client.ping()
38+
except ConnectionError, e:
39+
self.error = "Can't connect to docker. Is it installed/running?"
40+
raise
41+
42+
# confirm that the image we want to run is present and pull if not
43+
try:
44+
self.client.inspect_image(self.image)
45+
except APIError, e:
46+
if '404' in str(e.message):
47+
print >>sys.stderr, "%s image not found; pulling..." % self.image
48+
result = self.client.pull(self.image)
49+
if 'error' in result:
50+
raise ConfigurationError(result['error'])
51+
52+
# create and start the container
53+
self.container = self.client.create_container(self.image)
54+
self.client.start(self.container)
55+
self.container_data = self.client.inspect_container(self.container)
56+
57+
if self.ready_callback is not None:
58+
# wait for the container to be "ready"
59+
print >>sys.stderr, "Waiting for container to start..."
60+
tries = self.ready_tries
61+
while tries > 0:
62+
try:
63+
print >>sys.stderr, "Number of tries left: {}".format(tries)
64+
self.ready_callback(self.container_data)
65+
break
66+
except ContainerNotReady:
67+
tries -= 1
68+
sleep(self.ready_sleep)
69+
70+
self.is_ready.set()
71+
72+
except Exception, e:
73+
self.exc_info = sys.exc_info()
74+
if self.error is None:
75+
self.error = e.message
76+
self.is_ready.set()
77+
78+
79+
def terminate(self):
80+
if hasattr(self, 'container'):
81+
self.client.stop(self.container)
82+
self.client.remove_container(self.container)
83+
84+
85+
86+
class PythonDockerTestMixin(object):
87+
88+
@classmethod
89+
def setUpClass(cls):
90+
"""
91+
Creates the client connection to the Docker server. Checks that image
92+
defined in cls.CONTAINER_IMAGE is present and pulls if not. Starts
93+
the container in a separate thread to allow for better cleanup if
94+
exceptions occur during test setup.
95+
"""
96+
if not hasattr(cls, 'CONTAINER_IMAGE'):
97+
raise ConfigurationError("Test class missing CONTAINER_IMAGE attribute")
98+
99+
ready_tries = getattr(cls, 'CONTAINER_READY_TRIES', DEFAULT_READY_TRIES)
100+
ready_sleep = getattr(cls, 'CONTAINER_READY_SLEEP', DEFAULT_READY_SLEEP)
101+
ready_callback = getattr(cls, 'container_ready_callback')
102+
103+
cls.container_start_thread = ContainerStartThread(
104+
cls.CONTAINER_IMAGE, ready_callback, ready_tries, ready_sleep
105+
)
106+
cls.container_start_thread.daemon = True
107+
cls.container_start_thread.start()
108+
109+
# wait for the container startup to complete
110+
cls.container_start_thread.is_ready.wait()
111+
if cls.container_start_thread.error:
112+
# Clean up behind ourselves, since tearDownClass won't get called in
113+
# case of errors.
114+
cls._tearDownClassInternal()
115+
raise cls.container_start_thread.exc_info[1], None, cls.container_start_thread.exc_info[2]
116+
117+
cls.container_data = cls.container_start_thread.container_data
118+
119+
super(PythonDockerTestMixin, cls).setUpClass()
120+
121+
122+
@classmethod
123+
def _tearDownClassInternal(cls):
124+
if hasattr(cls, 'container_start_thread'):
125+
cls.container_start_thread.terminate()
126+
cls.container_start_thread.join()
127+
delattr(cls, 'container_start_thread')
128+
129+
@classmethod
130+
def tearDownClass(cls):
131+
super(PythonDockerTestMixin, cls).tearDownClass()
132+
cls._tearDownClassInternal()
133+
134+
def setUp(self):
135+
self.container_ip = self.container_data['NetworkSettings']['IPAddress']
136+
self.docker_gateway_ip = self.container_data['NetworkSettings']['Gateway']
137+

setup.py

+6-6
Original file line numberDiff line numberDiff line change
@@ -8,19 +8,19 @@
88
def read(path):
99
return codecs.open(os.path.join(here, path), 'r', 'utf-8').read()
1010

11-
version_file = read('canvas_docker_test/__init__.py')
11+
version_file = read('python_docker_test/__init__.py')
1212
version = re.search(r"^__version__ = ['\"]([^'\"]*)['\"]", version_file, re.M).group(1)
1313

1414
setup(
15-
name='canvas-docker-test',
15+
name='python-docker-test',
1616
version=version,
17-
description='unittest.TestCase mixin for executing integration tests against canvas-docker',
17+
description='A unittest.TestCase mixin for executing integration/acceptance tests against containerized services',
1818
author='Jay Luker',
1919
author_email='[email protected]',
20-
url='https://github.com/harvard-dce/canvas-docker-test',
20+
url='https://github.com/harvard-dce/python-docker-test',
2121
packages=find_packages(),
22-
install_requires=[''],
22+
install_requires=['docker-py', 'requests'],
2323
license='MIT License',
24-
keywords='canvas lms lti',
24+
keywords='docker integration acceptance testing',
2525
zip_safe=True
2626
)

0 commit comments

Comments
 (0)