test_middleware_system.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. import unittest
  2. from cloudbridge.cloud.base.events import SimpleEventDispatcher
  3. from cloudbridge.cloud.base.middleware import BaseMiddleware
  4. from cloudbridge.cloud.base.middleware import EventDebugLoggingMiddleware
  5. from cloudbridge.cloud.base.middleware import ExceptionWrappingMiddleware
  6. from cloudbridge.cloud.base.middleware import SimpleMiddlewareManager
  7. from cloudbridge.cloud.base.middleware import dispatch_event
  8. from cloudbridge.cloud.base.middleware import implement
  9. from cloudbridge.cloud.base.middleware import intercept
  10. from cloudbridge.cloud.base.middleware import observe
  11. from cloudbridge.cloud.interfaces.exceptions import CloudBridgeBaseException
  12. from cloudbridge.cloud.interfaces.exceptions import \
  13. InvalidConfigurationException
  14. from cloudbridge.cloud.interfaces.middleware import Middleware
  15. from .helpers import skipIfPython
  16. class MiddlewareSystemTestCase(unittest.TestCase):
  17. def test_basic_middleware(self):
  18. class DummyMiddleWare(Middleware):
  19. def __init__(self):
  20. self.invocation_order = ""
  21. def install(self, event_manager):
  22. self.event_manager = event_manager
  23. self.invocation_order += "install_"
  24. def uninstall(self):
  25. self.invocation_order += "uninstall"
  26. dispatcher = SimpleEventDispatcher()
  27. manager = SimpleMiddlewareManager(dispatcher)
  28. middleware = DummyMiddleWare()
  29. manager.add(middleware)
  30. self.assertEqual(middleware.invocation_order, "install_",
  31. "install should be called when adding new middleware")
  32. manager.remove(middleware)
  33. self.assertEqual(middleware.invocation_order, "install_uninstall",
  34. "uninstall should be called when removing middleware")
  35. def test_base_middleware(self):
  36. EVENT_NAME = "some.event.occurred"
  37. class DummyMiddleWare(BaseMiddleware):
  38. def __init__(self):
  39. self.invocation_order = ""
  40. @intercept(event_pattern="some.event.*", priority=900)
  41. def my_callback_intcpt(self, event_args, *args, **kwargs):
  42. self.invocation_order += "intcpt_"
  43. assert 'first_pos_arg' in args
  44. assert kwargs.get('a_keyword_arg') == "something"
  45. next_handler = event_args.get('next_handler')
  46. return next_handler.invoke(event_args, *args, **kwargs)
  47. @implement(event_pattern="some.event.*", priority=950)
  48. def my_callback_impl(self, *args, **kwargs):
  49. self.invocation_order += "impl_"
  50. assert 'first_pos_arg' in args
  51. assert kwargs.get('a_keyword_arg') == "something"
  52. return "hello"
  53. @observe(event_pattern="some.event.*", priority=1000)
  54. def my_callback_obs(self, event_args, *args, **kwargs):
  55. self.invocation_order += "obs"
  56. assert 'first_pos_arg' in args
  57. assert event_args['result'] == "hello"
  58. assert kwargs.get('a_keyword_arg') == "something"
  59. dispatcher = SimpleEventDispatcher()
  60. manager = SimpleMiddlewareManager(dispatcher)
  61. middleware = DummyMiddleWare()
  62. manager.add(middleware)
  63. dispatcher.dispatch(self, EVENT_NAME, 'first_pos_arg',
  64. a_keyword_arg='something')
  65. self.assertEqual(middleware.invocation_order, "intcpt_impl_obs")
  66. self.assertListEqual(
  67. [middleware.my_callback_intcpt, middleware.my_callback_impl,
  68. middleware.my_callback_obs],
  69. [handler.callback for handler
  70. in dispatcher.get_handlers_for_event(EVENT_NAME)])
  71. manager.remove(middleware)
  72. self.assertListEqual([], dispatcher.get_handlers_for_event(EVENT_NAME))
  73. def test_multiple_middleware(self):
  74. EVENT_NAME = "some.really.interesting.event.occurred"
  75. class DummyMiddleWare1(BaseMiddleware):
  76. @observe(event_pattern="some.really.*", priority=1000)
  77. def my_obs1_3(self, *args, **kwargs):
  78. pass
  79. @implement(event_pattern="some.*", priority=970)
  80. def my_impl1_2(self, *args, **kwargs):
  81. return "hello"
  82. @intercept(event_pattern="some.*", priority=900)
  83. def my_intcpt1_1(self, event_args, *args, **kwargs):
  84. next_handler = event_args.get('next_handler')
  85. return next_handler.invoke(event_args, *args, **kwargs)
  86. class DummyMiddleWare2(BaseMiddleware):
  87. @observe(event_pattern="some.really.*", priority=1050)
  88. def my_obs2_3(self, *args, **kwargs):
  89. pass
  90. @intercept(event_pattern="*", priority=950)
  91. def my_intcpt2_2(self, event_args, *args, **kwargs):
  92. next_handler = event_args.get('next_handler')
  93. return next_handler.invoke(event_args, *args, **kwargs)
  94. @implement(event_pattern="some.really.*", priority=920)
  95. def my_impl2_1(self, *args, **kwargs):
  96. pass
  97. dispatcher = SimpleEventDispatcher()
  98. manager = SimpleMiddlewareManager(dispatcher)
  99. middleware1 = DummyMiddleWare1()
  100. middleware2 = DummyMiddleWare2()
  101. manager.add(middleware1)
  102. manager.add(middleware2)
  103. dispatcher.dispatch(self, EVENT_NAME)
  104. # Callbacks in both middleware classes should be registered
  105. self.assertListEqual(
  106. [middleware1.my_intcpt1_1, middleware2.my_impl2_1,
  107. middleware2.my_intcpt2_2, middleware1.my_impl1_2,
  108. middleware1.my_obs1_3, middleware2.my_obs2_3],
  109. [handler.callback for handler
  110. in dispatcher.get_handlers_for_event(EVENT_NAME)])
  111. manager.remove(middleware1)
  112. # Only middleware2 callbacks should be registered
  113. self.assertListEqual(
  114. [middleware2.my_impl2_1, middleware2.my_intcpt2_2,
  115. middleware2.my_obs2_3],
  116. [handler.callback for handler in
  117. dispatcher.get_handlers_for_event(EVENT_NAME)])
  118. # add middleware back to check that internal state is properly handled
  119. manager.add(middleware1)
  120. # should one again equal original list
  121. self.assertListEqual(
  122. [middleware1.my_intcpt1_1, middleware2.my_impl2_1,
  123. middleware2.my_intcpt2_2, middleware1.my_impl1_2,
  124. middleware1.my_obs1_3, middleware2.my_obs2_3],
  125. [handler.callback for handler
  126. in dispatcher.get_handlers_for_event(EVENT_NAME)])
  127. def test_automatic_middleware(self):
  128. EVENT_NAME = "another.interesting.event.occurred"
  129. class SomeDummyClass(object):
  130. @observe(event_pattern="another.really.*", priority=1000)
  131. def not_a_match(self, *args, **kwargs):
  132. pass
  133. @intercept(event_pattern="another.*", priority=900)
  134. def my_callback_intcpt2(self, *args, **kwargs):
  135. pass
  136. def not_an_event_handler(self, *args, **kwargs):
  137. pass
  138. @observe(event_pattern="another.interesting.*", priority=1000)
  139. def my_callback_obs1(self, *args, **kwargs):
  140. pass
  141. @implement(event_pattern="another.interesting.*", priority=1050)
  142. def my_callback_impl(self, *args, **kwargs):
  143. pass
  144. dispatcher = SimpleEventDispatcher()
  145. manager = SimpleMiddlewareManager(dispatcher)
  146. some_obj = SomeDummyClass()
  147. middleware = manager.add(some_obj)
  148. dispatcher.dispatch(self, EVENT_NAME)
  149. # Middleware should be discovered even if class containing interceptors
  150. # doesn't inherit from Middleware
  151. self.assertListEqual(
  152. [some_obj.my_callback_intcpt2, some_obj.my_callback_obs1,
  153. some_obj.my_callback_impl],
  154. [handler.callback for handler
  155. in dispatcher.get_handlers_for_event(EVENT_NAME)])
  156. manager.remove(middleware)
  157. # Callbacks should be correctly removed
  158. self.assertListEqual(
  159. [],
  160. [handler.callback for handler in
  161. dispatcher.get_handlers_for_event(EVENT_NAME)])
  162. def test_event_decorator(self):
  163. EVENT_NAME = "some.event.occurred"
  164. class SomeDummyClass(object):
  165. def __init__(self):
  166. self.invocation_order = ""
  167. self.events = SimpleEventDispatcher()
  168. @intercept(event_pattern="some.event.*", priority=900)
  169. def my_callback_intcpt(self, event_args, *args, **kwargs):
  170. self.invocation_order += "intcpt_"
  171. assert 'first_pos_arg' in args
  172. assert kwargs.get('a_keyword_arg') == "something"
  173. next_handler = event_args.get('next_handler')
  174. return next_handler.invoke(event_args, *args, **kwargs)
  175. @observe(event_pattern="some.event.*", priority=3000)
  176. def my_callback_obs(self, event_args, *args, **kwargs):
  177. self.invocation_order += "obs"
  178. assert 'first_pos_arg' in args
  179. assert event_args['result'] == "hello"
  180. assert kwargs.get('a_keyword_arg') == "something"
  181. @dispatch_event(EVENT_NAME)
  182. def my_callback_impl(self, *args, **kwargs):
  183. self.invocation_order += "impl_"
  184. assert 'first_pos_arg' in args
  185. assert kwargs.get('a_keyword_arg') == "something"
  186. return "hello"
  187. obj = SomeDummyClass()
  188. manager = SimpleMiddlewareManager(obj.events)
  189. middleware = manager.add(obj)
  190. # calling my_implementation should trigger all events
  191. result = obj.my_callback_impl(
  192. 'first_pos_arg', a_keyword_arg='something')
  193. self.assertEqual(result, "hello")
  194. self.assertEqual(obj.invocation_order, "intcpt_impl_obs")
  195. callbacks = [handler.callback for handler
  196. in middleware.events.get_handlers_for_event(EVENT_NAME)]
  197. self.assertNotIn(
  198. obj.my_callback_impl, callbacks,
  199. "The event impl callback should not be directly contained"
  200. " in callbacks to avoid a circular dispatch")
  201. self.assertEqual(
  202. len(set(callbacks).difference(
  203. set([obj.my_callback_intcpt,
  204. obj.my_callback_obs]))),
  205. 1,
  206. "The event impl callback should be included in the list of"
  207. " callbacks indirectly")
  208. manager.remove(middleware)
  209. # calling my_implementation again should trigger a None response
  210. result = obj.my_callback_impl(
  211. 'first_pos_arg', a_keyword_arg='something')
  212. self.assertEqual(result, None)
  213. class ExceptionWrappingMiddlewareTestCase(unittest.TestCase):
  214. def test_unknown_exception_is_wrapped(self):
  215. EVENT_NAME = "an.exceptional.event"
  216. class SomeDummyClass(object):
  217. @implement(event_pattern=EVENT_NAME, priority=2500)
  218. def raise_a_non_cloudbridge_exception(self, *args, **kwargs):
  219. raise Exception("Some unhandled exception")
  220. dispatcher = SimpleEventDispatcher()
  221. manager = SimpleMiddlewareManager(dispatcher)
  222. middleware = ExceptionWrappingMiddleware()
  223. manager.add(middleware)
  224. # no exception should be raised when there's no next handler
  225. dispatcher.dispatch(self, EVENT_NAME)
  226. some_obj = SomeDummyClass()
  227. manager.add(some_obj)
  228. with self.assertRaises(CloudBridgeBaseException):
  229. dispatcher.dispatch(self, EVENT_NAME)
  230. def test_cloudbridge_exception_is_passed_through(self):
  231. EVENT_NAME = "an.exceptional.event"
  232. class SomeDummyClass(object):
  233. @implement(event_pattern=EVENT_NAME, priority=2500)
  234. def raise_a_cloudbridge_exception(self, *args, **kwargs):
  235. raise InvalidConfigurationException()
  236. dispatcher = SimpleEventDispatcher()
  237. manager = SimpleMiddlewareManager(dispatcher)
  238. some_obj = SomeDummyClass()
  239. manager.add(some_obj)
  240. middleware = ExceptionWrappingMiddleware()
  241. manager.add(middleware)
  242. with self.assertRaises(InvalidConfigurationException):
  243. dispatcher.dispatch(self, EVENT_NAME)
  244. class EventDebugLoggingMiddlewareTestCase(unittest.TestCase):
  245. # Only python 3 has assertLogs support
  246. @skipIfPython("<", 3, 0)
  247. def test_messages_logged(self):
  248. EVENT_NAME = "an.exceptional.event"
  249. class SomeDummyClass(object):
  250. @implement(event_pattern=EVENT_NAME, priority=2500)
  251. def return_some_value(self, *args, **kwargs):
  252. return "hello world"
  253. dispatcher = SimpleEventDispatcher()
  254. manager = SimpleMiddlewareManager(dispatcher)
  255. middleware = EventDebugLoggingMiddleware()
  256. manager.add(middleware)
  257. some_obj = SomeDummyClass()
  258. manager.add(some_obj)
  259. with self.assertLogs('cloudbridge.cloud.base.middleware',
  260. level='DEBUG') as cm:
  261. dispatcher.dispatch(self, EVENT_NAME,
  262. "named_param", keyword_param="hello")
  263. self.assertTrue(
  264. "named_param" in cm.output[0]
  265. and "keyword_param" in cm.output[0] and "hello" in cm.output[0],
  266. "Log output {0} not as expected".format(cm.output[0]))
  267. self.assertTrue(
  268. "hello world" in cm.output[1],
  269. "Log output {0} does not contain result".format(cm.output[1]))