test_middleware_system.py 11 KB

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