| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 |
- # This file is part of Radicale Server - Calendar Server
- # Copyright © 2014 Jean-Marc Martins
- # Copyright © 2012-2016 Guillaume Ayoub
- #
- # This library 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 3 of the License, or
- # (at your option) any later version.
- #
- # This library 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 Radicale. If not, see <http://www.gnu.org/licenses/>.
- """
- Storage backends.
- This module loads the storage backend, according to the storage
- configuration.
- Default storage uses one folder per collection and one file per collection
- entry.
- """
- import json
- import os
- import posixpath
- import shutil
- import sys
- import time
- from contextlib import contextmanager
- from . import config, ical, log
- def _load():
- """Load the storage manager chosen in configuration."""
- storage_type = config.get("storage", "type")
- if storage_type == "multifilesystem":
- module = sys.modules[__name__]
- else:
- __import__(storage_type)
- module = sys.modules[storage_type]
- ical.Collection = module.Collection
- FOLDER = os.path.expanduser(config.get("storage", "filesystem_folder"))
- FILESYSTEM_ENCODING = sys.getfilesystemencoding()
- def is_safe_path_component(path):
- """Check if path is a single component of a POSIX path.
- Check that the path is safe to join too.
- """
- if not path:
- return False
- if posixpath.split(path)[0]:
- return False
- if path in (".", ".."):
- return False
- return True
- def is_safe_filesystem_path_component(path):
- """Check if path is a single component of a filesystem path.
- Check that the path is safe to join too.
- """
- if not path:
- return False
- drive, _ = os.path.splitdrive(path)
- if drive:
- return False
- head, _ = os.path.split(path)
- if head:
- return False
- if path in (os.curdir, os.pardir):
- return False
- return True
- def path_to_filesystem(path):
- """Convert path to a local filesystem path relative to base_folder.
- Conversion is done in a secure manner, or raises ``ValueError``.
- """
- sane_path = ical.sanitize_path(path).strip("/")
- safe_path = FOLDER
- if not sane_path:
- return safe_path
- for part in sane_path.split("/"):
- if not is_safe_filesystem_path_component(part):
- log.LOGGER.debug(
- "Can't translate path safely to filesystem: %s", path)
- raise ValueError("Unsafe path")
- safe_path = os.path.join(safe_path, part)
- return safe_path
- @contextmanager
- def _open(path, mode="r"):
- """Open a file at ``path`` with encoding set in the configuration."""
- abs_path = os.path.join(FOLDER, path.replace("/", os.sep))
- with open(abs_path, mode, encoding=config.get("encoding", "stock")) as fd:
- yield fd
- class Collection(ical.Collection):
- """Collection stored in several files per calendar."""
- @property
- def _filesystem_path(self):
- """Absolute path of the file at local ``path``."""
- return path_to_filesystem(self.path)
- @property
- def _props_path(self):
- """Absolute path of the file storing the collection properties."""
- return self._filesystem_path + ".props"
- def _create_dirs(self):
- """Create folder storing the collection if absent."""
- if not os.path.exists(self._filesystem_path):
- os.makedirs(self._filesystem_path)
- def set_mimetype(self, mimetype):
- self._create_dirs()
- return super().set_mimetype(mimetype)
- def save(self, text):
- self._create_dirs()
- item_types = (
- ical.Timezone, ical.Event, ical.Todo, ical.Journal, ical.Card)
- for name, component in self._parse(text, item_types).items():
- if not is_safe_filesystem_path_component(name):
- log.LOGGER.debug(
- "Can't tranlate name safely to filesystem, "
- "skipping component: %s", name)
- continue
- filename = os.path.join(self._filesystem_path, name)
- with _open(filename, "w") as fd:
- fd.write(component.text)
- @property
- def headers(self):
- return (
- ical.Header("PRODID:-//Radicale//NONSGML Radicale Server//EN"),
- ical.Header("VERSION:%s" % self.version))
- def delete(self):
- shutil.rmtree(self._filesystem_path)
- os.remove(self._props_path)
- def remove(self, name):
- if not is_safe_filesystem_path_component(name):
- log.LOGGER.debug(
- "Can't tranlate name safely to filesystem, "
- "skipping component: %s", name)
- return
- if name in self.items:
- del self.items[name]
- filesystem_path = os.path.join(self._filesystem_path, name)
- if os.path.exists(filesystem_path):
- os.remove(filesystem_path)
- @property
- def text(self):
- components = (
- ical.Timezone, ical.Event, ical.Todo, ical.Journal, ical.Card)
- items = {}
- try:
- filenames = os.listdir(self._filesystem_path)
- except (OSError, IOError) as e:
- log.LOGGER.info(
- "Error while reading collection %r: %r" % (
- self._filesystem_path, e))
- return ""
- for filename in filenames:
- path = os.path.join(self._filesystem_path, filename)
- try:
- with _open(path) as fd:
- items.update(self._parse(fd.read(), components))
- except (OSError, IOError) as e:
- log.LOGGER.warning(
- "Error while reading item %r: %r" % (path, e))
- return ical.serialize(
- self.tag, self.headers, sorted(items.values(), key=lambda x: x.name))
- @classmethod
- def children(cls, path):
- filesystem_path = path_to_filesystem(path)
- _, directories, files = next(os.walk(filesystem_path))
- for filename in directories + files:
- # make sure that the local filename can be translated
- # into an internal path
- if not is_safe_path_component(filename):
- log.LOGGER.debug("Skipping unsupported filename: %s", filename)
- continue
- rel_filename = posixpath.join(path, filename)
- if cls.is_node(rel_filename) or cls.is_leaf(rel_filename):
- yield cls(rel_filename)
- @classmethod
- def is_node(cls, path):
- filesystem_path = path_to_filesystem(path)
- return (
- os.path.isdir(filesystem_path) and
- not os.path.exists(filesystem_path + ".props"))
- @classmethod
- def is_leaf(cls, path):
- filesystem_path = path_to_filesystem(path)
- return (
- os.path.isdir(filesystem_path) and
- os.path.exists(filesystem_path + ".props"))
- @property
- def last_modified(self):
- last = max([
- os.path.getmtime(os.path.join(self._filesystem_path, filename))
- for filename in os.listdir(self._filesystem_path)] or [0])
- return time.strftime("%a, %d %b %Y %H:%M:%S +0000", time.gmtime(last))
- @property
- @contextmanager
- def props(self):
- # On enter
- properties = {}
- if os.path.exists(self._props_path):
- with open(self._props_path) as prop_file:
- properties.update(json.load(prop_file))
- old_properties = properties.copy()
- yield properties
- # On exit
- if old_properties != properties:
- with open(self._props_path, "w") as prop_file:
- json.dump(properties, prop_file)
|