test_xishutil.py 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. # Copyright (c) Twisted Matrix Laboratories.
  2. # See LICENSE for details.
  3. """
  4. Test cases for twisted.words.xish.utility
  5. """
  6. from __future__ import absolute_import, division
  7. from collections import OrderedDict
  8. from twisted.trial import unittest
  9. from twisted.words.xish import utility
  10. from twisted.words.xish.domish import Element
  11. from twisted.words.xish.utility import EventDispatcher
  12. class CallbackTracker:
  13. """
  14. Test helper for tracking callbacks.
  15. Increases a counter on each call to L{call} and stores the object
  16. passed in the call.
  17. """
  18. def __init__(self):
  19. self.called = 0
  20. self.obj = None
  21. def call(self, obj):
  22. self.called = self.called + 1
  23. self.obj = obj
  24. class OrderedCallbackTracker:
  25. """
  26. Test helper for tracking callbacks and their order.
  27. """
  28. def __init__(self):
  29. self.callList = []
  30. def call1(self, object):
  31. self.callList.append(self.call1)
  32. def call2(self, object):
  33. self.callList.append(self.call2)
  34. def call3(self, object):
  35. self.callList.append(self.call3)
  36. class EventDispatcherTests(unittest.TestCase):
  37. """
  38. Tests for L{EventDispatcher}.
  39. """
  40. def testStuff(self):
  41. d = EventDispatcher()
  42. cb1 = CallbackTracker()
  43. cb2 = CallbackTracker()
  44. cb3 = CallbackTracker()
  45. d.addObserver("/message/body", cb1.call)
  46. d.addObserver("/message", cb1.call)
  47. d.addObserver("/presence", cb2.call)
  48. d.addObserver("//event/testevent", cb3.call)
  49. msg = Element(("ns", "message"))
  50. msg.addElement("body")
  51. pres = Element(("ns", "presence"))
  52. pres.addElement("presence")
  53. d.dispatch(msg)
  54. self.assertEqual(cb1.called, 2)
  55. self.assertEqual(cb1.obj, msg)
  56. self.assertEqual(cb2.called, 0)
  57. d.dispatch(pres)
  58. self.assertEqual(cb1.called, 2)
  59. self.assertEqual(cb2.called, 1)
  60. self.assertEqual(cb2.obj, pres)
  61. self.assertEqual(cb3.called, 0)
  62. d.dispatch(d, "//event/testevent")
  63. self.assertEqual(cb3.called, 1)
  64. self.assertEqual(cb3.obj, d)
  65. d.removeObserver("/presence", cb2.call)
  66. d.dispatch(pres)
  67. self.assertEqual(cb2.called, 1)
  68. def test_addObserverTwice(self):
  69. """
  70. Test adding two observers for the same query.
  71. When the event is dispatched both of the observers need to be called.
  72. """
  73. d = EventDispatcher()
  74. cb1 = CallbackTracker()
  75. cb2 = CallbackTracker()
  76. d.addObserver("//event/testevent", cb1.call)
  77. d.addObserver("//event/testevent", cb2.call)
  78. d.dispatch(d, "//event/testevent")
  79. self.assertEqual(cb1.called, 1)
  80. self.assertEqual(cb1.obj, d)
  81. self.assertEqual(cb2.called, 1)
  82. self.assertEqual(cb2.obj, d)
  83. def test_addObserverInDispatch(self):
  84. """
  85. Test for registration of an observer during dispatch.
  86. """
  87. d = EventDispatcher()
  88. msg = Element(("ns", "message"))
  89. cb = CallbackTracker()
  90. def onMessage(_):
  91. d.addObserver("/message", cb.call)
  92. d.addOnetimeObserver("/message", onMessage)
  93. d.dispatch(msg)
  94. self.assertEqual(cb.called, 0)
  95. d.dispatch(msg)
  96. self.assertEqual(cb.called, 1)
  97. d.dispatch(msg)
  98. self.assertEqual(cb.called, 2)
  99. def test_addOnetimeObserverInDispatch(self):
  100. """
  101. Test for registration of a onetime observer during dispatch.
  102. """
  103. d = EventDispatcher()
  104. msg = Element(("ns", "message"))
  105. cb = CallbackTracker()
  106. def onMessage(msg):
  107. d.addOnetimeObserver("/message", cb.call)
  108. d.addOnetimeObserver("/message", onMessage)
  109. d.dispatch(msg)
  110. self.assertEqual(cb.called, 0)
  111. d.dispatch(msg)
  112. self.assertEqual(cb.called, 1)
  113. d.dispatch(msg)
  114. self.assertEqual(cb.called, 1)
  115. def testOnetimeDispatch(self):
  116. d = EventDispatcher()
  117. msg = Element(("ns", "message"))
  118. cb = CallbackTracker()
  119. d.addOnetimeObserver("/message", cb.call)
  120. d.dispatch(msg)
  121. self.assertEqual(cb.called, 1)
  122. d.dispatch(msg)
  123. self.assertEqual(cb.called, 1)
  124. def testDispatcherResult(self):
  125. d = EventDispatcher()
  126. msg = Element(("ns", "message"))
  127. pres = Element(("ns", "presence"))
  128. cb = CallbackTracker()
  129. d.addObserver("/presence", cb.call)
  130. result = d.dispatch(msg)
  131. self.assertEqual(False, result)
  132. result = d.dispatch(pres)
  133. self.assertEqual(True, result)
  134. def testOrderedXPathDispatch(self):
  135. d = EventDispatcher()
  136. cb = OrderedCallbackTracker()
  137. d.addObserver("/message/body", cb.call2)
  138. d.addObserver("/message", cb.call3, -1)
  139. d.addObserver("/message/body", cb.call1, 1)
  140. msg = Element(("ns", "message"))
  141. msg.addElement("body")
  142. d.dispatch(msg)
  143. self.assertEqual(cb.callList, [cb.call1, cb.call2, cb.call3],
  144. "Calls out of order: %s" %
  145. repr([c.__name__ for c in cb.callList]))
  146. # Observers are put into CallbackLists that are then put into dictionaries
  147. # keyed by the event trigger. Upon removal of the last observer for a
  148. # particular event trigger, the (now empty) CallbackList and corresponding
  149. # event trigger should be removed from those dictionaries to prevent
  150. # slowdown and memory leakage.
  151. def test_cleanUpRemoveEventObserver(self):
  152. """
  153. Test observer clean-up after removeObserver for named events.
  154. """
  155. d = EventDispatcher()
  156. cb = CallbackTracker()
  157. d.addObserver('//event/test', cb.call)
  158. d.dispatch(None, '//event/test')
  159. self.assertEqual(1, cb.called)
  160. d.removeObserver('//event/test', cb.call)
  161. self.assertEqual(0, len(d._eventObservers.pop(0)))
  162. def test_cleanUpRemoveXPathObserver(self):
  163. """
  164. Test observer clean-up after removeObserver for XPath events.
  165. """
  166. d = EventDispatcher()
  167. cb = CallbackTracker()
  168. msg = Element((None, "message"))
  169. d.addObserver('/message', cb.call)
  170. d.dispatch(msg)
  171. self.assertEqual(1, cb.called)
  172. d.removeObserver('/message', cb.call)
  173. self.assertEqual(0, len(d._xpathObservers.pop(0)))
  174. def test_cleanUpOnetimeEventObserver(self):
  175. """
  176. Test observer clean-up after onetime named events.
  177. """
  178. d = EventDispatcher()
  179. cb = CallbackTracker()
  180. d.addOnetimeObserver('//event/test', cb.call)
  181. d.dispatch(None, '//event/test')
  182. self.assertEqual(1, cb.called)
  183. self.assertEqual(0, len(d._eventObservers.pop(0)))
  184. def test_cleanUpOnetimeXPathObserver(self):
  185. """
  186. Test observer clean-up after onetime XPath events.
  187. """
  188. d = EventDispatcher()
  189. cb = CallbackTracker()
  190. msg = Element((None, "message"))
  191. d.addOnetimeObserver('/message', cb.call)
  192. d.dispatch(msg)
  193. self.assertEqual(1, cb.called)
  194. self.assertEqual(0, len(d._xpathObservers.pop(0)))
  195. def test_observerRaisingException(self):
  196. """
  197. Test that exceptions in observers do not bubble up to dispatch.
  198. The exceptions raised in observers should be logged and other
  199. observers should be called as if nothing happened.
  200. """
  201. class OrderedCallbackList(utility.CallbackList):
  202. def __init__(self):
  203. self.callbacks = OrderedDict()
  204. class TestError(Exception):
  205. pass
  206. def raiseError(_):
  207. raise TestError()
  208. d = EventDispatcher()
  209. cb = CallbackTracker()
  210. originalCallbackList = utility.CallbackList
  211. try:
  212. utility.CallbackList = OrderedCallbackList
  213. d.addObserver('//event/test', raiseError)
  214. d.addObserver('//event/test', cb.call)
  215. try:
  216. d.dispatch(None, '//event/test')
  217. except TestError:
  218. self.fail("TestError raised. Should have been logged instead.")
  219. self.assertEqual(1, len(self.flushLoggedErrors(TestError)))
  220. self.assertEqual(1, cb.called)
  221. finally:
  222. utility.CallbackList = originalCallbackList
  223. class XmlPipeTests(unittest.TestCase):
  224. """
  225. Tests for L{twisted.words.xish.utility.XmlPipe}.
  226. """
  227. def setUp(self):
  228. self.pipe = utility.XmlPipe()
  229. def test_sendFromSource(self):
  230. """
  231. Send an element from the source and observe it from the sink.
  232. """
  233. def cb(obj):
  234. called.append(obj)
  235. called = []
  236. self.pipe.sink.addObserver('/test[@xmlns="testns"]', cb)
  237. element = Element(('testns', 'test'))
  238. self.pipe.source.send(element)
  239. self.assertEqual([element], called)
  240. def test_sendFromSink(self):
  241. """
  242. Send an element from the sink and observe it from the source.
  243. """
  244. def cb(obj):
  245. called.append(obj)
  246. called = []
  247. self.pipe.source.addObserver('/test[@xmlns="testns"]', cb)
  248. element = Element(('testns', 'test'))
  249. self.pipe.sink.send(element)
  250. self.assertEqual([element], called)