Add --verify-storage argument

This commit is contained in:
Unrud 2017-08-25 19:13:09 +02:00
parent 36bca799e1
commit 2df009fac8
2 changed files with 72 additions and 5 deletions

View File

@ -33,7 +33,7 @@ import sys
from wsgiref.simple_server import make_server from wsgiref.simple_server import make_server
from . import (VERSION, Application, RequestHandler, ThreadedHTTPServer, from . import (VERSION, Application, RequestHandler, ThreadedHTTPServer,
ThreadedHTTPSServer, config, log) ThreadedHTTPSServer, config, log, storage)
def run(): def run():
@ -42,6 +42,8 @@ def run():
parser = argparse.ArgumentParser(usage="radicale [OPTIONS]") parser = argparse.ArgumentParser(usage="radicale [OPTIONS]")
parser.add_argument("--version", action="version", version=VERSION) parser.add_argument("--version", action="version", version=VERSION)
parser.add_argument("--verify-storage", action="store_true",
help="check the storage for errors and exit")
parser.add_argument( parser.add_argument(
"-C", "--config", help="use a specific configuration file") "-C", "--config", help="use a specific configuration file")
@ -103,6 +105,10 @@ def run():
if value is not None: if value is not None:
configuration.set(section, action.split('_', 1)[1], value) configuration.set(section, action.split('_', 1)[1], value)
if args.verify_storage:
# Write to stderr when storage verification is requested
configuration["logging"]["config"] = ""
# Start logging # Start logging
filename = os.path.expanduser(configuration.get("logging", "config")) filename = os.path.expanduser(configuration.get("logging", "config"))
debug = configuration.getboolean("logging", "debug") debug = configuration.getboolean("logging", "debug")
@ -114,6 +120,20 @@ def run():
raise raise
exit(1) exit(1)
if args.verify_storage:
logger.info("Verifying storage")
try:
Collection = storage.load(configuration, logger)
with Collection.acquire_lock("r"):
if not Collection.verify():
logger.error("Storage verifcation failed")
exit(1)
except Exception as e:
logger.error("An exception occurred during storage verification: "
"%s", e, exc_info=True)
exit(1)
return
try: try:
serve(configuration, logger) serve(configuration, logger)
except Exception as e: except Exception as e:

View File

@ -699,6 +699,11 @@ class BaseCollection:
""" """
raise NotImplementedError raise NotImplementedError
@classmethod
def verify(cls):
"""Check the storage for errors."""
return True
class Collection(BaseCollection): class Collection(BaseCollection):
"""Collection stored in several files per calendar.""" """Collection stored in several files per calendar."""
@ -807,7 +812,8 @@ class Collection(BaseCollection):
cls._sync_directory(parent_filesystem_path) cls._sync_directory(parent_filesystem_path)
@classmethod @classmethod
def discover(cls, path, depth="0"): def discover(cls, path, depth="0", child_context_manager=(
lambda path, href=None: contextlib.ExitStack())):
# Path should already be sanitized # Path should already be sanitized
sane_path = sanitize_path(path).strip("/") sane_path = sanitize_path(path).strip("/")
attributes = sane_path.split("/") if sane_path else [] attributes = sane_path.split("/") if sane_path else []
@ -844,8 +850,9 @@ class Collection(BaseCollection):
if depth == "0": if depth == "0":
return return
for item in collection.list(): for href in collection.list():
yield collection.get(item) with child_context_manager(sane_path, href):
yield collection.get(href)
for href in scandir(filesystem_path, only_dirs=True): for href in scandir(filesystem_path, only_dirs=True):
if not is_safe_filesystem_path_component(href): if not is_safe_filesystem_path_component(href):
@ -854,7 +861,47 @@ class Collection(BaseCollection):
sane_path) sane_path)
continue continue
child_path = posixpath.join(sane_path, href) child_path = posixpath.join(sane_path, href)
yield cls(child_path) with child_context_manager(child_path):
yield cls(child_path)
@classmethod
def verify(cls):
item_errors = collection_errors = 0
@contextlib.contextmanager
def exception_cm(path, href=None):
nonlocal item_errors, collection_errors
try:
yield
except Exception as e:
if href:
item_errors += 1
name = "item %r in %r" % (href, path.strip("/"))
else:
collection_errors += 1
name = "collection %r" % path.strip("/")
cls.logger.error("Invalid %s: %s", name, e, exc_info=True)
remaining_paths = [""]
while remaining_paths:
path = remaining_paths.pop(0)
cls.logger.debug("Verifying collection %r", path)
with exception_cm(path):
saved_item_errors = item_errors
collection = None
for item in cls.discover(path, "1", exception_cm):
if not collection:
collection = item
collection.get_meta()
continue
if isinstance(item, BaseCollection):
remaining_paths.append(item.path)
else:
cls.logger.debug("Verified item %r in %r",
item.href, path)
if item_errors == saved_item_errors:
collection.sync()
return item_errors == 0 and collection_errors == 0
@classmethod @classmethod
def create_collection(cls, href, collection=None, props=None): def create_collection(cls, href, collection=None, props=None):