annotate service/reasoning/sse_collector.py @ 1101:2c7fd4e246ed

start sse_collector Ignore-this: eba53ef3b8b7b34089e018595c41d202 darcs-hash:7e63e5133499c94c22545c3357d87c520227656c
author drewp <drewp@bigasterisk.com>
date Fri, 19 Aug 2016 10:54:38 -0700
parents
children 8d89da1915df
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
1101
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
1 """
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
2 requesting /graph/foo returns an SSE patch stream that's the
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
3 result of fetching multiple other SSE patch streams. The result stream
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
4 may include new statements injected by this service.
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
5
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
6 Future:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
7 - filter out unneeded stmts from the sources
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
8 - give a time resolution and concatenate patches faster than that res
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
9 """
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
10
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
11 config = {
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
12 'streams': [
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
13 {'id': 'home',
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
14 'sources': [
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
15 #'http://bang:9059/graph/events',
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
16 'http://plus:9075/graph/events',
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
17 ]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
18 },
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
19 ]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
20 }
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
21
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
22 from crochet import no_setup
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
23 no_setup()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
24
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
25 import sys, logging, weakref, traceback, json
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
26 from twisted.internet import reactor
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
27 import cyclone.web, cyclone.sse
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
28 from rdflib import ConjunctiveGraph
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
29 from rdflib.parser import StringInputSource
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
30 from docopt import docopt
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
31
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
32 from twisted_sse_demo.eventsource import EventSource
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
33
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
34 sys.path.append("../../lib")
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
35 from logsetup import log
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
36 from patchablegraph import patchAsJson
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
37
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
38 sys.path.append("/my/proj/light9")
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
39 from light9.rdfdb.patch import Patch
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
40
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
41 def patchFromJson(j):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
42 body = json.loads(j)['patch']
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
43 a = ConjunctiveGraph()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
44 a.parse(StringInputSource(json.dumps(body['adds'])), format='json-ld')
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
45 d = ConjunctiveGraph()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
46 d.parse(StringInputSource(json.dumps(body['deletes'])), format='json-ld')
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
47 return Patch(addGraph=a, delGraph=d)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
48
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
49 class PatchSource(object):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
50 """wrap EventSource so it emits Patch objects and has an explicit stop method"""
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
51 def __init__(self, url):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
52 self.url = url
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
53 self._listeners = set()#weakref.WeakSet()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
54 log.info('start read from %s', url)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
55 self._eventSource = EventSource(url)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
56 self._eventSource.protocol.delimiter = '\n'
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
57
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
58 self._eventSource.addEventListener('fullGraph', self._onFullGraph)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
59 self._eventSource.addEventListener('patch', self._onMessage)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
60
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
61 def _onFullGraph(self, message):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
62 try:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
63 g = ConjunctiveGraph()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
64 g.parse(StringInputSource(message), format='json-ld')
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
65 p = Patch(addGraph=g)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
66 self._sendPatch(p)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
67 except:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
68 traceback.print_exc()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
69
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
70 def _onMessage(self, message):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
71 try:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
72 p = patchFromJson(message)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
73 self._sendPatch(p)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
74 except:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
75 traceback.print_exc()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
76
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
77 def _sendPatch(self, p):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
78 log.info('output patch to %s listeners', p, len(self._listeners))
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
79 for lis in self._listeners:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
80 lis(p)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
81
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
82 def addPatchListener(self, func):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
83 self._listeners.add(func)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
84
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
85 def stop(self):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
86 log.info('stop read from %s', self.url)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
87 self._eventSource.protocol.stopProducing() #?
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
88 self._eventSource = None
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
89
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
90 def __del__(self):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
91 if self._eventSource:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
92 raise ValueError
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
93
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
94 class GraphClient(object):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
95 """A listener of some EventSources that sends patches to one of our clients."""
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
96
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
97 def __init__(self, handler, patchSources):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
98 self.handler = handler
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
99
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
100 for ps in patchSources:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
101 ps.addPatchListener(self.onPatch)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
102
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
103 def onPatch(self, p):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
104 self.handler.sendEvent(message=patchAsJson(p), event='patch')
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
105
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
106 class GraphClients(object):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
107 """All the active EventClient objects"""
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
108 def __init__(self):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
109 self.clients = {} # url: EventClient
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
110 self.listeners = {} # url: [GraphClient]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
111
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
112 def addSseHandler(self, handler, streamId):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
113 log.info('addSseHandler %r %r', handler, streamId)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
114 matches = [s for s in config['streams'] if s['id'] == streamId]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
115 if len(matches) != 1:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
116 raise ValueError("%s matches for %r" % (len(matches), streamId))
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
117 ecs = []
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
118 for source in matches[0]['sources']:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
119 if source not in self.clients:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
120 self.clients[source] = PatchSource(source)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
121 ecs.append(self.clients[source])
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
122
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
123 self.listeners.setdefault(source, []).append(GraphClient(handler, ecs))
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
124 print self.__dict__
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
125
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
126 def removeSseHandler(self, handler):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
127 log.info('removeSseHandler %r', handler)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
128 for url, graphClients in self.listeners.items():
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
129 keep = []
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
130 for gc in graphClients:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
131 if gc.handler != handler:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
132 keep.append(gc)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
133 graphClients[:] = keep
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
134 if not keep:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
135 self.clients[url].stop()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
136 del self.clients[url]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
137 del self.listeners[url]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
138
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
139 class SomeGraph(cyclone.sse.SSEHandler):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
140 def __init__(self, application, request):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
141 cyclone.sse.SSEHandler.__init__(self, application, request)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
142 self.id = request.uri[len('/graph/'):]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
143 self.graphClients = self.settings.graphClients
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
144
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
145 def bind(self):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
146 self.graphClients.addSseHandler(self, self.id)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
147
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
148 def unbind(self):
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
149 self.graphClients.removeSseHandler(self)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
150
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
151 if __name__ == '__main__':
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
152
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
153 arg = docopt("""
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
154 Usage: sse_collector.py [options]
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
155
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
156 -v Verbose
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
157 """)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
158
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
159 if arg['-v']:
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
160 import twisted.python.log
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
161 twisted.python.log.startLogging(sys.stdout)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
162 log.setLevel(logging.DEBUG)
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
163
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
164
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
165 graphClients = GraphClients()
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
166
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
167 reactor.listenTCP(
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
168 9071,
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
169 cyclone.web.Application(
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
170 handlers=[
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
171 (r'/graph/(.*)', SomeGraph),
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
172 ],
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
173 graphClients=graphClients),
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
174 interface='::')
2c7fd4e246ed start sse_collector
drewp <drewp@bigasterisk.com>
parents:
diff changeset
175 reactor.run()