| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
- # This file is part of Radicale - CalDAV and CardDAV server
- # Copyright © 2014 Jean-Marc Martins
- # Copyright © 2012-2017 Guillaume Ayoub
- # Copyright © 2017-2021 Unrud <unrud@outlook.com>
- # Copyright © 2024-2025 Peter Bieringer <pb@bieringer.de>
- #
- # 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/>.
- import os
- from tempfile import TemporaryDirectory
- from typing import Dict, Iterable, List, Optional, Tuple, cast
- import radicale.item as radicale_item
- from radicale import pathutils
- from radicale.log import logger
- from radicale.storage import multifilesystem
- from radicale.storage.multifilesystem.base import StorageBase
- class StoragePartCreateCollection(StorageBase):
- def _discover_existing_items_pre_overwrite(self,
- tmp_collection: "multifilesystem.Collection",
- dst_path: str) -> Tuple[Dict[str, radicale_item.Item], List[str]]:
- """Discover existing items in the collection before overwriting them."""
- existing_items = {}
- new_item_hrefs = []
- existing_collection = self._collection_class(
- cast(multifilesystem.Storage, self),
- pathutils.unstrip_path(dst_path, True))
- existing_item_hrefs = set(existing_collection._list())
- tmp_collection_hrefs = set(tmp_collection._list())
- for item_href in tmp_collection_hrefs:
- if item_href not in existing_item_hrefs:
- # Item in temporary collection does not exist in the existing collection (is new)
- new_item_hrefs.append(item_href)
- continue
- # Item exists in both collections, grab the existing item for reference
- try:
- item = existing_collection._get(item_href, verify_href=False)
- if item is not None:
- existing_items[item_href] = item
- except Exception:
- # TODO: Log exception?
- continue
- return existing_items, new_item_hrefs
- def create_collection(self, href: str,
- items: Optional[Iterable[radicale_item.Item]] = None,
- props=None) -> Tuple["multifilesystem.Collection", Dict[str, radicale_item.Item], List[str]]:
- folder = self._get_collection_root_folder()
- # Path should already be sanitized
- sane_path = pathutils.strip_path(href)
- filesystem_path = pathutils.path_to_filesystem(folder, sane_path)
- logger.debug("Create collection: %r" % filesystem_path)
- if not props:
- self._makedirs_synced(filesystem_path)
- return self._collection_class(
- cast(multifilesystem.Storage, self),
- pathutils.unstrip_path(sane_path, True)), {}, []
- parent_dir = os.path.dirname(filesystem_path)
- self._makedirs_synced(parent_dir)
- replaced_items: Dict[str, radicale_item.Item] = {}
- new_item_hrefs: List[str] = []
- # Create a temporary directory with an unsafe name
- try:
- with TemporaryDirectory(prefix=".Radicale.tmp-", dir=parent_dir
- ) as tmp_dir:
- # The temporary directory itself can't be renamed
- tmp_filesystem_path = os.path.join(tmp_dir, "collection")
- os.makedirs(tmp_filesystem_path)
- col = self._collection_class(
- cast(multifilesystem.Storage, self),
- pathutils.unstrip_path(sane_path, True),
- filesystem_path=tmp_filesystem_path)
- col.set_meta(props)
- if items is not None:
- if props.get("tag") == "VCALENDAR":
- col._upload_all_nonatomic(items, suffix=".ics")
- elif props.get("tag") == "VADDRESSBOOK":
- col._upload_all_nonatomic(items, suffix=".vcf")
- if os.path.lexists(filesystem_path):
- replaced_items, new_item_hrefs = self._discover_existing_items_pre_overwrite(
- tmp_collection=col,
- dst_path=sane_path)
- pathutils.rename_exchange(tmp_filesystem_path, filesystem_path)
- else:
- # If the destination path does not exist, obviously all items are new
- new_item_hrefs = list(col._list())
- os.rename(tmp_filesystem_path, filesystem_path)
- self._sync_directory(parent_dir)
- except Exception as e:
- raise ValueError("Failed to create collection %r as %r %s" %
- (href, filesystem_path, e)) from e
- # TODO: Return new-old pairs and just-new items (new vs updated)
- return self._collection_class(
- cast(multifilesystem.Storage, self),
- pathutils.unstrip_path(sane_path, True)), replaced_items, new_item_hrefs
|