mirror of
https://github.com/Kozea/Radicale.git
synced 2025-04-04 21:57:43 +03:00
Merge pull request #1695 from pbiering/issue-1693-fix-http-return-codes
Issue 1693 fix http return codes
This commit is contained in:
commit
aa35c678ce
8 changed files with 129 additions and 42 deletions
|
@ -3,6 +3,7 @@
|
||||||
## 3.4.2.dev
|
## 3.4.2.dev
|
||||||
|
|
||||||
* Add: option [auth] type oauth2 by code migration from https://gitlab.mim-libre.fr/alphabet/radicale_oauth/-/blob/dev/oauth2/
|
* Add: option [auth] type oauth2 by code migration from https://gitlab.mim-libre.fr/alphabet/radicale_oauth/-/blob/dev/oauth2/
|
||||||
|
* Fix: catch OS errors on PUT MKCOL MKCALENDAR MOVE PROPPATCH (insufficient storage, access denied, internal server error)
|
||||||
|
|
||||||
## 3.4.1
|
## 3.4.1
|
||||||
* Add: option [auth] dovecot_connection_type / dovecot_host / dovecot_port
|
* Add: option [auth] dovecot_connection_type / dovecot_host / dovecot_port
|
||||||
|
|
|
@ -2,7 +2,8 @@
|
||||||
# Copyright © 2008 Nicolas Kandel
|
# Copyright © 2008 Nicolas Kandel
|
||||||
# Copyright © 2008 Pascal Halter
|
# Copyright © 2008 Pascal Halter
|
||||||
# Copyright © 2008-2017 Guillaume Ayoub
|
# Copyright © 2008-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2018 Unrud <unrud@outlook.com>
|
# 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
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,7 +18,9 @@
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import errno
|
||||||
import posixpath
|
import posixpath
|
||||||
|
import re
|
||||||
import socket
|
import socket
|
||||||
from http import client
|
from http import client
|
||||||
|
|
||||||
|
@ -70,7 +73,20 @@ class ApplicationPartMkcalendar(ApplicationBase):
|
||||||
try:
|
try:
|
||||||
self._storage.create_collection(path, props=props)
|
self._storage.create_collection(path, props=props)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
logger.warning(
|
# return better matching HTTP result in case errno is provided and catched
|
||||||
"Bad MKCALENDAR request on %r: %s", path, e, exc_info=True)
|
errno_match = re.search("\\[Errno ([0-9]+)\\]", str(e))
|
||||||
return httputils.BAD_REQUEST
|
if errno_match:
|
||||||
|
logger.error(
|
||||||
|
"Failed MKCALENDAR request on %r: %s", path, e, exc_info=True)
|
||||||
|
errno_e = int(errno_match.group(1))
|
||||||
|
if errno_e == errno.ENOSPC:
|
||||||
|
return httputils.INSUFFICIENT_STORAGE
|
||||||
|
elif errno_e in [errno.EPERM, errno.EACCES]:
|
||||||
|
return httputils.FORBIDDEN
|
||||||
|
else:
|
||||||
|
return httputils.INTERNAL_SERVER_ERROR
|
||||||
|
else:
|
||||||
|
logger.warning(
|
||||||
|
"Bad MKCALENDAR request on %r: %s", path, e, exc_info=True)
|
||||||
|
return httputils.BAD_REQUEST
|
||||||
return client.CREATED, {}, None
|
return client.CREATED, {}, None
|
||||||
|
|
|
@ -2,7 +2,8 @@
|
||||||
# Copyright © 2008 Nicolas Kandel
|
# Copyright © 2008 Nicolas Kandel
|
||||||
# Copyright © 2008 Pascal Halter
|
# Copyright © 2008 Pascal Halter
|
||||||
# Copyright © 2008-2017 Guillaume Ayoub
|
# Copyright © 2008-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2018 Unrud <unrud@outlook.com>
|
# 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
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,7 +18,9 @@
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import errno
|
||||||
import posixpath
|
import posixpath
|
||||||
|
import re
|
||||||
import socket
|
import socket
|
||||||
from http import client
|
from http import client
|
||||||
|
|
||||||
|
@ -74,8 +77,21 @@ class ApplicationPartMkcol(ApplicationBase):
|
||||||
try:
|
try:
|
||||||
self._storage.create_collection(path, props=props)
|
self._storage.create_collection(path, props=props)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
logger.warning(
|
# return better matching HTTP result in case errno is provided and catched
|
||||||
"Bad MKCOL request on %r (type:%s): %s", path, collection_type, e, exc_info=True)
|
errno_match = re.search("\\[Errno ([0-9]+)\\]", str(e))
|
||||||
return httputils.BAD_REQUEST
|
if errno_match:
|
||||||
|
logger.error(
|
||||||
|
"Failed MKCOL request on %r (type:%s): %s", path, collection_type, e, exc_info=True)
|
||||||
|
errno_e = int(errno_match.group(1))
|
||||||
|
if errno_e == errno.ENOSPC:
|
||||||
|
return httputils.INSUFFICIENT_STORAGE
|
||||||
|
elif errno_e in [errno.EPERM, errno.EACCES]:
|
||||||
|
return httputils.FORBIDDEN
|
||||||
|
else:
|
||||||
|
return httputils.INTERNAL_SERVER_ERROR
|
||||||
|
else:
|
||||||
|
logger.warning(
|
||||||
|
"Bad MKCOL request on %r (type:%s): %s", path, collection_type, e, exc_info=True)
|
||||||
|
return httputils.BAD_REQUEST
|
||||||
logger.info("MKCOL request %r (type:%s): %s", path, collection_type, "successful")
|
logger.info("MKCOL request %r (type:%s): %s", path, collection_type, "successful")
|
||||||
return client.CREATED, {}, None
|
return client.CREATED, {}, None
|
||||||
|
|
|
@ -2,7 +2,8 @@
|
||||||
# Copyright © 2008 Nicolas Kandel
|
# Copyright © 2008 Nicolas Kandel
|
||||||
# Copyright © 2008 Pascal Halter
|
# Copyright © 2008 Pascal Halter
|
||||||
# Copyright © 2008-2017 Guillaume Ayoub
|
# Copyright © 2008-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2018 Unrud <unrud@outlook.com>
|
# Copyright © 2017-2023 Unrud <unrud@outlook.com>
|
||||||
|
# Copyright © 2023-2025 Peter Bieringer <pb@bieringer.de>
|
||||||
#
|
#
|
||||||
# This library is free software: you can redistribute it and/or modify
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,6 +18,7 @@
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import errno
|
||||||
import posixpath
|
import posixpath
|
||||||
import re
|
import re
|
||||||
from http import client
|
from http import client
|
||||||
|
@ -109,7 +111,20 @@ class ApplicationPartMove(ApplicationBase):
|
||||||
try:
|
try:
|
||||||
self._storage.move(item, to_collection, to_href)
|
self._storage.move(item, to_collection, to_href)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
logger.warning(
|
# return better matching HTTP result in case errno is provided and catched
|
||||||
"Bad MOVE request on %r: %s", path, e, exc_info=True)
|
errno_match = re.search("\\[Errno ([0-9]+)\\]", str(e))
|
||||||
return httputils.BAD_REQUEST
|
if errno_match:
|
||||||
|
logger.error(
|
||||||
|
"Failed MOVE request on %r: %s", path, e, exc_info=True)
|
||||||
|
errno_e = int(errno_match.group(1))
|
||||||
|
if errno_e == errno.ENOSPC:
|
||||||
|
return httputils.INSUFFICIENT_STORAGE
|
||||||
|
elif errno_e in [errno.EPERM, errno.EACCES]:
|
||||||
|
return httputils.FORBIDDEN
|
||||||
|
else:
|
||||||
|
return httputils.INTERNAL_SERVER_ERROR
|
||||||
|
else:
|
||||||
|
logger.warning(
|
||||||
|
"Bad MOVE request on %r: %s", path, e, exc_info=True)
|
||||||
|
return httputils.BAD_REQUEST
|
||||||
return client.NO_CONTENT if to_item else client.CREATED, {}, None
|
return client.NO_CONTENT if to_item else client.CREATED, {}, None
|
||||||
|
|
|
@ -2,7 +2,9 @@
|
||||||
# Copyright © 2008 Nicolas Kandel
|
# Copyright © 2008 Nicolas Kandel
|
||||||
# Copyright © 2008 Pascal Halter
|
# Copyright © 2008 Pascal Halter
|
||||||
# Copyright © 2008-2017 Guillaume Ayoub
|
# Copyright © 2008-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2018 Unrud <unrud@outlook.com>
|
# Copyright © 2017-2020 Unrud <unrud@outlook.com>
|
||||||
|
# Copyright © 2020-2020 Tuna Celik <tuna@jakpark.com>
|
||||||
|
# Copyright © 2025-2025 Peter Bieringer <pb@bieringer.de>
|
||||||
#
|
#
|
||||||
# This library is free software: you can redistribute it and/or modify
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,6 +19,8 @@
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import errno
|
||||||
|
import re
|
||||||
import socket
|
import socket
|
||||||
import xml.etree.ElementTree as ET
|
import xml.etree.ElementTree as ET
|
||||||
from http import client
|
from http import client
|
||||||
|
@ -107,7 +111,20 @@ class ApplicationPartProppatch(ApplicationBase):
|
||||||
)
|
)
|
||||||
self._hook.notify(hook_notification_item)
|
self._hook.notify(hook_notification_item)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
logger.warning(
|
# return better matching HTTP result in case errno is provided and catched
|
||||||
"Bad PROPPATCH request on %r: %s", path, e, exc_info=True)
|
errno_match = re.search("\\[Errno ([0-9]+)\\]", str(e))
|
||||||
return httputils.BAD_REQUEST
|
if errno_match:
|
||||||
|
logger.error(
|
||||||
|
"Failed PROPPATCH request on %r: %s", path, e, exc_info=True)
|
||||||
|
errno_e = int(errno_match.group(1))
|
||||||
|
if errno_e == errno.ENOSPC:
|
||||||
|
return httputils.INSUFFICIENT_STORAGE
|
||||||
|
elif errno_e in [errno.EPERM, errno.EACCES]:
|
||||||
|
return httputils.FORBIDDEN
|
||||||
|
else:
|
||||||
|
return httputils.INTERNAL_SERVER_ERROR
|
||||||
|
else:
|
||||||
|
logger.warning(
|
||||||
|
"Bad PROPPATCH request on %r: %s", path, e, exc_info=True)
|
||||||
|
return httputils.BAD_REQUEST
|
||||||
return client.MULTI_STATUS, headers, self._xml_response(xml_answer)
|
return client.MULTI_STATUS, headers, self._xml_response(xml_answer)
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
# Copyright © 2008-2017 Guillaume Ayoub
|
# Copyright © 2008-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2020 Unrud <unrud@outlook.com>
|
# Copyright © 2017-2020 Unrud <unrud@outlook.com>
|
||||||
# Copyright © 2020-2023 Tuna Celik <tuna@jakpark.com>
|
# Copyright © 2020-2023 Tuna Celik <tuna@jakpark.com>
|
||||||
# Copyright © 2024-2024 Peter Bieringer <pb@bieringer.de>
|
# Copyright © 2024-2025 Peter Bieringer <pb@bieringer.de>
|
||||||
#
|
#
|
||||||
# This library is free software: you can redistribute it and/or modify
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -19,8 +19,10 @@
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
# along with Radicale. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import errno
|
||||||
import itertools
|
import itertools
|
||||||
import posixpath
|
import posixpath
|
||||||
|
import re
|
||||||
import socket
|
import socket
|
||||||
import sys
|
import sys
|
||||||
from http import client
|
from http import client
|
||||||
|
@ -264,9 +266,22 @@ class ApplicationPartPut(ApplicationBase):
|
||||||
)
|
)
|
||||||
self._hook.notify(hook_notification_item)
|
self._hook.notify(hook_notification_item)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
logger.warning(
|
# return better matching HTTP result in case errno is provided and catched
|
||||||
"Bad PUT request on %r (upload): %s", path, e, exc_info=True)
|
errno_match = re.search("\\[Errno ([0-9]+)\\]", str(e))
|
||||||
return httputils.BAD_REQUEST
|
if errno_match:
|
||||||
|
logger.error(
|
||||||
|
"Failed PUT request on %r (upload): %s", path, e, exc_info=True)
|
||||||
|
errno_e = int(errno_match.group(1))
|
||||||
|
if errno_e == errno.ENOSPC:
|
||||||
|
return httputils.INSUFFICIENT_STORAGE
|
||||||
|
elif errno_e in [errno.EPERM, errno.EACCES]:
|
||||||
|
return httputils.FORBIDDEN
|
||||||
|
else:
|
||||||
|
return httputils.INTERNAL_SERVER_ERROR
|
||||||
|
else:
|
||||||
|
logger.warning(
|
||||||
|
"Bad PUT request on %r (upload): %s", path, e, exc_info=True)
|
||||||
|
return httputils.BAD_REQUEST
|
||||||
|
|
||||||
headers = {"ETag": etag}
|
headers = {"ETag": etag}
|
||||||
return client.CREATED, headers, None
|
return client.CREATED, headers, None
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
# Copyright © 2008 Pascal Halter
|
# Copyright © 2008 Pascal Halter
|
||||||
# Copyright © 2008-2017 Guillaume Ayoub
|
# Copyright © 2008-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2022 Unrud <unrud@outlook.com>
|
# Copyright © 2017-2022 Unrud <unrud@outlook.com>
|
||||||
# Copyright © 2024-2024 Peter Bieringer <pb@bieringer.de>
|
# Copyright © 2024-2025 Peter Bieringer <pb@bieringer.de>
|
||||||
#
|
#
|
||||||
# This library is free software: you can redistribute it and/or modify
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -79,6 +79,9 @@ REMOTE_DESTINATION: types.WSGIResponse = (
|
||||||
DIRECTORY_LISTING: types.WSGIResponse = (
|
DIRECTORY_LISTING: types.WSGIResponse = (
|
||||||
client.FORBIDDEN, (("Content-Type", "text/plain"),),
|
client.FORBIDDEN, (("Content-Type", "text/plain"),),
|
||||||
"Directory listings are not supported.")
|
"Directory listings are not supported.")
|
||||||
|
INSUFFICIENT_STORAGE: types.WSGIResponse = (
|
||||||
|
client.INSUFFICIENT_STORAGE, (("Content-Type", "text/plain"),),
|
||||||
|
"Insufficient Storage. Please contact the administrator.")
|
||||||
INTERNAL_SERVER_ERROR: types.WSGIResponse = (
|
INTERNAL_SERVER_ERROR: types.WSGIResponse = (
|
||||||
client.INTERNAL_SERVER_ERROR, (("Content-Type", "text/plain"),),
|
client.INTERNAL_SERVER_ERROR, (("Content-Type", "text/plain"),),
|
||||||
"A server error occurred. Please contact the administrator.")
|
"A server error occurred. Please contact the administrator.")
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
# Copyright © 2014 Jean-Marc Martins
|
# Copyright © 2014 Jean-Marc Martins
|
||||||
# Copyright © 2012-2017 Guillaume Ayoub
|
# Copyright © 2012-2017 Guillaume Ayoub
|
||||||
# Copyright © 2017-2021 Unrud <unrud@outlook.com>
|
# Copyright © 2017-2021 Unrud <unrud@outlook.com>
|
||||||
# Copyright © 2024-2024 Peter Bieringer <pb@bieringer.de>
|
# Copyright © 2024-2025 Peter Bieringer <pb@bieringer.de>
|
||||||
#
|
#
|
||||||
# This library is free software: you can redistribute it and/or modify
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -50,27 +50,31 @@ class StoragePartCreateCollection(StorageBase):
|
||||||
self._makedirs_synced(parent_dir)
|
self._makedirs_synced(parent_dir)
|
||||||
|
|
||||||
# Create a temporary directory with an unsafe name
|
# Create a temporary directory with an unsafe name
|
||||||
with TemporaryDirectory(prefix=".Radicale.tmp-", dir=parent_dir
|
try:
|
||||||
) as tmp_dir:
|
with TemporaryDirectory(prefix=".Radicale.tmp-", dir=parent_dir
|
||||||
# The temporary directory itself can't be renamed
|
) as tmp_dir:
|
||||||
tmp_filesystem_path = os.path.join(tmp_dir, "collection")
|
# The temporary directory itself can't be renamed
|
||||||
os.makedirs(tmp_filesystem_path)
|
tmp_filesystem_path = os.path.join(tmp_dir, "collection")
|
||||||
col = self._collection_class(
|
os.makedirs(tmp_filesystem_path)
|
||||||
cast(multifilesystem.Storage, self),
|
col = self._collection_class(
|
||||||
pathutils.unstrip_path(sane_path, True),
|
cast(multifilesystem.Storage, self),
|
||||||
filesystem_path=tmp_filesystem_path)
|
pathutils.unstrip_path(sane_path, True),
|
||||||
col.set_meta(props)
|
filesystem_path=tmp_filesystem_path)
|
||||||
if items is not None:
|
col.set_meta(props)
|
||||||
if props.get("tag") == "VCALENDAR":
|
if items is not None:
|
||||||
col._upload_all_nonatomic(items, suffix=".ics")
|
if props.get("tag") == "VCALENDAR":
|
||||||
elif props.get("tag") == "VADDRESSBOOK":
|
col._upload_all_nonatomic(items, suffix=".ics")
|
||||||
col._upload_all_nonatomic(items, suffix=".vcf")
|
elif props.get("tag") == "VADDRESSBOOK":
|
||||||
|
col._upload_all_nonatomic(items, suffix=".vcf")
|
||||||
|
|
||||||
if os.path.lexists(filesystem_path):
|
if os.path.lexists(filesystem_path):
|
||||||
pathutils.rename_exchange(tmp_filesystem_path, filesystem_path)
|
pathutils.rename_exchange(tmp_filesystem_path, filesystem_path)
|
||||||
else:
|
else:
|
||||||
os.rename(tmp_filesystem_path, filesystem_path)
|
os.rename(tmp_filesystem_path, filesystem_path)
|
||||||
self._sync_directory(parent_dir)
|
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
|
||||||
|
|
||||||
return self._collection_class(
|
return self._collection_class(
|
||||||
cast(multifilesystem.Storage, self),
|
cast(multifilesystem.Storage, self),
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue