Mercurial > libervia-web
annotate libervia.tac @ 25:46c8d5431198
browser side: added CSS for contactsChooser class
author | Goffi <goffi@goffi.org> |
---|---|
date | Sat, 07 May 2011 23:51:53 +0200 |
parents | 28e203f13144 |
children | 7684e3ceb12d |
rev | line source |
---|---|
0 | 1 #!/usr/bin/python |
2 # -*- coding: utf-8 -*- | |
3 | |
4 """ | |
5 Libervia: a Salut à Toi frontend | |
6 Copyright (C) 2011 Jérôme Poisson (goffi@goffi.org) | |
7 | |
8 This program is free software: you can redistribute it and/or modify | |
9 it under the terms of the GNU Affero General Public License as published by | |
10 the Free Software Foundation, either version 3 of the License, or | |
11 (at your option) any later version. | |
12 | |
13 This program is distributed in the hope that it will be useful, | |
14 but WITHOUT ANY WARRANTY; without even the implied warranty of | |
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
16 GNU Affero General Public License for more details. | |
17 | |
18 You should have received a copy of the GNU Affero General Public License | |
19 along with this program. If not, see <http://www.gnu.org/licenses/>. | |
20 """ | |
21 | |
22 from twisted.application import internet, service | |
23 from twisted.internet import glib2reactor | |
24 glib2reactor.install() | |
25 from twisted.internet import reactor, defer | |
26 | |
27 from twisted.web import server | |
28 from twisted.web import error as weberror | |
29 from twisted.web.static import File | |
30 from twisted.web.resource import Resource | |
10 | 31 from twisted.words.protocols.jabber.jid import JID |
0 | 32 from txjsonrpc.web import jsonrpc |
33 from txjsonrpc import jsonrpclib | |
34 from sat_frontends.bridge.DBus import DBusBridgeFrontend,BridgeExceptionNoService | |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
35 import re |
10 | 36 from server_side.blog import MicroBlog |
37 | |
0 | 38 TIMEOUT = 120 #Session's time out, after that the user will be disconnected |
39 | |
40 | |
41 class MethodHandler(jsonrpc.JSONRPC): | |
42 | |
43 def __init__(self, sat_host): | |
44 jsonrpc.JSONRPC.__init__(self) | |
45 self.sat_host=sat_host | |
46 | |
47 def render(self, request): | |
1 | 48 self.session = request.getSession() |
0 | 49 try: |
1 | 50 profile = self.session.sat_profile |
0 | 51 except AttributeError: |
52 #user is not identified, we return a jsonrpc fault | |
53 parsed = jsonrpclib.loads(request.content.read()) | |
54 fault = jsonrpclib.Fault(0, "Not allowed") #FIXME: define some standard error codes for libervia | |
55 return jsonrpc.JSONRPC._cbRender(self, fault, request, parsed.get('id'), parsed.get('jsonrpc')) | |
56 return jsonrpc.JSONRPC.render(self, request) | |
19 | 57 |
58 def jsonrpc_getProfileJid(self): | |
59 """Return the jid of the profile""" | |
60 profile = self.session.sat_profile | |
61 self.session.sat_jid = self.sat_host.bridge.getParamA("JabberID", "Connection", profile_key=profile) | |
62 return self.session.sat_jid | |
0 | 63 |
64 def jsonrpc_getContacts(self): | |
65 """Return all passed args.""" | |
1 | 66 profile = self.session.sat_profile |
67 return self.sat_host.bridge.getContacts(profile) | |
20 | 68 |
69 def jsonrpc_setStatus(self, status): | |
70 """Change the status""" | |
71 profile = self.session.sat_profile | |
72 print "new status received:", status | |
73 self.sat_host.bridge.setPresence('', '', 0, {'':status}, profile) | |
74 | |
19 | 75 |
76 def jsonrpc_sendMessage(self, to_jid, msg, subject, type): | |
77 """send message""" | |
78 profile = self.session.sat_profile | |
79 return self.sat_host.bridge.sendMessage(to_jid, msg, subject, type, profile) | |
0 | 80 |
11
331c093e4eb3
magicBox is now able to send global microblog
Goffi <goffi@goffi.org>
parents:
10
diff
changeset
|
81 def jsonrpc_sendMblog(self, raw_text): |
331c093e4eb3
magicBox is now able to send global microblog
Goffi <goffi@goffi.org>
parents:
10
diff
changeset
|
82 """Parse raw_text of the microblog box, and send message consequently""" |
331c093e4eb3
magicBox is now able to send global microblog
Goffi <goffi@goffi.org>
parents:
10
diff
changeset
|
83 profile = self.session.sat_profile |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
84 match = re.match(r'@(.+?): *(.*$)', raw_text) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
85 if match: |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
86 recip = match.group(1) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
87 text = match.group(2) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
88 if recip == '@' and text: |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
89 #This text if for the public microblog |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
90 print "Sending message to everybody" |
11
331c093e4eb3
magicBox is now able to send global microblog
Goffi <goffi@goffi.org>
parents:
10
diff
changeset
|
91 return self.sat_host.bridge.sendPersonalEvent("MICROBLOG", {'content':text}, profile) |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
92 else: |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
93 return self.sat_host.bridge.sendGroupBlog([recip], text, profile) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
94 |
20 | 95 def jsonrpc_getPresenceStatus(self): |
96 """Get Presence information for connected contacts""" | |
97 profile = self.session.sat_profile | |
98 return self.sat_host.bridge.getPresenceStatus(profile) | |
99 | |
19 | 100 def jsonrpc_getHistory(self, from_jid, to_jid, size): |
101 """Return history for the from_jid/to_jid couple""" | |
102 #FIXME: this method should definitely be asynchrone, need to fix it !!! | |
103 profile = self.session.sat_profile | |
104 try: | |
105 _jid = JID(self.session.sat_jid) | |
106 except: | |
107 error("No jid saved for this profile") | |
108 return {} | |
109 if JID(from_jid).userhost() != _jid.userhost() and JID(to_jid) != _jid.userhost(): | |
110 error("Trying to get history from a different jid, maybe a hack attempt ?") | |
111 return {} | |
24 | 112 return self.sat_host.bridge.getHistory(from_jid, to_jid, size) |
19 | 113 |
24 | 114 def jsonrpc_launchTarotGame(self, other_players): |
115 """Create a room, invite the other players and start a Tarot game""" | |
116 profile = self.session.sat_profile | |
117 self.sat_host.bridge.tarotGameLaunch(other_players, profile) | |
11
331c093e4eb3
magicBox is now able to send global microblog
Goffi <goffi@goffi.org>
parents:
10
diff
changeset
|
118 |
0 | 119 class Register(jsonrpc.JSONRPC): |
120 """This class manage the registration procedure with SàT | |
121 It provide an api for the browser, check password and setup the web server""" | |
122 | |
123 def __init__(self, sat_host): | |
124 jsonrpc.JSONRPC.__init__(self) | |
125 self.sat_host=sat_host | |
126 self.profiles_waiting={} | |
127 self.request=None | |
128 | |
129 def getWaitingRequest(self, profile): | |
130 """Tell if a profile is trying to log in""" | |
131 if self.profiles_waiting.has_key(profile): | |
132 return self.profiles_waiting[profile] | |
133 else: | |
134 return None | |
135 | |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
136 def _fillMblogNodes(self, result, session): |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
137 """Fill the microblog nodes association for this session""" |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
138 print "Filling session for %s with %s" % (session.sat_profile, result) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
139 session.sat_mblog_nodes = dict(result) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
140 |
0 | 141 def render(self, request): |
142 """ | |
143 Render method with some hacks: | |
144 - if login is requested, try to login with form data | |
145 - except login, every method is jsonrpc | |
146 - user doesn't need to be authentified for isRegistered, but must be for all other methods | |
147 """ | |
148 if request.postpath==['login']: | |
149 return self.login(request) | |
150 _session = request.getSession() | |
151 parsed = jsonrpclib.loads(request.content.read()) | |
152 if parsed.get("method")!="isRegistered": | |
153 #if we don't call login or isRegistered, we need to be identified | |
154 try: | |
155 profile = _session.sat_profile | |
156 except AttributeError: | |
157 #user is not identified, we return a jsonrpc fault | |
158 fault = jsonrpclib.Fault(0, "Not allowed") #FIXME: define some standard error codes for libervia | |
159 return jsonrpc.JSONRPC._cbRender(self, fault, request, parsed.get('id'), parsed.get('jsonrpc')) | |
160 self.request = request | |
161 return jsonrpc.JSONRPC.render(self, request) | |
162 | |
163 def login(self, request): | |
164 """ | |
165 this method is called with the POST information from the registering form | |
166 it test if the password is ok, and log in if it's the case, | |
167 else it return an error | |
168 @param request: request of the register formulaire, must have "login" and "password" as arguments | |
169 @return: A constant indicating the state: | |
170 - BAD REQUEST: something is wrong in the request (bad arguments, profile_key for login) | |
171 - AUTH ERROR: either the profile or the password is wrong | |
172 - ALREADY WAITING: a request has already be made for this profile | |
173 - server.NOT_DONE_YET: the profile is being processed, the return value will be given by self._logged or self._logginError | |
174 """ | |
175 try: | |
176 _login = request.args['login'][0] | |
177 if _login.startswith('@'): | |
178 raise Exception('No profile_key allowed') | |
179 _pass = request.args['password'][0] | |
180 except KeyError: | |
181 return "BAD REQUEST" | |
182 | |
183 _profile_check = self.sat_host.bridge.getProfileName(_login) | |
184 _profile_pass = self.sat_host.bridge.getParamA("Password", "Connection", profile_key=_login) | |
185 | |
186 if not _profile_check or _profile_check != _login or _profile_pass != _pass: | |
187 return "AUTH ERROR" | |
188 | |
189 if self.profiles_waiting.has_key(_login): | |
190 return "ALREADY WAITING" | |
191 | |
192 if self.sat_host.bridge.isConnected(_login): | |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
193 return self._logged(_login, request, finish=False) |
0 | 194 |
195 self.profiles_waiting[_login] = request | |
196 self.sat_host.bridge.connect(_login) | |
197 return server.NOT_DONE_YET | |
198 | |
199 def __cleanWaiting(self, login): | |
200 """Remove login from waiting queue""" | |
201 try: | |
202 del self.profiles_waiting[login] | |
203 except KeyError: | |
204 pass | |
205 | |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
206 def _logged(self, profile, request, finish=True): |
0 | 207 """Set everything when a user just logged |
208 and return "LOGGED" to the requester""" | |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
209 self.__cleanWaiting(profile) |
0 | 210 _session = request.getSession() |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
211 _session.sat_profile = profile |
24 | 212 self.sat_host.prof_connected.add(profile) |
14
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
213 d = defer.Deferred() |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
214 self.sat_host.bridge.getMblogNodes(profile, d.callback, d.errback) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
215 d.addCallback(self._fillMblogNodes, _session) |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
216 if finish: |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
217 request.write('LOGGED') |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
218 request.finish() |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
219 else: |
9bf8ed012adc
- Group microblog management, first draft
Goffi <goffi@goffi.org>
parents:
11
diff
changeset
|
220 return "LOGGED" |
0 | 221 |
222 def _logginError(self, login, request, error_type): | |
223 """Something went wrong during loggin, return an error""" | |
224 self.__cleanWaiting(login) | |
225 return error_type | |
226 | |
227 def jsonrpc_isConnected(self): | |
228 _session = self.request.getSession() | |
229 profile = _session.sat_profile | |
230 return self.sat_host.bridge.isConnected(profile) | |
231 | |
232 def jsonrpc_connect(self): | |
233 _session = self.request.getSession() | |
234 profile = _session.sat_profile | |
235 if self.profiles_waiting.has_key(profile): | |
236 raise jsonrpclib.Fault('1','Already waiting') #FIXME: define some standard error codes for libervia | |
237 self.profiles_waiting[profile] = self.request | |
238 self.sat_host.bridge.connect(profile) | |
239 return server.NOT_DONE_YET | |
240 | |
241 def jsonrpc_isRegistered(self): | |
242 """Tell if the user is already registered""" | |
243 _session = self.request.getSession() | |
244 try: | |
245 profile = _session.sat_profile | |
246 except AttributeError: | |
247 return False | |
248 return True | |
249 | |
250 class SignalHandler(jsonrpc.JSONRPC): | |
251 | |
252 def __init__(self, sat_host): | |
253 Resource.__init__(self) | |
254 self.register=None | |
255 self.sat_host=sat_host | |
3
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
256 self.signalDeferred = {} |
24 | 257 self.queue = {} #XXX: gof: don't forgot to purge queue on session end |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
258 |
0 | 259 def plugRegister(self, register): |
260 self.register = register | |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
261 |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
262 def jsonrpc_getSignals(self): |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
263 """Keep the connection alive until a signal is received, then send it |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
264 @return: (signal, *signal_args)""" |
3
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
265 _session = self.request.getSession() |
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
266 profile = _session.sat_profile |
24 | 267 if profile in self.queue: #if we have signals to send in queue |
268 if self.queue[profile]: | |
269 return self.queue[profile].pop(0) | |
270 else: | |
271 #the queue is empty, we delete the profile from queue | |
272 del self.queue[profile] | |
3
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
273 self.signalDeferred[profile] = defer.Deferred() |
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
274 return self.signalDeferred[profile] |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
275 |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
276 def getGenericCb(self, function_name): |
3
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
277 """Return a generic function which send all params to signalDeferred.callback |
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
278 function must have profile as last argument""" |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
279 def genericCb(*args): |
3
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
280 profile = args[-1] |
24 | 281 if not profile in self.sat_host.prof_connected: |
282 return | |
3
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
283 if profile in self.signalDeferred: |
154d4caa57f4
server side: proper profile management in signals generic callback
Goffi <goffi@goffi.org>
parents:
2
diff
changeset
|
284 self.signalDeferred[profile].callback((function_name,args[:-1])) |
24 | 285 del self.signalDeferred[profile] |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
286 else: |
24 | 287 if not self.queue.has_key(profile): |
288 self.queue[profile] = [] | |
289 self.queue[profile].append((function_name, args[:-1])) | |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
290 return genericCb |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
291 |
0 | 292 def connected(self, profile): |
293 assert(self.register) #register must be plugged | |
294 request = self.register.getWaitingRequest(profile) | |
295 if request: | |
296 self.register._logged(profile, request) | |
297 | |
298 def connectionError(self, error_type, profile): | |
299 assert(self.register) #register must be plugged | |
300 request = self.register.getWaitingRequest(profile) | |
301 if request: #The user is trying to log in | |
302 if error_type == "AUTH_ERROR": | |
303 _error_t = "AUTH ERROR" | |
304 else: | |
305 _error_t = "UNKNOWN" | |
306 self.register._logginError(profile, request, _error_t) | |
307 | |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
308 def render(self, request): |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
309 """ |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
310 Render method wich reject access if user is not identified |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
311 """ |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
312 _session = request.getSession() |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
313 parsed = jsonrpclib.loads(request.content.read()) |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
314 try: |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
315 profile = _session.sat_profile |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
316 except AttributeError: |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
317 #user is not identified, we return a jsonrpc fault |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
318 fault = jsonrpclib.Fault(0, "Not allowed") #FIXME: define some standard error codes for libervia |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
319 return jsonrpc.JSONRPC._cbRender(self, fault, request, parsed.get('id'), parsed.get('jsonrpc')) |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
320 self.request = request |
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
321 return jsonrpc.JSONRPC.render(self, request) |
0 | 322 |
10 | 323 |
0 | 324 class Libervia(service.Service): |
325 | |
326 def __init__(self): | |
327 root = File("output/") | |
328 self.signal_handler = SignalHandler(self) | |
329 _register = Register(self) | |
330 self.signal_handler.plugRegister(_register) | |
331 self.sessions = {} #key = session value = user | |
24 | 332 self.prof_connected = set() #Profiles connected |
0 | 333 ## bridge ## |
334 try: | |
335 self.bridge=DBusBridgeFrontend() | |
336 except BridgeExceptionNoService: | |
337 print(u"Can't connect to SàT backend, are you sure it's launched ?") | |
338 import sys | |
339 sys.exit(1) | |
340 self.bridge.register("connected", self.signal_handler.connected) | |
341 self.bridge.register("connectionError", self.signal_handler.connectionError) | |
24 | 342 for signal_name in ['presenceUpdate', 'personalEvent', 'newMessage', 'roomJoined']: |
2
669c531a857e
signals handling and first draft of microblogging
Goffi <goffi@goffi.org>
parents:
1
diff
changeset
|
343 self.bridge.register(signal_name, self.signal_handler.getGenericCb(signal_name)) |
10 | 344 root.putChild('json_signal_api', self.signal_handler) |
345 root.putChild('json_api', MethodHandler(self)) | |
346 root.putChild('register_api', _register) | |
347 root.putChild('blog', MicroBlog(self)) | |
348 root.putChild('css', File("server_css/")) | |
349 self.site = server.Site(root) | |
0 | 350 |
351 def startService(self): | |
352 reactor.listenTCP(8080, self.site) | |
1 | 353 |
0 | 354 def run(self): |
355 reactor.run() | |
356 | |
357 def stop(self): | |
358 reactor.stop() | |
359 | |
360 | |
361 | |
362 application = service.Application('Libervia') | |
363 service = Libervia() | |
364 service.setServiceParent(application) |