test_middleware_system.py 9.5 KB

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