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, window):
41 super(FreeswitchConfigServer, self).__init__(window)
45 self.connection_id = None
46 self.is_running = False
48 GenClass(window, self) for GenClass in (
51 self.httpRequestParser = HttpRequestParser(self)
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
69 # Restart server if necessary.
73 def startServer(self):
75 Start listening on our socket.
77 if not self.is_running:
78 if self.host and self.port:
79 self.newConnection.connect(self.clientConnecting)
80 self.listen(QtNetwork.QHostAddress(self.host), self.port)
81 self.is_running = True
85 Stop listening on our socket.
89 self.is_running = False
91 def restartServer(self):
98 def clientConnecting(self):
100 Handle client connection.
102 if self.hasPendingConnections():
103 self.socket = self.nextPendingConnection()
104 self.socket.readyRead.connect(self.receiveData)
106 def receiveData(self):
107 # TODO: read in chunks.
108 for line in str(self.socket.readAll()).split('\r\n'):
110 self.httpRequestParser.handle(line)
112 response = self.httpRequestParser.result or self.configNotFound
113 http_response = self.responseTemplate % (len(response), response)
114 self.socket.write(http_response)
115 self.httpRequestParser.reset()
118 class HttpParseError(Exception):
120 Error parsing HTTP request.
124 class HttpRequestParser(object):
126 A simple state machine for parsing HTTP requests.
128 HTTP_NONE, HTTP_REQUEST, HTTP_HEADERS, HTTP_EMPTY, HTTP_MESSAGE, \
130 HTTP_STATES = ['NONE', 'REQUEST', 'HEADERS', 'EMPTY', 'MESSAGE', 'DONE']
132 def __init__(self, parent):
138 Reset parser to initial state.
140 # Initial values for request data.
142 self.request_path = None
143 self.http_version = None
149 self.state = self.HTTP_NONE
151 def handle(self, line):
153 Dispatch line to current state handler.
155 for state in self.HTTP_STATES:
156 if getattr(self, 'HTTP_%s' % state) == self.state:
157 getattr(self, 'handle%s' % state.title())(line)
160 raise HttpParseError('Unknown HTTP state')
162 def handleNone(self, line):
164 Pass line to next state.
169 def handleRequest(self, line):
171 Retrieve HTTP method, request path and HTTP version from request.
173 self.method, self.request_path, self.http_version = line.split(' ')
176 def handleHeaders(self, line):
178 Parse headers while not found an empty line.
181 key, value = line.split(': ')
182 self.headers[key] = value
187 def handleEmpty(self, line):
189 Empty line separator is found - proceed to next state.
193 def handleMessage(self, line):
195 Append to message body.
197 self.data = dict(pair.split('=', 2) for pair in line.split('&'))
199 #for k, v in self.data.items():
203 for generator in self.parent.generators:
204 if generator.canHandle(self.data):
206 self.result = etree.tostring(generator.generateConfig(
211 class FreeswitchConfigGenerator(object):
213 Base class for generating XML configs.
219 def __init__(self, model, parent):
223 def canHandle(self, params):
224 for key, value in self.param_match.iteritems():
225 if params.get(key, None) != value:
230 def baseElements(self):
231 root_elt = etree.Element('document')
232 section_elt = etree.SubElement(
233 root_elt, 'section', name=self.section_name)
234 return root_elt, section_elt
235 baseElements = property(baseElements)
237 def generateConfig(self, params):
238 return NotImplemented
241 def addParams(parent_elt, params):
242 for name, value in params:
244 parent_elt, 'param', name=name, value=str(value))
247 class SofiaConfGenerator(FreeswitchConfigGenerator):
249 Generates sofia.conf.xml config file.
251 param_match = {'section': 'configuration', 'key_value': 'sofia.conf'}
252 section_name = 'configuration'
253 config_name = 'sofia.conf'
255 def generateConfig(self, params):
257 root_elt, section_elt = self.baseElements
259 # Create configuration, settings and profiles elements.
260 configuration_elt = etree.SubElement(
261 section_elt, 'configuration', name=self.config_name,
262 description='%s config' % self.config_name)
263 settings_elt = etree.SubElement(configuration_elt, 'settings')
264 profiles_elt = etree.SubElement(settings_elt, 'profiles')
266 database = self.model.controllers['connection'].model.database()
268 # Create all profiles for current host.
269 profiles_query = database.exec_(
271 select id, name, external_sip_ip, external_rtp_ip, sip_ip, rtp_ip,
272 sip_port, accept_blind_registration, authenticate_calls
273 from ipypbxweb_sipprofile where connection_id = %i
274 ''' % self.parent.connection_id)
275 while profiles_query.next():
276 profile_id, _ok = profiles_query.value(0).toInt()
277 profile_elt = etree.SubElement(
278 profiles_elt, 'profile',
279 name=profiles_query.value(1).toString())
281 # Create domains for current profile.
282 domains_elt = etree.SubElement(profile_elt, 'domains')
283 domains_query = database.exec_(
284 'select host_name from ipypbxweb_domain where profile_id = '
286 while domains_query.next():
287 domain_elt = etree.SubElement(
288 domains_elt, 'domain', name=domains_quey.value(0),
289 alias='true', parse='true')
291 profile_sip_port, _ok = profiles_query.value(6).toInt()
293 # Create settings for current profile.
294 settings_elt = etree.SubElement(profile_elt, 'settings')
296 ('dialplan', 'XML,enum'),
297 ('ext-sip-ip', profiles_query.value(2).toString()),
298 ('ext-rtp-ip', profiles_query.value(3).toString()),
299 ('sip-ip', profiles_query.value(4).toString()),
300 ('rtp-ip', profiles_query.value(5).toString()),
301 ('sip-port', profile_sip_port),
303 ('rtp-timer-name', 'soft'),
304 ('codec-prefs', 'PCMU@20i'),
307 ('dtmf-duration', '100'),
309 ('accept-blind-reg', profiles_query.value(7).toBool()),
310 ('auth-calls', profiles_query.value(8).toBool()))
311 self.addParams(settings_elt, params)
313 # Create gateways for current profile.
314 gateways_elt = etree.SubElement(profile_elt, 'gateways')
315 gateways_query = database.exec_(
317 select name, username, realm, from_domain, password,
318 retry_seconds, expire_seconds, caller_id_in_from, extension
319 from ipypbxweb_gateway where sipprofile_id = %i
321 while gateways_query.next():
322 gateway_elt = etree.SubElement(
323 gateways_elt, 'gateway', name=gateways_query.value(0).toString())
324 retry_seconds, _ok = gateways_query.value(5).toInt()
325 expire_seconds, _ok = gateways_query.value(6).toInt()
327 ('username', gateways_query.value(1).toString()),
328 ('realm', gateways_query.value(2).toString()),
329 ('from-domain', gateways_query.value(3).toString()),
330 ('password', gateways_query.value(4).toString()),
331 ('retry-seconds', retry_seconds),
332 ('expire-seconds', expire_seconds),
333 ('caller-id-in-from', gateways_query.value(7).toBool()),
334 ('extension', gateways_query.value(8).toString()),
335 # TODO: proxy, register
337 self.addParams(gateway_elt, params)