2014-12-22 17:14:16 +00:00
|
|
|
import unittest
|
|
|
|
import etcd
|
2014-12-22 21:22:07 +00:00
|
|
|
import time
|
2014-12-23 19:09:04 +00:00
|
|
|
import json
|
2016-07-08 18:52:14 +00:00
|
|
|
import uuid
|
2014-12-22 17:14:16 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
from trollius import coroutine, get_event_loop, From, Future, Return
|
|
|
|
from mock import Mock
|
2014-12-22 21:22:07 +00:00
|
|
|
from threading import Event
|
2014-12-22 17:14:16 +00:00
|
|
|
|
|
|
|
from buildman.manager.executor import BuilderExecutor
|
2016-07-08 18:52:14 +00:00
|
|
|
from buildman.manager.ephemeral import EphemeralBuilderManager, EXECUTORS
|
2014-12-22 17:14:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
BUILD_UUID = 'deadbeef-dead-beef-dead-deadbeefdead'
|
2014-12-31 16:33:56 +00:00
|
|
|
REALM_ID = '1234-realm'
|
2014-12-22 17:14:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
def async_test(f):
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
coro = coroutine(f)
|
|
|
|
future = coro(*args, **kwargs)
|
|
|
|
loop = get_event_loop()
|
|
|
|
loop.run_until_complete(future)
|
|
|
|
return wrapper
|
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
|
|
|
|
class TestExecutor(BuilderExecutor):
|
|
|
|
job_started = None
|
|
|
|
|
|
|
|
@coroutine
|
|
|
|
def start_builder(self, realm, token, build_uuid):
|
|
|
|
self.job_started = True
|
|
|
|
raise Return(str(uuid.uuid4))
|
|
|
|
|
|
|
|
|
2014-12-22 17:14:16 +00:00
|
|
|
class TestEphemeral(unittest.TestCase):
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
self.etcd_client_mock = None
|
2014-12-22 21:22:07 +00:00
|
|
|
self.etcd_wait_event = Event()
|
2014-12-22 17:14:16 +00:00
|
|
|
self.test_executor = None
|
|
|
|
super(TestEphemeral, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
def _create_mock_etcd_client(self, *args, **kwargs):
|
2014-12-22 21:22:07 +00:00
|
|
|
def hang_until_event(*args, **kwargs):
|
|
|
|
time.sleep(.01) # 10ms to simulate network latency
|
|
|
|
self.etcd_wait_event.wait()
|
|
|
|
|
2014-12-22 17:14:16 +00:00
|
|
|
self.etcd_client_mock = Mock(spec=etcd.Client, name='etcd.Client')
|
2014-12-22 21:22:07 +00:00
|
|
|
self.etcd_client_mock.watch = Mock(side_effect=hang_until_event)
|
2016-07-08 18:52:14 +00:00
|
|
|
self.etcd_client_mock.read = Mock(side_effect=KeyError)
|
|
|
|
self.etcd_client_mock.write = Mock()
|
2014-12-22 17:14:16 +00:00
|
|
|
return self.etcd_client_mock
|
|
|
|
|
2014-12-31 16:33:56 +00:00
|
|
|
def _create_completed_future(self, result=None):
|
|
|
|
def inner(*args, **kwargs):
|
|
|
|
new_future = Future()
|
|
|
|
new_future.set_result(result)
|
|
|
|
return new_future
|
|
|
|
return inner
|
2014-12-22 17:14:16 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
def _create_build_job(self, namespace='namespace', retries=3):
|
2014-12-22 17:14:16 +00:00
|
|
|
mock_job = Mock()
|
|
|
|
mock_job.job_details = {
|
|
|
|
'build_uuid': BUILD_UUID,
|
|
|
|
}
|
2014-12-31 16:33:56 +00:00
|
|
|
mock_job.job_item = {
|
|
|
|
'body': json.dumps(mock_job.job_details),
|
|
|
|
'id': 1,
|
|
|
|
}
|
2016-07-08 18:52:14 +00:00
|
|
|
|
|
|
|
mock_job.namespace = namespace
|
|
|
|
mock_job.retries_remaining = retries
|
2014-12-22 17:14:16 +00:00
|
|
|
return mock_job
|
|
|
|
|
|
|
|
def setUp(self):
|
2016-07-08 18:52:14 +00:00
|
|
|
self._existing_executors = dict(EXECUTORS)
|
2014-12-22 17:14:16 +00:00
|
|
|
|
|
|
|
self.old_etcd_client_klass = EphemeralBuilderManager._etcd_client_klass
|
|
|
|
EphemeralBuilderManager._etcd_client_klass = self._create_mock_etcd_client
|
2016-07-08 18:52:14 +00:00
|
|
|
|
2014-12-22 21:22:07 +00:00
|
|
|
self.etcd_wait_event.clear()
|
2014-12-22 17:14:16 +00:00
|
|
|
|
|
|
|
self.register_component_callback = Mock()
|
2014-12-31 16:33:56 +00:00
|
|
|
self.unregister_component_callback = Mock()
|
2014-12-22 17:14:16 +00:00
|
|
|
self.job_heartbeat_callback = Mock()
|
|
|
|
self.job_complete_callback = Mock()
|
|
|
|
|
|
|
|
self.manager = EphemeralBuilderManager(
|
|
|
|
self.register_component_callback,
|
2014-12-31 16:33:56 +00:00
|
|
|
self.unregister_component_callback,
|
2014-12-22 17:14:16 +00:00
|
|
|
self.job_heartbeat_callback,
|
|
|
|
self.job_complete_callback,
|
2014-12-22 22:24:44 +00:00
|
|
|
'127.0.0.1',
|
|
|
|
30,
|
2014-12-22 17:14:16 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
def tearDown(self):
|
2014-12-22 21:22:07 +00:00
|
|
|
self.etcd_wait_event.set()
|
|
|
|
self.manager.shutdown()
|
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
EXECUTORS = self._existing_executors
|
2014-12-22 17:14:16 +00:00
|
|
|
EphemeralBuilderManager._etcd_client_klass = self.old_etcd_client_klass
|
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
def test_verify_executor_oldconfig(self):
|
|
|
|
EXECUTORS['test'] = TestExecutor
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTOR': 'test',
|
|
|
|
'EXECUTOR_CONFIG': dict(MINIMUM_RETRY_THRESHOLD=42)
|
2014-12-31 16:33:56 +00:00
|
|
|
})
|
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Ensure that we have a single test executor.
|
|
|
|
self.assertEquals(1, len(self.manager.registered_executors))
|
|
|
|
self.assertEquals(42, self.manager.registered_executors[0].minimum_retry_threshold)
|
|
|
|
|
|
|
|
def test_verify_executor_newconfig(self):
|
|
|
|
EXECUTORS['test'] = TestExecutor
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTORS': [{
|
|
|
|
'EXECUTOR': 'test',
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 42
|
|
|
|
}]
|
|
|
|
})
|
2014-12-31 16:33:56 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Ensure that we have a single test executor.
|
|
|
|
self.assertEquals(1, len(self.manager.registered_executors))
|
|
|
|
self.assertEquals(42, self.manager.registered_executors[0].minimum_retry_threshold)
|
|
|
|
|
|
|
|
def test_verify_multiple_executors(self):
|
|
|
|
EXECUTORS['test'] = TestExecutor
|
|
|
|
EXECUTORS['anotherexecutor'] = TestExecutor
|
|
|
|
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTORS': [
|
|
|
|
{
|
|
|
|
'EXECUTOR': 'test',
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 42
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'EXECUTOR': 'anotherexecutor',
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 24
|
|
|
|
},
|
|
|
|
]
|
|
|
|
})
|
2014-12-31 16:33:56 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Ensure that we have a two test executors.
|
|
|
|
self.assertEquals(2, len(self.manager.registered_executors))
|
|
|
|
self.assertEquals(42, self.manager.registered_executors[0].minimum_retry_threshold)
|
|
|
|
self.assertEquals(24, self.manager.registered_executors[1].minimum_retry_threshold)
|
|
|
|
|
|
|
|
def test_skip_invalid_executor(self):
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTORS': [
|
|
|
|
{
|
|
|
|
'EXECUTOR': 'unknown',
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 42
|
|
|
|
},
|
|
|
|
]
|
|
|
|
})
|
2014-12-22 17:14:16 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.assertEquals(0, len(self.manager.registered_executors))
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2014-12-31 16:33:56 +00:00
|
|
|
@async_test
|
2016-07-08 18:52:14 +00:00
|
|
|
def test_schedule_job_namespace_filter(self):
|
|
|
|
EXECUTORS['test'] = TestExecutor
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTORS': [{
|
|
|
|
'EXECUTOR': 'test',
|
|
|
|
'NAMESPACE_WHITELIST': ['something'],
|
|
|
|
}]
|
2014-12-31 16:33:56 +00:00
|
|
|
})
|
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try with a build job in an invalid namespace.
|
|
|
|
build_job = self._create_build_job(namespace='somethingelse')
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertFalse(result[0])
|
2014-12-31 16:33:56 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try with a valid namespace.
|
|
|
|
build_job = self._create_build_job(namespace='something')
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertTrue(result[0])
|
2014-12-31 16:33:56 +00:00
|
|
|
|
2014-12-22 21:22:07 +00:00
|
|
|
@async_test
|
2016-07-08 18:52:14 +00:00
|
|
|
def test_schedule_job_retries_filter(self):
|
|
|
|
EXECUTORS['test'] = TestExecutor
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTORS': [{
|
|
|
|
'EXECUTOR': 'test',
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 2,
|
|
|
|
}]
|
|
|
|
})
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try with a build job that has too few retries.
|
|
|
|
build_job = self._create_build_job(retries=1)
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertFalse(result[0])
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try with a valid job.
|
|
|
|
build_job = self._create_build_job(retries=2)
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertTrue(result[0])
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2015-05-20 15:32:37 +00:00
|
|
|
|
|
|
|
@async_test
|
2016-07-08 18:52:14 +00:00
|
|
|
def test_schedule_job_executor_fallback(self):
|
|
|
|
EXECUTORS['primary'] = TestExecutor
|
|
|
|
EXECUTORS['secondary'] = TestExecutor
|
|
|
|
|
|
|
|
self.manager.initialize({
|
|
|
|
'EXECUTORS': [
|
|
|
|
{
|
|
|
|
'EXECUTOR': 'primary',
|
|
|
|
'NAMESPACE_WHITELIST': ['something'],
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'EXECUTOR': 'secondary',
|
|
|
|
'MINIMUM_RETRY_THRESHOLD': 2,
|
|
|
|
},
|
|
|
|
]
|
2015-05-20 15:32:37 +00:00
|
|
|
})
|
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try a job not matching the primary's namespace filter. Should schedule on secondary.
|
|
|
|
build_job = self._create_build_job(namespace='somethingelse')
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertTrue(result[0])
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.assertIsNone(self.manager.registered_executors[0].job_started)
|
|
|
|
self.assertIsNotNone(self.manager.registered_executors[1].job_started)
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.manager.registered_executors[0].job_started = None
|
|
|
|
self.manager.registered_executors[1].job_started = None
|
2015-05-20 15:32:37 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try a job not matching the primary's retry minimum. Should schedule on secondary.
|
|
|
|
build_job = self._create_build_job(namespace='something', retries=2)
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertTrue(result[0])
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.assertIsNone(self.manager.registered_executors[0].job_started)
|
|
|
|
self.assertIsNotNone(self.manager.registered_executors[1].job_started)
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.manager.registered_executors[0].job_started = None
|
|
|
|
self.manager.registered_executors[1].job_started = None
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try a job matching the primary. Should schedule on the primary.
|
|
|
|
build_job = self._create_build_job(namespace='something', retries=3)
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertTrue(result[0])
|
2014-12-22 21:22:07 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.assertIsNotNone(self.manager.registered_executors[0].job_started)
|
|
|
|
self.assertIsNone(self.manager.registered_executors[1].job_started)
|
|
|
|
|
|
|
|
self.manager.registered_executors[0].job_started = None
|
|
|
|
self.manager.registered_executors[1].job_started = None
|
2014-12-22 22:24:44 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
# Try a job not matching either's restrictions.
|
|
|
|
build_job = self._create_build_job(namespace='somethingelse', retries=1)
|
|
|
|
result = yield From(self.manager.schedule(build_job))
|
|
|
|
self.assertFalse(result[0])
|
2014-12-22 22:24:44 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.assertIsNone(self.manager.registered_executors[0].job_started)
|
|
|
|
self.assertIsNone(self.manager.registered_executors[1].job_started)
|
2014-12-22 22:24:44 +00:00
|
|
|
|
2016-07-08 18:52:14 +00:00
|
|
|
self.manager.registered_executors[0].job_started = None
|
|
|
|
self.manager.registered_executors[1].job_started = None
|
2015-01-29 23:40:41 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|
2015-05-20 15:32:37 +00:00
|
|
|
|