preliminary iCal/iPhone support introduced

This commit is contained in:
Lukasz Langa 2011-06-01 12:43:49 +02:00
parent 911cd48efe
commit e05e94a129
3 changed files with 201 additions and 113 deletions

@ -29,7 +29,6 @@ should have been included in this package.
""" """
import os import os
import posixpath
import pprint import pprint
import base64 import base64
import socket import socket
@ -151,28 +150,19 @@ class Application(object):
else: else:
content = None content = None
# Find calendar # Find calendar(s)
attributes = posixpath.normpath( items = ical.Calendar.from_path(environ["PATH_INFO"],
environ["PATH_INFO"].strip("/")).split("/") environ.get("HTTP_DEPTH", "0"))
if attributes:
if attributes[-1].endswith(".ics"):
attributes.pop()
path = "/".join(attributes[:min(len(attributes), 2)])
calendar = ical.Calendar(path)
else:
calendar = None
# Get function corresponding to method # Get function corresponding to method
function = getattr(self, environ["REQUEST_METHOD"].lower()) function = getattr(self, environ["REQUEST_METHOD"].lower())
# Check rights # Check rights
if not calendar or not self.acl: if not items or not self.acl:
# No calendar or no acl, don't check rights # No calendar or no acl, don't check rights
status, headers, answer = function(environ, calendar, content) status, headers, answer = function(environ, items, content)
else: else:
# Ask authentication backend to check rights # Ask authentication backend to check rights
log.LOGGER.info(
"Checking rights for calendar owned by %s" % calendar.owner)
authorization = environ.get("HTTP_AUTHORIZATION", None) authorization = environ.get("HTTP_AUTHORIZATION", None)
if authorization: if authorization:
@ -182,11 +172,27 @@ class Application(object):
else: else:
user = password = None user = password = None
last_allowed = False
calendars = []
for calendar in items:
if not isinstance(calendar, ical.Calendar):
if last_allowed:
calendars.append(calendar)
continue
log.LOGGER.info(
"Checking rights for calendar owned by %s" % calendar.owner)
if self.acl.has_right(calendar.owner, user, password): if self.acl.has_right(calendar.owner, user, password):
log.LOGGER.info("%s allowed" % (user or "anonymous user")) log.LOGGER.info("%s allowed" % (user or "anonymous user"))
status, headers, answer = function(environ, calendar, content) calendars.append(calendar)
last_allowed = True
else: else:
log.LOGGER.info("%s refused" % (user or "anonymous user")) log.LOGGER.info("%s refused" % (user or "anonymous user"))
last_allowed = False
if calendars:
status, headers, answer = function(environ, calendars, content)
else:
status = client.UNAUTHORIZED status = client.UNAUTHORIZED
headers = { headers = {
"WWW-Authenticate": "WWW-Authenticate":
@ -209,8 +215,9 @@ class Application(object):
# All these functions must have the same parameters, some are useless # All these functions must have the same parameters, some are useless
# pylint: disable=W0612,W0613,R0201 # pylint: disable=W0612,W0613,R0201
def get(self, environ, calendar, content): def get(self, environ, calendars, content):
"""Manage GET request.""" """Manage GET request."""
calendar = calendars[0]
item_name = xmlutils.name_from_path(environ["PATH_INFO"], calendar) item_name = xmlutils.name_from_path(environ["PATH_INFO"], calendar)
if item_name: if item_name:
# Get calendar item # Get calendar item
@ -235,13 +242,14 @@ class Application(object):
answer = answer_text.encode(self.encoding) answer = answer_text.encode(self.encoding)
return client.OK, headers, answer return client.OK, headers, answer
def head(self, environ, calendar, content): def head(self, environ, calendars, content):
"""Manage HEAD request.""" """Manage HEAD request."""
status, headers, answer = self.get(environ, calendar, content) status, headers, answer = self.get(environ, calendars, content)
return status, headers, None return status, headers, None
def delete(self, environ, calendar, content): def delete(self, environ, calendars, content):
"""Manage DELETE request.""" """Manage DELETE request."""
calendar = calendars[0]
item = calendar.get_item( item = calendar.get_item(
xmlutils.name_from_path(environ["PATH_INFO"], calendar)) xmlutils.name_from_path(environ["PATH_INFO"], calendar))
if item and environ.get("HTTP_IF_MATCH", item.etag) == item.etag: if item and environ.get("HTTP_IF_MATCH", item.etag) == item.etag:
@ -254,8 +262,9 @@ class Application(object):
status = client.PRECONDITION_FAILED status = client.PRECONDITION_FAILED
return status, {}, answer return status, {}, answer
def mkcalendar(self, environ, calendar, content): def mkcalendar(self, environ, calendars, content):
"""Manage MKCALENDAR request.""" """Manage MKCALENDAR request."""
calendar = calendars[0]
props = xmlutils.props_from_request(content) props = xmlutils.props_from_request(content)
tz = props.get('C:calendar-timezone') tz = props.get('C:calendar-timezone')
if tz: if tz:
@ -267,7 +276,7 @@ class Application(object):
calendar.write() calendar.write()
return client.CREATED, {}, None return client.CREATED, {}, None
def options(self, environ, calendar, content): def options(self, environ, calendars, content):
"""Manage OPTIONS request.""" """Manage OPTIONS request."""
headers = { headers = {
"Allow": "DELETE, HEAD, GET, MKCALENDAR, " \ "Allow": "DELETE, HEAD, GET, MKCALENDAR, " \
@ -275,26 +284,27 @@ class Application(object):
"DAV": "1, calendar-access"} "DAV": "1, calendar-access"}
return client.OK, headers, None return client.OK, headers, None
def propfind(self, environ, calendar, content): def propfind(self, environ, calendars, content):
"""Manage PROPFIND request.""" """Manage PROPFIND request."""
headers = { headers = {
"DAV": "1, calendar-access", "DAV": "1, calendar-access",
"Content-Type": "text/xml"} "Content-Type": "text/xml"}
answer = xmlutils.propfind( answer = xmlutils.propfind(
environ["PATH_INFO"], content, calendar, environ["PATH_INFO"], content, calendars)
environ.get("HTTP_DEPTH", "infinity"))
return client.MULTI_STATUS, headers, answer return client.MULTI_STATUS, headers, answer
def proppatch(self, environ, calendar, content): def proppatch(self, environ, calendars, content):
"""Manage PROPPATCH request.""" """Manage PROPPATCH request."""
calendar = calendars[0]
answer = xmlutils.proppatch(environ["PATH_INFO"], content, calendar) answer = xmlutils.proppatch(environ["PATH_INFO"], content, calendar)
headers = { headers = {
"DAV": "1, calendar-access", "DAV": "1, calendar-access",
"Content-Type": "text/xml"} "Content-Type": "text/xml"}
return client.MULTI_STATUS, headers, answer return client.MULTI_STATUS, headers, answer
def put(self, environ, calendar, content): def put(self, environ, calendars, content):
"""Manage PUT request.""" """Manage PUT request."""
calendar = calendars[0]
headers = {} headers = {}
item_name = xmlutils.name_from_path(environ["PATH_INFO"], calendar) item_name = xmlutils.name_from_path(environ["PATH_INFO"], calendar)
item = calendar.get_item(item_name) item = calendar.get_item(item_name)
@ -312,8 +322,10 @@ class Application(object):
status = client.PRECONDITION_FAILED status = client.PRECONDITION_FAILED
return status, headers, None return status, headers, None
def report(self, environ, calendar, content): def report(self, environ, calendars, content):
"""Manage REPORT request.""" """Manage REPORT request."""
# TODO: support multiple calendars here
calendar = calendars[0]
headers = {'Content-Type': 'text/xml'} headers = {'Content-Type': 'text/xml'}
answer = xmlutils.report(environ["PATH_INFO"], content, calendar) answer = xmlutils.report(environ["PATH_INFO"], content, calendar)
return client.MULTI_STATUS, headers, answer return client.MULTI_STATUS, headers, answer

@ -29,6 +29,7 @@ import codecs
from contextlib import contextmanager from contextlib import contextmanager
import json import json
import os import os
import posixpath
import time import time
from radicale import config from radicale import config
@ -155,13 +156,60 @@ class Calendar(object):
"""Internal calendar class.""" """Internal calendar class."""
tag = "VCALENDAR" tag = "VCALENDAR"
def __init__(self, path): def __init__(self, path, principal=False):
"""Initialize the calendar with ``cal`` and ``user`` parameters.""" """Initialize the calendar with ``cal`` and ``user`` parameters."""
self.encoding = "utf-8" self.encoding = "utf-8"
split_path = path.split("/") split_path = path.split("/")
self.owner = split_path[0] if len(split_path) > 1 else None self.owner = split_path[0] if len(split_path) > 1 else None
self.path = os.path.join(FOLDER, path.replace("/", os.path.sep)) self.path = os.path.join(FOLDER, path.replace("/", os.sep))
self.local_path = path self.local_path = path
self.is_principal = principal
@classmethod
def from_path(cls, path, depth="infinite", include_container=True):
"""Return a list of calendars and/or sub-items under the given ``path``
relative to the storage folder. If ``depth`` is "0", only the actual
object under `path` is returned. Otherwise, also sub-items are appended
to the result. If `include_container` is True (the default), the
containing object is included in the result.
"""
attributes = posixpath.normpath(path.strip("/")).split("/")
if not attributes:
return None
if attributes[-1].endswith(".ics"):
attributes.pop()
result = []
path = "/".join(attributes[:min(len(attributes), 2)])
path = path.replace("/", os.sep)
abs_path = os.path.join(FOLDER, path)
if os.path.isdir(abs_path):
if depth == "0":
result.append(cls(path, principal=True))
else:
if include_container:
result.append(cls(path, principal=True))
for f in os.walk(abs_path).next()[2]:
f_path = os.path.join(path, f)
if cls.is_vcalendar(os.path.join(abs_path, f)):
result.append(cls(f_path))
else:
calendar = cls(path)
if depth == "0":
result.append(calendar)
else:
if include_container:
result.append(calendar)
result.extend(calendar.components)
return result
@staticmethod
def is_vcalendar(path):
"""Return `True` if there is a VCALENDAR file under `path`."""
with open(path) as f:
return 'BEGIN:VCALENDAR' == f.read(15)
@staticmethod @staticmethod
def _parse(text, item_types, name=None): def _parse(text, item_types, name=None):
@ -340,3 +388,7 @@ class Calendar(object):
# on exit # on exit
with open(props_path, 'w') as prop_file: with open(props_path, 'w') as prop_file:
json.dump(properties, prop_file) json.dump(properties, prop_file)
@property
def url(self):
return '/{}/'.format(self.local_path).replace('//', '/')

@ -42,7 +42,8 @@ NAMESPACES = {
"C": "urn:ietf:params:xml:ns:caldav", "C": "urn:ietf:params:xml:ns:caldav",
"D": "DAV:", "D": "DAV:",
"CS": "http://calendarserver.org/ns/", "CS": "http://calendarserver.org/ns/",
"ICAL": "http://apple.com/ns/ical/"} "ICAL": "http://apple.com/ns/ical/",
"ME": "http://me.com/_namespace/"}
NAMESPACES_REV = {} NAMESPACES_REV = {}
@ -104,10 +105,8 @@ def _tag_from_clark(name):
args = { args = {
'ns': NAMESPACES_REV[match.group('namespace')], 'ns': NAMESPACES_REV[match.group('namespace')],
'tag': match.group('tag')} 'tag': match.group('tag')}
tag_name = '%(ns)s:%(tag)s' % args return '%(ns)s:%(tag)s' % args
else: return name
tag_name = prop.tag
return tag_name
def _response(code): def _response(code):
@ -168,7 +167,7 @@ def delete(path, calendar):
return _pretty_xml(multistatus) return _pretty_xml(multistatus)
def propfind(path, xml_request, calendar, depth): def propfind(path, xml_request, calendars):
"""Read and answer PROPFIND requests. """Read and answer PROPFIND requests.
Read rfc4918-9.1 for info. Read rfc4918-9.1 for info.
@ -183,53 +182,51 @@ def propfind(path, xml_request, calendar, depth):
# Writing answer # Writing answer
multistatus = ET.Element(_tag("D", "multistatus")) multistatus = ET.Element(_tag("D", "multistatus"))
if calendar: for calendar in calendars:
if depth == "0": response = _propfind_response(path, calendar, props)
items = [calendar]
else:
# Depth is 1, infinity or not specified
# We limit ourselves to depth == 1
items = [calendar] + calendar.components
else:
items = []
for item in items:
is_calendar = isinstance(item, ical.Calendar)
response = ET.Element(_tag("D", "response"))
multistatus.append(response) multistatus.append(response)
return _pretty_xml(multistatus)
def _propfind_response(path, item, props):
is_calendar = isinstance(item, ical.Calendar)
if is_calendar:
with item.props as cal_props:
calendar_props = cal_props
response = ET.Element(_tag("D", "response"))
href = ET.Element(_tag("D", "href")) href = ET.Element(_tag("D", "href"))
href.text = path if is_calendar else path + item.name href.text = item.url if is_calendar else path + item.name
response.append(href) response.append(href)
propstat = ET.Element(_tag("D", "propstat")) propstat404 = ET.Element(_tag("D", "propstat"))
response.append(propstat) propstat200 = ET.Element(_tag("D", "propstat"))
response.append(propstat200)
prop = ET.Element(_tag("D", "prop")) prop200 = ET.Element(_tag("D", "prop"))
propstat.append(prop) propstat200.append(prop200)
prop404 = ET.Element(_tag("D", "prop"))
propstat404.append(prop404)
for tag in props: for tag in props:
element = ET.Element(tag) element = ET.Element(tag)
if tag == _tag("D", "resourcetype") and is_calendar: is404 = False
tag = ET.Element(_tag("C", "calendar")) if tag == _tag("D", "owner"):
element.append(tag) if item.owner:
tag = ET.Element(_tag("D", "collection")) element.text = item.owner
element.append(tag)
elif tag == _tag("D", "owner"):
if calendar.owner:
element.text = calendar.owner
elif tag == _tag("D", "getcontenttype"): elif tag == _tag("D", "getcontenttype"):
element.text = "text/calendar" element.text = "text/calendar"
elif tag == _tag("CS", "getctag") and is_calendar:
element.text = item.etag
elif tag == _tag("D", "getetag"): elif tag == _tag("D", "getetag"):
element.text = item.etag element.text = item.etag
elif tag == _tag("D", "displayname") and is_calendar:
element.text = calendar.name
elif tag == _tag("D", "principal-URL"): elif tag == _tag("D", "principal-URL"):
# TODO: use a real principal URL, read rfc3744-4.2 for info # TODO: use a real principal URL, read rfc3744-4.2 for info
tag = ET.Element(_tag("D", "href")) tag = ET.Element(_tag("D", "href"))
if item.owner:
tag.text = "/{}/".format(item.owner).replace("//", "/")
else:
tag.text = path tag.text = path
element.append(tag) element.append(tag)
elif tag in ( elif tag in (
@ -260,13 +257,40 @@ def propfind(path, xml_request, calendar, depth):
report_tag.text = report_name report_tag.text = report_name
supported.append(report_tag) supported.append(report_tag)
element.append(supported) element.append(supported)
prop.append(element) elif is_calendar:
if tag == _tag("D", "resourcetype"):
if is_calendar and not item.is_principal:
tag = ET.Element(_tag("C", "calendar"))
element.append(tag)
tag = ET.Element(_tag("D", "collection"))
element.append(tag)
elif tag == _tag("CS", "getctag"):
element.text = item.etag
else:
human_tag = _tag_from_clark(tag)
if human_tag in calendar_props:
element.text = calendar_props[human_tag]
else:
is404 = True
else:
is404 = True
status = ET.Element(_tag("D", "status")) if is404:
status.text = _response(200) prop404.append(element)
propstat.append(status) else:
prop200.append(element)
return _pretty_xml(multistatus) status200 = ET.Element(_tag("D", "status"))
status200.text = _response(200)
propstat200.append(status200)
status404 = ET.Element(_tag("D", "status"))
status404.text = _response(404)
propstat404.append(status404)
if len(prop404):
response.append(propstat404)
return response
def _add_propstat_to(element, tag, status_number): def _add_propstat_to(element, tag, status_number):