__main__.py 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. # This file is part of Radicale Server - Calendar Server
  2. # Copyright © 2011-2017 Guillaume Ayoub
  3. # Copyright © 2017-2019 Unrud <unrud@outlook.com>
  4. #
  5. # This library is free software: you can redistribute it and/or modify
  6. # it under the terms of the GNU General Public License as published by
  7. # the Free Software Foundation, either version 3 of the License, or
  8. # (at your option) any later version.
  9. #
  10. # This library is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with Radicale. If not, see <http://www.gnu.org/licenses/>.
  17. """
  18. Radicale executable module.
  19. This module can be executed from a command line with ``$python -m radicale``.
  20. Uses the built-in WSGI server.
  21. """
  22. import argparse
  23. import contextlib
  24. import os
  25. import signal
  26. import socket
  27. import sys
  28. from types import FrameType
  29. from typing import List, cast
  30. from radicale import VERSION, config, log, server, storage, types
  31. from radicale.log import logger
  32. def run() -> None:
  33. """Run Radicale as a standalone server."""
  34. exit_signal_numbers = [signal.SIGTERM, signal.SIGINT]
  35. if os.name == "posix":
  36. exit_signal_numbers.append(signal.SIGHUP)
  37. exit_signal_numbers.append(signal.SIGQUIT)
  38. if sys.platform == "win32":
  39. exit_signal_numbers.append(signal.SIGBREAK)
  40. # Raise SystemExit when signal arrives to run cleanup code
  41. # (like destructors, try-finish etc.), otherwise the process exits
  42. # without running any of them
  43. def exit_signal_handler(signal_number: "signal.Signals",
  44. stack_frame: FrameType) -> None:
  45. sys.exit(1)
  46. for signal_number in exit_signal_numbers:
  47. signal.signal(signal_number, exit_signal_handler)
  48. log.setup()
  49. # Get command-line arguments
  50. # Configuration options are stored in dest with format "c:SECTION:OPTION"
  51. parser = argparse.ArgumentParser(
  52. prog="radicale", usage="%(prog)s [OPTIONS]", allow_abbrev=False)
  53. parser.add_argument("--version", action="version", version=VERSION)
  54. parser.add_argument("--verify-storage", action="store_true",
  55. help="check the storage for errors and exit")
  56. parser.add_argument("-C", "--config",
  57. help="use specific configuration files", nargs="*")
  58. parser.add_argument("-D", "--debug", action="store_const", const="debug",
  59. dest="c:logging:level", default=argparse.SUPPRESS,
  60. help="print debug information")
  61. for section, section_data in config.DEFAULT_CONFIG_SCHEMA.items():
  62. if section.startswith("_"):
  63. continue
  64. assert ":" not in section # check field separator
  65. assert "-" not in section and "_" not in section # not implemented
  66. group = parser.add_argument_group(section)
  67. for option, data in section_data.items():
  68. if option.startswith("_"):
  69. continue
  70. kwargs = data.copy()
  71. long_name = "--%s-%s" % (section, option.replace("_", "-"))
  72. args: List[str] = list(kwargs.pop("aliases", ()))
  73. args.append(long_name)
  74. kwargs["dest"] = "c:%s:%s" % (section, option)
  75. kwargs["metavar"] = "VALUE"
  76. kwargs["default"] = argparse.SUPPRESS
  77. del kwargs["value"]
  78. with contextlib.suppress(KeyError):
  79. del kwargs["internal"]
  80. if kwargs["type"] == bool:
  81. del kwargs["type"]
  82. opposite_args = list(kwargs.pop("opposite_aliases", ()))
  83. opposite_args.append("--no%s" % long_name[1:])
  84. kwargs["action"] = "store_const"
  85. kwargs["const"] = "True"
  86. group.add_argument(*args, **kwargs)
  87. # Opposite argument
  88. kwargs["const"] = "False"
  89. kwargs["help"] = "do not %s (opposite of %s)" % (
  90. kwargs["help"], long_name)
  91. group.add_argument(*opposite_args, **kwargs)
  92. else:
  93. del kwargs["type"]
  94. group.add_argument(*args, **kwargs)
  95. args_ns = parser.parse_args()
  96. # Preliminary configure logging
  97. with contextlib.suppress(ValueError):
  98. log.set_level(config.DEFAULT_CONFIG_SCHEMA["logging"]["level"]["type"](
  99. vars(args_ns).get("c:logging:level", "")))
  100. # Update Radicale configuration according to arguments
  101. arguments_config: types.MUTABLE_CONFIG = {}
  102. for key, value in vars(args_ns).items():
  103. if key.startswith("c:"):
  104. _, section, option = key.split(":", maxsplit=2)
  105. arguments_config[section] = arguments_config.get(section, {})
  106. arguments_config[section][option] = value
  107. try:
  108. configuration = config.load(config.parse_compound_paths(
  109. config.DEFAULT_CONFIG_PATH,
  110. os.environ.get("RADICALE_CONFIG"),
  111. os.pathsep.join(args_ns.config) if args_ns.config else None))
  112. if arguments_config:
  113. configuration.update(arguments_config, "command line arguments")
  114. except Exception as e:
  115. logger.critical("Invalid configuration: %s", e, exc_info=True)
  116. sys.exit(1)
  117. # Configure logging
  118. log.set_level(cast(str, configuration.get("logging", "level")))
  119. # Log configuration after logger is configured
  120. for source, miss in configuration.sources():
  121. logger.info("%s %s", "Skipped missing" if miss else "Loaded", source)
  122. if args_ns.verify_storage:
  123. logger.info("Verifying storage")
  124. try:
  125. storage_ = storage.load(configuration)
  126. with storage_.acquire_lock("r"):
  127. if not storage_.verify():
  128. logger.critical("Storage verifcation failed")
  129. sys.exit(1)
  130. except Exception as e:
  131. logger.critical("An exception occurred during storage "
  132. "verification: %s", e, exc_info=True)
  133. sys.exit(1)
  134. return
  135. # Create a socket pair to notify the server of program shutdown
  136. shutdown_socket, shutdown_socket_out = socket.socketpair()
  137. # Shutdown server when signal arrives
  138. def shutdown_signal_handler(signal_number: "signal.Signals",
  139. stack_frame: FrameType) -> None:
  140. shutdown_socket.close()
  141. for signal_number in exit_signal_numbers:
  142. signal.signal(signal_number, shutdown_signal_handler)
  143. try:
  144. server.serve(configuration, shutdown_socket_out)
  145. except Exception as e:
  146. logger.critical("An exception occurred during server startup: %s", e,
  147. exc_info=True)
  148. sys.exit(1)
  149. if __name__ == "__main__":
  150. run()