dff9b24a1c4001dbaa7b47cba36202e18061627b
[ipypbx] / src / ipypbx / http.py
1 # Copyright (c) Stas Shtin, 2010
2
3 # This file is part of IPyPBX.
4
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.
9
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.
14
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/>.
17
18 import xml.etree.ElementTree as etree
19 from PyQt4 import QtCore, QtNetwork
20
21
22 class FreeswitchConfigServer(QtNetwork.QTcpServer):
23     """
24     TCP server that receives config requests from freeswitch.
25     """
26     def __init__(self, parent):
27         super(FreeswitchConfigServer, self).__init__(parent)
28
29         self.host = None
30         self.port = None
31         self.is_running = False
32         self.generators = [
33             GenClass(self.parent().model) for GenClass in (
34                 SofiaConfGenerator,)]
35         
36         self.httpRequestParser = HttpRequestParser(self)
37         
38     def setSocketData(self, host, port):
39         """
40         Set host and port for socket to listen on.
41
42         If the settings differ from previous values, server gets restarted.
43         """
44         # Check if restart is needed before new settings are applied.
45         needs_restart = (host, port) != (self.host, self.port)
46
47         # Save new settings.
48         self.host = host
49         self.port = port
50
51         # Restart server if necessary.
52         if needs_restart:
53             self.restartServer()
54
55     def startServer(self):
56         """
57         Start listening on our socket.
58         """
59         if not self.is_running:
60             if self.host and self.port:
61                 self.newConnection.connect(self.clientConnecting)
62                 self.listen(QtNetwork.QHostAddress(self.host), self.port)
63                 self.is_running = True
64
65     def stopServer(self):
66         """
67         Stop listening on our socket.
68         """
69         if self.is_running:
70             self.close()
71             self.is_running = False
72
73     def restartServer(self):
74         """
75         Restart server.
76         """
77         self.stopServer()
78         self.startServer()
79
80     def clientConnecting(self):
81         """
82         Handle client connection.
83         """
84         if self.hasPendingConnections():
85             self.socket = self.nextPendingConnection()
86             self.socket.readyRead.connect(self.receiveData)
87
88     def receiveData(self):
89         # TODO: read in chunks.
90         for line in str(self.socket.readAll()).split('\r\n'):
91             self.httpRequestParser.handle(line)
92
93
94 class HttpParseError(Exception):
95     """
96     Error parsing HTTP request.
97     """
98
99
100 class HttpRequestParser(object):
101     """
102     A simple state machine for parsing HTTP requests.
103     """
104     HTTP_NONE, HTTP_REQUEST, HTTP_HEADERS, HTTP_EMPTY, HTTP_MESSAGE, \
105         HTTP_DONE = range(6)
106     HTTP_STATES = ['NONE', 'REQUEST', 'HEADERS', 'EMPTY', 'MESSAGE', 'DONE']
107     
108     def __init__(self, parent):
109         self.parent = parent
110         self.reset()
111
112     def reset(self):
113         """
114         Reset parser to initial state.
115         """
116         # Initial values for request data.
117         self.method = None
118         self.request_path = None
119         self.http_version = None
120         self.headers = {}
121         self.data = {}
122         
123         # Set initial state.
124         self.state = self.HTTP_NONE        
125
126     def handle(self, line):
127         """
128         Dispatch line to current state handler.
129         """
130         for state in self.HTTP_STATES:
131             if getattr(self, 'HTTP_%s' % state) == self.state:
132                 getattr(self, 'handle%s' % state.title())(line)
133                 break
134         else:
135             raise HttpParseError('Unknown HTTP state')
136                 
137     def handleNone(self, line):
138         """
139         Pass line to next state.
140         """
141         self.state += 1
142         self.handle(line)
143
144     def handleRequest(self, line):
145         """
146         Retrieve HTTP method, request path and HTTP version from request.
147         """
148         self.method, self.request_path, self.http_version = line.split(' ')
149         self.state += 1
150
151     def handleHeaders(self, line):
152         """
153         Parse headers while not found an empty line.
154         """
155         if line:
156             key, value = line.split(': ')
157             self.headers[key] = value
158         else:
159             self.state += 1
160             self.handle(line)
161
162     def handleEmpty(self, line):
163         """
164         Empty line separator is found - proceed to next state.
165         """
166         self.state += 1
167
168     def handleMessage(self, line):
169         """
170         Append to message body.
171         """
172         self.data = dict(pair.split('=', 2) for pair in line.split('&'))
173
174         for k, v in self.data.items():
175             print k, '=>', v
176         print
177
178         for generator in self.parent.generators:
179             if generator.canHandle(self.data):
180                 self.state += 1
181                 print generator.generateConfig(self.headers)
182         else:
183             print 'No generator found'
184             
185
186
187 class FreeswitchConfigGenerator(object):
188     """
189     Base class for generating XML configs.
190     """
191     
192     param_match = {}
193     section_name = None
194
195     def __init__(self, model):
196         self.model = model
197
198     def canHandle(self, params):
199         for key, value in self.param_match.iteritems():
200             print key, value, params.get(key, None)
201             if params.get(key, None) != value:
202                 return False
203         else:
204             return True
205
206     def baseElements(self):
207         root_elt = etree.Element('document')
208         section_elt = etree.SubElement(
209             root_elt, 'section', name=self.section_name)
210         return root_elt, section_elt
211     baseElements = property(baseElements)
212
213     def generateConfig(self, params):
214         return NotImplemented
215
216     def addParams(parent_elt, params):
217         for name, value in params:
218             etree.SubElement(parent_elt, 'param', name=name, value=value)
219             
220         
221 class SofiaConfGenerator(FreeswitchConfigGenerator):
222     """
223     Generates sofia.conf.xml config file.
224     """
225     param_match = {'section': 'configuration', 'key_value': 'sofia.conf'}
226     section_name = 'configuration'
227     config_name = 'sofia.conf'
228
229     def generateConfig(self, params):
230         # Get base elements.
231         root_elt, section_elt = self.baseElements
232
233         # Create configuration, settings and profiles elements.
234         configuration_elt = etree.SubElement(
235             section_elt, 'configuration', name=self.config_name,
236             description='%s config' % self.config_name)
237         settings_elt = etree.SubElement(configuration_elt, 'settings')
238         profiles_elt = etree.SubElement(self.settings_elt, 'profiles')
239
240         # Create all profiles for current host.
241         for profile in self.parent.get_profiles():
242             profile_elt = etree.SubElement(profiles_elt, 'profile')
243
244             # Create domains for current profile.
245             domains_elt = etree.SubElement(profile_elt, 'domains')
246             for domain in self.parent.get_domains_for_profile(profile):
247                 domain_elt = etree.SubElement(
248                     domains_elt, 'domain', name=domain.host_name,
249                     alias='true', parse='true')
250
251             # Create settings for current profile.
252             settings_elt = etree.SubElement(profile_elt, 'settings')
253             params = (
254                 ('dialplan', 'XML,enum'),
255                 ('ext-sip-ip', profile.ext_sip_ip),
256                 ('ext-rtp-ip', profile.ext_rtp_ip),
257                 ('sip-ip', profile.sip_ip),
258                 ('rtp-ip', profile.rtp_ip),
259                 ('sip-port', profile.sip_port),
260                 ('nonce-ttl', '60'),
261                 ('rtp-timer-name', 'soft'),
262                 ('codec-prefs', 'PCMU@20i'),
263                 ('debug', '1'),
264                 ('rfc2833-pt', '1'),
265                 ('dtmf-duration', '100'),
266                 ('codec-ms', '20'),
267                 ('accept-blind-reg', profile.accept_blind_registration),
268                 ('auth-calls', profile.authenticate_calls))
269             self.add_params(settings_elt, params)
270
271             # Create gateways for current profile.
272             gateways_elt = etree.SubElement(profile, 'gateways')
273             for gateway in self.parent.get_gateways_for_profile(profile):
274                 gateway_elt = etree.SubElement(gateways_elt, 'gateway', name=gateway.name)
275                 params = (
276                     ('username', gateway.username),
277                     ('realm', gateway.realm),
278                     ('from-domain', gateway.from_domain),
279                     ('password', gateway.password),
280                     ('retry-seconds', gateway.retry_seconds),
281                     ('expire-seconds', gateway.expire_seconds),
282                     ('caller-id-in-from', gateway.caller_id_in_from),
283                     ('extension', gateway.extension),
284                     # TODO: proxy, register
285                     ('expire-seconds', gateway.expire_seconds),
286                     ('retry-seconds', gateway.retry_seconds))
287                 self.add_params(gateway_elt, params)
288
289         return root_elt