1 # Copyright (c) Stas Shtin, 2010
3 # This file is part of IPyPBX.
5 # IPyPBX is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU General Public License as published by
7 # the Free Software Foundation, either version 3 of the License, or
8 # (at your option) any later version.
10 # IPyPBX is distributed in the hope that it will be useful,
11 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 # GNU General Public License for more details.
15 # You should have received a copy of the GNU General Public License
16 # along with IPyPBX. If not, see <http://www.gnu.org/licenses/>.
18 import xml.etree.ElementTree as etree
19 from PyQt4 import QtCore, QtNetwork
22 class FreeswitchConfigServer(QtNetwork.QTcpServer):
24 TCP server that receives config requests from freeswitch.
27 <?xml version="1.0" encoding="UTF-8" standalone="no"?>
28 <document type="freeswitch/xml">
29 <section name="result">
30 <result status="not found" />
34 responseTemplate = '''HTTP/1.1 200 OK
35 Content-Type: text/xml; charset=utf-8
40 def __init__(self, database, parent=None):
41 super(FreeswitchConfigServer, self).__init__(parent)
45 self.connection_id = None
46 self.is_running = False
48 GenClass(database) for GenClass in (
51 self.httpRequestParser = HttpRequestParser()
53 def setSocketData(self, host, port, connection_id):
55 Set host and port for socket to listen on.
57 If the settings differ from previous values, server gets restarted.
59 # Check if restart is needed before new settings are applied.
61 (host, port) != (self.host, self.port)) and connection_id
67 self.connection_id = connection_id
68 for generator in self.generators:
69 generator.connection_id = connection_id
71 # Restart server if necessary.
75 def startServer(self):
77 Start listening on our socket.
79 if not self.is_running:
80 if self.host and self.port:
81 self.newConnection.connect(self.clientConnecting)
82 self.listen(QtNetwork.QHostAddress(self.host), self.port)
83 self.is_running = True
87 Stop listening on our socket.
91 self.is_running = False
93 def restartServer(self):
100 def clientConnecting(self):
102 Handle client connection.
104 if self.hasPendingConnections():
105 self.socket = self.nextPendingConnection()
106 self.socket.readyRead.connect(self.receiveData)
108 def receiveData(self):
109 # TODO: read in chunks.
110 for line in str(self.socket.readAll()).split('\r\n'):
111 self.httpRequestParser.handle(line)
113 for generator in self.generators:
114 if generator.canHandle(self.httpRequestParser.data):
116 self.result = etree.tostring(generator.generateConfig(
117 self.httpRequestParser.headers))
120 response = self.httpRequestParser.result or self.configNotFound
121 http_response = self.responseTemplate % (len(response), response)
122 self.socket.write(http_response)
123 self.httpRequestParser.reset()
127 class HttpParseError(Exception):
129 Error parsing HTTP request.
133 class HttpRequestParser(object):
135 A simple state machine for parsing HTTP requests.
137 HTTP_NONE, HTTP_REQUEST, HTTP_HEADERS, HTTP_EMPTY, HTTP_BODY, HTTP_DONE = \
139 HTTP_STATES = ['NONE', 'REQUEST', 'HEADERS', 'EMPTY', 'BODY', 'DONE']
146 Reset parser to initial state.
148 # Initial values for request data.
150 self.request_path = None
151 self.http_version = None
157 self.state = self.HTTP_NONE
159 def handle(self, line):
161 Dispatch line to current state handler.
163 for state in self.HTTP_STATES:
164 if getattr(self, 'HTTP_%s' % state) == self.state:
165 getattr(self, 'handle%s' % state.title())(line)
168 raise HttpParseError('Unknown HTTP state')
170 def handleNone(self, line):
172 Pass line to next state.
177 def handleRequest(self, line):
179 Retrieve HTTP method, request path and HTTP version from request.
182 self.method, self.request_path, self.http_version = line.split(' ')
187 def handleHeaders(self, line):
189 Parse headers while not found an empty line.
192 key, value = line.split(': ')
193 self.headers[key] = value
198 def handleEmpty(self, line):
200 Empty line separator is found - proceed to next state.
204 def handleBody(self, line):
206 Append to message body.
208 if self.method != 'POST':
209 raise HttpParseError('Only POST request are supported')
211 self.data = dict(pair.split('=', 2) for pair in line.split('&'))
213 def handleDone(self, line):
214 raise HttpParseError("Can't read past request end")
217 class FreeswitchConfigGenerator(object):
219 Base class for generating XML configs.
224 def __init__(self, database=None, connection_id=None):
225 self.database = database
226 self.connection_id = connection_id
228 def canHandle(self, params):
230 Check if this generator can handle a request from freeswitch.
232 for key, value in self.param_match.iteritems():
233 if params.get(key, None) != value:
238 def baseElements(self):
239 root_elt = etree.Element('document', type='freeswitch/xml')
240 section_elt = etree.SubElement(
241 root_elt, 'section', name=self.param_match['section'])
242 return root_elt, section_elt
243 baseElements = property(baseElements)
245 def generateConfig(self, params):
246 return NotImplemented
249 def addParams(parent_elt, params):
251 Create params element based on data passed in a list.
253 # Create params element.
254 params_elt = etree.SubElement(parent_elt, 'params')
256 # Add param elements to params element.
257 for name, value in params:
259 params_elt, 'param', name=name, value=str(value))
264 class SofiaConfGenerator(FreeswitchConfigGenerator):
266 Generates sofia.conf.xml config file.
268 param_match = {'section': 'configuration', 'key_value': 'sofia.conf'}
269 config_name = 'sofia.conf'
271 def generateConfig(self, params):
273 root_elt, section_elt = self.baseElements
275 # Create configuration, settings and profiles elements.
276 configuration_elt = etree.SubElement(
277 section_elt, 'configuration', name=self.config_name,
278 description='%s config' % self.config_name)
279 profiles_elt = etree.SubElement(configuration_elt, 'profiles')
281 database = self.database
283 # Create all profiles for current host.
284 profiles_query = database.exec_(
286 select id, name, external_sip_ip, external_rtp_ip, sip_ip, rtp_ip,
287 sip_port, accept_blind_registration, authenticate_calls
288 from ipypbxweb_sipprofile where connection_id = %i
289 ''' % self.parent.connection_id)
290 while profiles_query.next():
291 # Create profile element.
292 profile_id, _ok = profiles_query.value(0).toInt()
293 profile_elt = etree.SubElement(
294 profiles_elt, 'profile',
295 name=profiles_query.value(1).toString())
297 # Create domains for current profile.
298 domains_elt = etree.SubElement(profile_elt, 'domains')
299 domains_query = database.exec_(
300 'select host_name from ipypbxweb_domain where sip_profile_id = '
302 while domains_query.next():
303 domain_elt = etree.SubElement(
304 domains_elt, 'domain',
305 name=domains_query.value(0).toString(), alias='true',
309 profile_sip_port, _ok = profiles_query.value(6).toInt()
311 # Create settings for current profile.
312 settings_elt = etree.SubElement(profile_elt, 'settings')
314 ('dialplan', 'XML,enum'),
315 ('ext-sip-ip', profiles_query.value(2).toString()),
316 ('ext-rtp-ip', profiles_query.value(3).toString()),
317 ('sip-ip', profiles_query.value(4).toString()),
318 ('rtp-ip', profiles_query.value(5).toString()),
319 ('sip-port', profile_sip_port),
321 ('rtp-timer-name', 'soft'),
322 ('codec-prefs', 'PCMU@20i'),
325 ('dtmf-duration', '100'),
327 ('accept-blind-reg', profiles_query.value(7).toBool()),
328 ('auth-calls', profiles_query.value(8).toBool()))
329 self.addParams(settings_elt, params)
331 # Create gateways for current profile.
332 gateways_elt = etree.SubElement(profile_elt, 'gateways')
333 gateways_query = database.exec_(
335 select name, username, realm, from_domain, password,
336 retry_in_seconds, expire_in_seconds, caller_id_in_from_field,
338 from ipypbxweb_gateway where sip_profile_id = %i
340 while gateways_query.next():
341 # Create gateway element.
342 gateway_elt = etree.SubElement(
343 gateways_elt, 'gateway', name=gateways_query.value(0).toString())
344 retry_seconds, _ok = gateways_query.value(5).toInt()
345 expire_seconds, _ok = gateways_query.value(6).toInt()
347 ('username', gateways_query.value(1).toString()),
348 ('realm', gateways_query.value(2).toString()),
349 ('from-domain', gateways_query.value(3).toString()),
350 ('password', gateways_query.value(4).toString()),
351 ('retry-seconds', retry_seconds),
352 ('expire-seconds', expire_seconds),
353 ('caller-id-in-from', gateways_query.value(7).toBool()),
354 ('extension', gateways_query.value(8).toString()),
355 # TODO: proxy, register
357 self.addParams(gateway_elt, params)
361 class DirectoryGenerator(FreeswitchConfigGenerator):
363 Generates user directory.
365 param_match = {'section': 'directory'}
367 def generateConfig(self, params):
369 root_elt, section_elt = self.baseELements
371 database = self.database
373 # Find profile id from params.
374 profile_query = database.exec_(
376 select id from ipypbxweb_sipprofile
377 where name= '%s' and connection_id = %i limit 1
378 ''' % (params['profile'], self.parent.connection_id))
381 if profile_query.next():
382 profile_id, _ok = profile_query.value(0).toInt()
385 # Matching SIP profile not found.
388 # List all domains for this profile.
389 domains_query = database.exec_(
391 select id, host_name from ipypbxweb_domain
392 where sip_profile_id = %i
395 while domains_query.next():
396 domain_id, _ok = domains_query.value(0).toInt()
398 # Create domaim element.
399 domain_elt = etree.SubElement(
400 section_elt, 'domain', name=domains_query.value(1).toString())
402 # TODO: add domain params section if we need it, i.e.:
404 # <param name="dial-string"
405 # value="{presence_id=${dialed_user}@${dialed_domain}}$\
406 # {sofia_contact(${dialed_user}@${dialed_domain})}"/>
409 # For new we put all users into one group called default.
410 groups_elt = etree.SubElement(domain_elt, 'groups')
411 group_elt = etree.SubElement(groups_elt, 'group', name='default')
413 users_elt = etree.SubElement(group_elt, 'users')
415 users_query = database.exec_(
417 select user_id, password from ipypbxweb_endpoint
421 # Create user entries for all endpoints for this domain.
422 while users_query.next():
423 user_elt = etree.SubElement(
424 users_elt, 'user', id=users_query.value(0).toString())
426 # Specify endpoint password.
428 ('password', users_query.value(1).toString()),
430 self.addParams(user_elt, params)
435 class DialplanGenerator(FreeswitchConfigGenerator):
437 Generates XML dialplans.
440 param_match = {'section': 'dialplan'}