1 # -*- encoding: utf-8 -*-
4 # Copyright P. Christeas <p_christ@hol.gr> 2008,2009
7 # WARNING: This program as such is intended to be used by professional
8 # programmers who take the whole responsability of assessing all potential
9 # consequences resulting from its eventual inadequacies and bugs
10 # End users who are looking for a ready-to-use solution with commercial
11 # garantees and support are strongly adviced to contract a Free Software
14 # This program is Free Software; you can redistribute it and/or
15 # modify it under the terms of the GNU General Public License
16 # as published by the Free Software Foundation; either version 2
17 # of the License, or (at your option) any later version.
19 # This program is distributed in the hope that it will be useful,
20 # but WITHOUT ANY WARRANTY; without even the implied warranty of
21 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 # GNU General Public License for more details.
24 # You should have received a copy of the GNU General Public License
25 # along with this program; if not, write to the Free Software
26 # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
27 ###############################################################################
32 from dav_fs import openerp_dav_handler
33 from tools.config import config
34 from DAV.WebDAVServer import DAVRequestHandler
35 from service.websrv_lib import HTTPDir, FixSendError, HttpOptions
36 from BaseHTTPServer import BaseHTTPRequestHandler
39 from string import atoi,split
40 from DAV.errors import *
41 # from DAV.constants import DAV_VERSION_1, DAV_VERSION_2
43 def OpenDAVConfig(**kw):
45 def __init__(self, **kw):
46 self.__dict__.update(**kw)
48 def getboolean(self, word):
49 return self.__dict__.get(word, False)
57 class DAVHandler(HttpOptions, FixSendError, DAVRequestHandler):
59 protocol_version = 'HTTP/1.1'
60 _HTTP_OPTIONS= { 'DAV' : ['1',],
61 'Allow' : [ 'GET', 'HEAD', 'COPY', 'MOVE', 'POST', 'PUT',
62 'PROPFIND', 'PROPPATCH', 'OPTIONS', 'MKCOL',
63 'DELETE', 'TRACE', 'REPORT', ]
66 def get_userinfo(self,user,pw):
68 def _log(self, message):
69 netsvc.Logger().notifyChannel("webdav",netsvc.LOG_DEBUG,message)
77 def get_db_from_path(self, uri):
78 # interface class will handle all cases.
79 res = self.IFACE_CLASS.get_db(uri, allow_last=True)
83 self.davpath = '/'+config.get_misc('webdav','vdir','webdav')
84 self.baseuri = "http://%s:%d/"% (self.server.server_name, self.server.server_port)
85 self.IFACE_CLASS = openerp_dav_handler(self, self.verbose)
87 def copymove(self, CLASS):
88 """ Our uri scheme removes the /webdav/ component from there, so we
89 need to mangle the header, too.
91 dest = self.headers['Destination']
92 up = urlparse.urlparse(urllib.unquote(self.headers['Destination']))
93 if up.path.startswith(self.davpath):
94 self.headers['Destination'] = up.path[len(self.davpath):]
96 raise DAV_Forbidden("Not allowed to copy/move outside webdav path")
97 DAVRequestHandler.copymove(self, CLASS)
99 def get_davpath(self):
102 def log_message(self, format, *args):
103 netsvc.Logger().notifyChannel('webdav', netsvc.LOG_DEBUG_RPC, format % args)
105 def log_error(self, format, *args):
106 netsvc.Logger().notifyChannel('xmlrpc', netsvc.LOG_WARNING, format % args)
108 def _prep_OPTIONS(self, opts):
111 uri=urlparse.urljoin(self.get_baseuri(dc), self.path)
112 uri=urllib.unquote(uri)
114 ret = dc.prep_http_options(uri, opts)
115 except DAV_Error, (ec,dd):
118 self.log_error("Error at options: %s", str(e))
122 def send_response(self, code, message=None):
123 # the BufferingHttpServer will send Connection: close , while
124 # the BaseHTTPRequestHandler will only accept int code.
125 # workaround both of them.
126 BaseHTTPRequestHandler.send_response(self, int(code), message)
128 def send_header(self, key, value):
129 if key == 'Connection' and value == 'close':
130 self.close_connection = 1
131 DAVRequestHandler.send_header(self, key, value)
133 def send_body(self, DATA, code = None, msg = None, desc = None, ctype='application/octet-stream', headers=None):
134 if headers and 'Connection' in headers:
136 elif self.request_version in ('HTTP/1.0', 'HTTP/0.9'):
138 elif self.close_connection == 1: # close header already sent
143 if self.headers.get('Connection',False) == 'Keep-Alive':
144 headers['Connection'] = 'keep-alive'
146 DAVRequestHandler.send_body(self, DATA, code=code, msg=msg, desc=desc,
147 ctype=ctype, headers=headers)
151 uri=urlparse.urljoin(self.get_baseuri(dc), self.path)
152 uri=urllib.unquote(uri)
154 if self.headers.has_key('If-Match'):
158 for match in self.headers['If-Match'].split(','):
159 if match.startswith('"') and match.endswith('"'):
166 if dc.match_prop(uri, match, "DAV:", "getetag"):
171 self.send_status(412)
174 # Handle If-None-Match
175 if self.headers.has_key('If-None-Match'):
178 for match in self.headers['If-None-Match'].split(','):
184 if dc.match_prop(uri, match, "DAV:", "getetag"):
189 self.send_status(412)
193 expect = self.headers.get('Expect', '')
194 if (expect.lower() == '100-continue' and
195 self.protocol_version >= 'HTTP/1.1' and
196 self.request_version >= 'HTTP/1.1'):
197 self.send_status(100)
201 body=self._get_body()
203 # locked resources are not allowed to be overwritten
204 if self._l_isLocked(uri):
205 return self.send_body(None, '423', 'Locked', 'Locked')
208 if self.headers.has_key("Content-Type"):
209 ct=self.headers['Content-Type']
211 location = dc.put(uri,body,ct)
212 except DAV_Error, (ec,dd):
213 return self.send_status(ec)
217 if location and isinstance(location, tuple):
219 location = location[0]
220 # note that we have allowed for > 2 elems
222 headers['Location'] = location
226 etag = dc.get_prop(location or uri, "DAV:", "getetag")
228 headers['ETag'] = str(etag)
232 self.send_body(None, '201', 'Created', '', headers=headers)
236 if self.headers.has_key("Content-Length"):
237 l=self.headers['Content-Length']
238 body=self.rfile.read(atoi(l))
243 DAVRequestHandler.do_DELETE(self)
244 except DAV_Error, (ec, dd):
245 return self.send_status(ec)
247 from service.http_server import reg_http_service,OpenERPAuthProvider
249 class DAVAuthProvider(OpenERPAuthProvider):
250 def authenticate(self, db, user, passwd, client_address):
251 """ authenticate, but also allow the False db, meaning to skip
252 authentication when no db is specified.
256 return OpenERPAuthProvider.authenticate(self, db, user, passwd, client_address)
258 from service.http_server import reg_http_service,OpenERPAuthProvider
260 class DAVAuthProvider(OpenERPAuthProvider):
261 def authenticate(self, db, user, passwd, client_address):
262 """ authenticate, but also allow the False db, meaning to skip
263 authentication when no db is specified.
267 return OpenERPAuthProvider.authenticate(self, db, user, passwd, client_address)
271 if (config.get_misc('webdav','enable',True)):
272 directory = '/'+config.get_misc('webdav','vdir','webdav')
274 verbose = config.get_misc('webdav','verbose',True)
275 handler.debug = config.get_misc('webdav','debug',True)
276 _dc = { 'verbose' : verbose,
277 'directory' : directory,
278 'lockemulation' : False,
282 conf = OpenDAVConfig(**_dc)
283 handler._config = conf
284 reg_http_service(HTTPDir(directory,DAVHandler,DAVAuthProvider()))
285 netsvc.Logger().notifyChannel('webdav', netsvc.LOG_INFO, "WebDAV service registered at path: %s/ "% directory)
287 logger = netsvc.Logger()
288 logger.notifyChannel('webdav', netsvc.LOG_ERROR, 'Cannot launch webdav: %s' % e)