2009-11-26 13:54:00 +00:00
|
|
|
# -*- encoding: utf-8 -*-
|
|
|
|
|
|
|
|
#
|
|
|
|
# Copyright P. Christeas <p_christ@hol.gr> 2008,2009
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# WARNING: This program as such is intended to be used by professional
|
|
|
|
# programmers who take the whole responsability of assessing all potential
|
|
|
|
# consequences resulting from its eventual inadequacies and bugs
|
|
|
|
# End users who are looking for a ready-to-use solution with commercial
|
|
|
|
# garantees and support are strongly adviced to contract a Free Software
|
|
|
|
# Service Company
|
|
|
|
#
|
|
|
|
# This program is Free Software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License
|
|
|
|
# as published by the Free Software Foundation; either version 2
|
|
|
|
# of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
|
|
import netsvc
|
2010-04-06 11:11:32 +00:00
|
|
|
import tools
|
2010-04-12 10:52:00 +00:00
|
|
|
from dav_fs import openerp_dav_handler
|
2009-11-26 13:54:00 +00:00
|
|
|
from tools.config import config
|
|
|
|
from DAV.WebDAVServer import DAVRequestHandler
|
2010-07-29 13:39:36 +00:00
|
|
|
from service.websrv_lib import HTTPDir, FixSendError, HttpOptions
|
2010-06-24 15:00:54 +00:00
|
|
|
from BaseHTTPServer import BaseHTTPRequestHandler
|
2010-04-21 13:34:02 +00:00
|
|
|
import urlparse
|
|
|
|
import urllib
|
|
|
|
from string import atoi,split
|
|
|
|
from DAV.errors import *
|
2010-07-29 13:39:36 +00:00
|
|
|
# from DAV.constants import DAV_VERSION_1, DAV_VERSION_2
|
2010-04-06 11:11:32 +00:00
|
|
|
|
|
|
|
def OpenDAVConfig(**kw):
|
|
|
|
class OpenDAV:
|
|
|
|
def __init__(self, **kw):
|
|
|
|
self.__dict__.update(**kw)
|
2010-08-19 11:51:57 +00:00
|
|
|
|
2010-06-22 15:03:39 +00:00
|
|
|
def getboolean(self, word):
|
|
|
|
return self.__dict__.get(word, False)
|
2010-04-06 11:11:32 +00:00
|
|
|
|
|
|
|
class Config:
|
|
|
|
DAV = OpenDAV(**kw)
|
|
|
|
|
|
|
|
return Config()
|
2009-11-26 13:54:00 +00:00
|
|
|
|
2010-04-12 10:52:00 +00:00
|
|
|
|
2010-07-29 13:39:36 +00:00
|
|
|
class DAVHandler(HttpOptions, FixSendError, DAVRequestHandler):
|
2009-11-26 13:54:00 +00:00
|
|
|
verbose = False
|
2010-06-24 15:00:54 +00:00
|
|
|
protocol_version = 'HTTP/1.1'
|
2010-10-12 10:40:35 +00:00
|
|
|
_HTTP_OPTIONS= { 'DAV' : ['1', '2'],
|
2010-08-19 11:51:57 +00:00
|
|
|
'Allow' : [ 'GET', 'HEAD', 'COPY', 'MOVE', 'POST', 'PUT',
|
2010-07-29 13:39:36 +00:00
|
|
|
'PROPFIND', 'PROPPATCH', 'OPTIONS', 'MKCOL',
|
|
|
|
'DELETE', 'TRACE', 'REPORT', ]
|
|
|
|
}
|
|
|
|
|
2009-11-26 13:54:00 +00:00
|
|
|
def get_userinfo(self,user,pw):
|
2009-12-15 14:31:05 +00:00
|
|
|
return False
|
2009-11-26 13:54:00 +00:00
|
|
|
def _log(self, message):
|
2009-12-15 14:31:05 +00:00
|
|
|
netsvc.Logger().notifyChannel("webdav",netsvc.LOG_DEBUG,message)
|
2010-08-19 11:51:57 +00:00
|
|
|
|
2009-11-26 13:54:00 +00:00
|
|
|
def handle(self):
|
2010-04-06 11:11:32 +00:00
|
|
|
self._init_buffer()
|
2009-11-26 13:54:00 +00:00
|
|
|
|
|
|
|
def finish(self):
|
|
|
|
pass
|
|
|
|
|
2010-04-06 11:11:32 +00:00
|
|
|
def get_db_from_path(self, uri):
|
2010-06-23 10:54:37 +00:00
|
|
|
# interface class will handle all cases.
|
|
|
|
res = self.IFACE_CLASS.get_db(uri, allow_last=True)
|
2010-04-06 11:11:32 +00:00
|
|
|
return res
|
|
|
|
|
2009-11-26 13:54:00 +00:00
|
|
|
def setup(self):
|
2010-06-23 10:54:37 +00:00
|
|
|
self.davpath = '/'+config.get_misc('webdav','vdir','webdav')
|
2010-10-12 11:20:30 +00:00
|
|
|
addr, port = self.server.server_name, self.server.server_port
|
|
|
|
server_proto = getattr(self.server,'proto', 'http').lower()
|
|
|
|
try:
|
2010-10-12 11:55:34 +00:00
|
|
|
if hasattr(self.request, 'getsockname'):
|
|
|
|
addr, port = self.request.getsockname()
|
2010-10-12 11:20:30 +00:00
|
|
|
except Exception, e:
|
2010-10-12 11:55:34 +00:00
|
|
|
self.log_error("Cannot calculate own address: %s" , e)
|
2010-10-12 11:20:30 +00:00
|
|
|
# Too early here to use self.headers
|
|
|
|
self.baseuri = "%s://%s:%d/"% (server_proto, addr, port)
|
2010-04-12 10:52:00 +00:00
|
|
|
self.IFACE_CLASS = openerp_dav_handler(self, self.verbose)
|
2010-08-19 11:51:57 +00:00
|
|
|
|
2010-07-12 19:32:36 +00:00
|
|
|
def copymove(self, CLASS):
|
|
|
|
""" Our uri scheme removes the /webdav/ component from there, so we
|
|
|
|
need to mangle the header, too.
|
|
|
|
"""
|
|
|
|
dest = self.headers['Destination']
|
|
|
|
up = urlparse.urlparse(urllib.unquote(self.headers['Destination']))
|
|
|
|
if up.path.startswith(self.davpath):
|
|
|
|
self.headers['Destination'] = up.path[len(self.davpath):]
|
|
|
|
else:
|
|
|
|
raise DAV_Forbidden("Not allowed to copy/move outside webdav path")
|
|
|
|
DAVRequestHandler.copymove(self, CLASS)
|
|
|
|
|
2010-06-23 10:54:37 +00:00
|
|
|
def get_davpath(self):
|
|
|
|
return self.davpath
|
2010-08-19 11:51:57 +00:00
|
|
|
|
2009-11-26 13:54:00 +00:00
|
|
|
def log_message(self, format, *args):
|
2010-06-16 11:51:39 +00:00
|
|
|
netsvc.Logger().notifyChannel('webdav', netsvc.LOG_DEBUG_RPC, format % args)
|
2009-11-26 13:54:00 +00:00
|
|
|
|
|
|
|
def log_error(self, format, *args):
|
2010-06-16 11:51:39 +00:00
|
|
|
netsvc.Logger().notifyChannel('xmlrpc', netsvc.LOG_WARNING, format % args)
|
2009-11-26 13:54:00 +00:00
|
|
|
|
2010-07-29 13:39:36 +00:00
|
|
|
def _prep_OPTIONS(self, opts):
|
|
|
|
ret = opts
|
|
|
|
dc=self.IFACE_CLASS
|
|
|
|
uri=urlparse.urljoin(self.get_baseuri(dc), self.path)
|
|
|
|
uri=urllib.unquote(uri)
|
|
|
|
try:
|
|
|
|
ret = dc.prep_http_options(uri, opts)
|
|
|
|
except DAV_Error, (ec,dd):
|
|
|
|
pass
|
2010-07-29 13:39:38 +00:00
|
|
|
except Exception,e:
|
|
|
|
self.log_error("Error at options: %s", str(e))
|
|
|
|
raise
|
2010-07-29 13:39:36 +00:00
|
|
|
return ret
|
2010-08-19 11:51:57 +00:00
|
|
|
|
2010-06-24 15:00:54 +00:00
|
|
|
def send_response(self, code, message=None):
|
|
|
|
# the BufferingHttpServer will send Connection: close , while
|
|
|
|
# the BaseHTTPRequestHandler will only accept int code.
|
|
|
|
# workaround both of them.
|
|
|
|
BaseHTTPRequestHandler.send_response(self, int(code), message)
|
|
|
|
|
|
|
|
def send_header(self, key, value):
|
|
|
|
if key == 'Connection' and value == 'close':
|
|
|
|
self.close_connection = 1
|
|
|
|
DAVRequestHandler.send_header(self, key, value)
|
|
|
|
|
2010-08-01 08:38:28 +00:00
|
|
|
def send_body(self, DATA, code = None, msg = None, desc = None, ctype='application/octet-stream', headers=None):
|
|
|
|
if headers and 'Connection' in headers:
|
|
|
|
pass
|
|
|
|
elif self.request_version in ('HTTP/1.0', 'HTTP/0.9'):
|
|
|
|
pass
|
|
|
|
elif self.close_connection == 1: # close header already sent
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
if headers is None:
|
|
|
|
headers = {}
|
|
|
|
if self.headers.get('Connection',False) == 'Keep-Alive':
|
|
|
|
headers['Connection'] = 'keep-alive'
|
|
|
|
|
|
|
|
DAVRequestHandler.send_body(self, DATA, code=code, msg=msg, desc=desc,
|
|
|
|
ctype=ctype, headers=headers)
|
|
|
|
|
2010-04-21 13:34:02 +00:00
|
|
|
def do_PUT(self):
|
2010-08-19 11:51:57 +00:00
|
|
|
dc=self.IFACE_CLASS
|
2010-04-21 13:34:02 +00:00
|
|
|
uri=urlparse.urljoin(self.get_baseuri(dc), self.path)
|
|
|
|
uri=urllib.unquote(uri)
|
|
|
|
# Handle If-Match
|
|
|
|
if self.headers.has_key('If-Match'):
|
|
|
|
test = False
|
|
|
|
etag = None
|
2010-08-19 11:51:57 +00:00
|
|
|
|
|
|
|
for match in self.headers['If-Match'].split(','):
|
2010-04-21 13:34:02 +00:00
|
|
|
if match == '*':
|
|
|
|
if dc.exists(uri):
|
|
|
|
test = True
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
if dc.match_prop(uri, match, "DAV:", "getetag"):
|
|
|
|
test = True
|
|
|
|
break
|
|
|
|
if not test:
|
2010-08-01 08:38:28 +00:00
|
|
|
self._get_body()
|
2010-04-21 13:34:02 +00:00
|
|
|
self.send_status(412)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Handle If-None-Match
|
|
|
|
if self.headers.has_key('If-None-Match'):
|
|
|
|
test = True
|
2010-08-19 11:51:57 +00:00
|
|
|
etag = None
|
2010-04-21 13:34:02 +00:00
|
|
|
for match in self.headers['If-None-Match'].split(','):
|
|
|
|
if match == '*':
|
|
|
|
if dc.exists(uri):
|
|
|
|
test = False
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
if dc.match_prop(uri, match, "DAV:", "getetag"):
|
|
|
|
test = False
|
|
|
|
break
|
|
|
|
if not test:
|
2010-08-01 08:38:28 +00:00
|
|
|
self._get_body()
|
2010-04-21 13:34:02 +00:00
|
|
|
self.send_status(412)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Handle expect
|
|
|
|
expect = self.headers.get('Expect', '')
|
|
|
|
if (expect.lower() == '100-continue' and
|
|
|
|
self.protocol_version >= 'HTTP/1.1' and
|
|
|
|
self.request_version >= 'HTTP/1.1'):
|
|
|
|
self.send_status(100)
|
|
|
|
self._flush()
|
|
|
|
|
|
|
|
# read the body
|
2010-08-01 08:38:28 +00:00
|
|
|
body=self._get_body()
|
2010-04-21 13:34:02 +00:00
|
|
|
|
|
|
|
# locked resources are not allowed to be overwritten
|
|
|
|
if self._l_isLocked(uri):
|
|
|
|
return self.send_body(None, '423', 'Locked', 'Locked')
|
|
|
|
|
|
|
|
ct=None
|
|
|
|
if self.headers.has_key("Content-Type"):
|
|
|
|
ct=self.headers['Content-Type']
|
|
|
|
try:
|
2010-10-12 10:38:30 +00:00
|
|
|
location = dc.put(uri, body, ct)
|
2010-04-21 13:34:02 +00:00
|
|
|
except DAV_Error, (ec,dd):
|
2010-10-12 10:38:30 +00:00
|
|
|
self._logger.warning("Cannot PUT to %s: %s", uri, dd, exc_info=True)
|
2010-04-21 13:34:02 +00:00
|
|
|
return self.send_status(ec)
|
|
|
|
|
|
|
|
headers = {}
|
2010-08-12 11:10:23 +00:00
|
|
|
etag = None
|
|
|
|
if location and isinstance(location, tuple):
|
|
|
|
etag = location[1]
|
|
|
|
location = location[0]
|
|
|
|
# note that we have allowed for > 2 elems
|
2010-04-21 13:34:02 +00:00
|
|
|
if location:
|
|
|
|
headers['Location'] = location
|
2010-08-12 11:10:25 +00:00
|
|
|
else:
|
|
|
|
try:
|
|
|
|
if not etag:
|
|
|
|
etag = dc.get_prop(location or uri, "DAV:", "getetag")
|
|
|
|
if etag:
|
|
|
|
headers['ETag'] = str(etag)
|
|
|
|
except Exception:
|
|
|
|
pass
|
2010-04-21 13:34:02 +00:00
|
|
|
|
|
|
|
self.send_body(None, '201', 'Created', '', headers=headers)
|
|
|
|
|
2010-08-01 08:38:28 +00:00
|
|
|
def _get_body(self):
|
|
|
|
body = None
|
|
|
|
if self.headers.has_key("Content-Length"):
|
|
|
|
l=self.headers['Content-Length']
|
|
|
|
body=self.rfile.read(atoi(l))
|
|
|
|
return body
|
|
|
|
|
2010-07-12 19:28:19 +00:00
|
|
|
def do_DELETE(self):
|
|
|
|
try:
|
|
|
|
DAVRequestHandler.do_DELETE(self)
|
|
|
|
except DAV_Error, (ec, dd):
|
|
|
|
return self.send_status(ec)
|
2009-11-26 13:54:00 +00:00
|
|
|
|
2010-06-23 10:54:37 +00:00
|
|
|
from service.http_server import reg_http_service,OpenERPAuthProvider
|
|
|
|
|
|
|
|
class DAVAuthProvider(OpenERPAuthProvider):
|
|
|
|
def authenticate(self, db, user, passwd, client_address):
|
|
|
|
""" authenticate, but also allow the False db, meaning to skip
|
|
|
|
authentication when no db is specified.
|
|
|
|
"""
|
2010-06-24 15:01:22 +00:00
|
|
|
if db is False:
|
2010-06-23 10:54:37 +00:00
|
|
|
return True
|
|
|
|
return OpenERPAuthProvider.authenticate(self, db, user, passwd, client_address)
|
|
|
|
|
2009-11-26 13:54:00 +00:00
|
|
|
try:
|
2010-04-06 11:11:32 +00:00
|
|
|
|
2009-12-29 09:36:55 +00:00
|
|
|
if (config.get_misc('webdav','enable',True)):
|
2010-08-19 11:51:57 +00:00
|
|
|
directory = '/'+config.get_misc('webdav','vdir','webdav')
|
2009-12-15 14:31:05 +00:00
|
|
|
handler = DAVHandler
|
2010-04-06 11:11:32 +00:00
|
|
|
verbose = config.get_misc('webdav','verbose',True)
|
2009-12-15 14:31:05 +00:00
|
|
|
handler.debug = config.get_misc('webdav','debug',True)
|
2010-04-06 11:11:32 +00:00
|
|
|
_dc = { 'verbose' : verbose,
|
|
|
|
'directory' : directory,
|
2010-10-12 10:40:35 +00:00
|
|
|
'lockemulation' : True,
|
2010-04-06 11:11:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
conf = OpenDAVConfig(**_dc)
|
|
|
|
handler._config = conf
|
2010-06-23 10:54:37 +00:00
|
|
|
reg_http_service(HTTPDir(directory,DAVHandler,DAVAuthProvider()))
|
2010-06-16 11:51:39 +00:00
|
|
|
netsvc.Logger().notifyChannel('webdav', netsvc.LOG_INFO, "WebDAV service registered at path: %s/ "% directory)
|
2009-11-26 13:54:00 +00:00
|
|
|
except Exception, e:
|
2009-12-15 14:31:05 +00:00
|
|
|
logger = netsvc.Logger()
|
|
|
|
logger.notifyChannel('webdav', netsvc.LOG_ERROR, 'Cannot launch webdav: %s' % e)
|
2009-11-26 13:54:00 +00:00
|
|
|
|
|
|
|
#eof
|
|
|
|
|
|
|
|
|
|
|
|
|