7
0
mirror of https://github.com/ChristianLight/tutor.git synced 2024-05-29 12:20:49 +00:00
tutor/tutor/config.py
Régis Behmo 0a670d7ead refactor: add type annotations
Annotations were generated with pyannotate:
https://github.com/dropbox/pyannotate

We are running in strict mode, which is awesome!

This affects a large part of the code base, which might be an issue for
people running a fork of Tutor. Nonetheless, the behavior should not be
affected. If anything, this process has helped find and resolve a few
type-related bugs. Thus, this is not considered as a breaking change.
2021-03-15 21:46:55 +01:00

228 lines
7.0 KiB
Python

import os
from typing import Dict, Any, Tuple
from . import exceptions
from . import env
from . import fmt
from . import plugins
from . import serialize
from . import utils
def update(root: str) -> Dict[str, Any]:
"""
Load and save the configuration.
"""
config, defaults = load_all(root)
save_config_file(root, config)
merge(config, defaults)
return config
def load(root: str) -> Dict[str, Any]:
"""
Load full configuration. This will raise an exception if there is no current
configuration in the project root.
"""
check_existing_config(root)
return load_no_check(root)
def load_no_check(root: str) -> Dict[str, Any]:
config, defaults = load_all(root)
merge(config, defaults)
return config
def load_all(root: str) -> Tuple[Dict[str, Any], Dict[str, Any]]:
"""
Return:
current (dict): params currently saved in config.yml
defaults (dict): default values of params which might be missing from the
current config
"""
defaults = load_defaults()
current = load_current(root, defaults)
return current, defaults
def merge(
config: Dict[str, str], defaults: Dict[str, str], force: bool = False
) -> None:
"""
Merge default values with user configuration and perform rendering of "{{...}}"
values.
"""
for key, value in defaults.items():
if force or key not in config:
config[key] = env.render_unknown(config, value)
def load_defaults() -> Dict[str, Any]:
return serialize.load(env.read_template_file("config.yml"))
def load_config_file(path: str) -> Dict[str, Any]:
with open(path) as f:
return serialize.load(f.read())
def load_current(root: str, defaults: Dict[str, str]) -> Dict[str, Any]:
"""
Load the configuration currently stored on disk.
Note: this modifies the defaults with the plugin default values.
"""
convert_json2yml(root)
config = load_user(root)
load_env(config, defaults)
load_required(config, defaults)
load_plugins(config, defaults)
return config
def load_user(root: str) -> Dict[str, Any]:
path = config_path(root)
if not os.path.exists(path):
return {}
config = load_config_file(path)
upgrade_obsolete(config)
return config
def load_env(config: Dict[str, str], defaults: Dict[str, str]) -> None:
for k in defaults.keys():
env_var = "TUTOR_" + k
if env_var in os.environ:
config[k] = serialize.parse(os.environ[env_var])
def load_required(config: Dict[str, str], defaults: Dict[str, str]) -> None:
"""
All these keys must be present in the user's config.yml. This includes all values
that are generated once and must be kept after that, such as passwords.
"""
for key in [
"OPENEDX_SECRET_KEY",
"MYSQL_ROOT_PASSWORD",
"OPENEDX_MYSQL_PASSWORD",
"ANDROID_OAUTH2_SECRET",
"ID",
"JWT_RSA_PRIVATE_KEY",
]:
if key not in config:
config[key] = env.render_unknown(config, defaults[key])
def load_plugins(config: Dict[str, str], defaults: Dict[str, str]) -> None:
"""
Add, override and set new defaults from plugins.
"""
for plugin in plugins.iter_enabled(config):
# Add new config key/values
for key, value in plugin.config_add.items():
new_key = plugin.config_key(key)
if new_key not in config:
config[new_key] = env.render_unknown(config, value)
# Create new defaults
for key, value in plugin.config_defaults.items():
defaults[plugin.config_key(key)] = value
# Set existing config key/values: here, we do not override existing values
# This must come last, as overridden values may depend on plugin defaults
for key, value in plugin.config_set.items():
if key not in config:
config[key] = env.render_unknown(config, value)
def is_service_activated(config: Dict[str, Any], service: str) -> bool:
return config["RUN_" + service.upper()] is not False
def upgrade_obsolete(config: Dict[str, Any]) -> None:
# Openedx-specific mysql passwords
if "MYSQL_PASSWORD" in config:
config["MYSQL_ROOT_PASSWORD"] = config["MYSQL_PASSWORD"]
config["OPENEDX_MYSQL_PASSWORD"] = config["MYSQL_PASSWORD"]
config.pop("MYSQL_PASSWORD")
if "MYSQL_DATABASE" in config:
config["OPENEDX_MYSQL_DATABASE"] = config.pop("MYSQL_DATABASE")
if "MYSQL_USERNAME" in config:
config["OPENEDX_MYSQL_USERNAME"] = config.pop("MYSQL_USERNAME")
if "RUN_NOTES" in config:
if config["RUN_NOTES"]:
plugins.enable(config, "notes")
config.pop("RUN_NOTES")
if "RUN_XQUEUE" in config:
if config["RUN_XQUEUE"]:
plugins.enable(config, "xqueue")
config.pop("RUN_XQUEUE")
if "SECRET_KEY" in config:
config["OPENEDX_SECRET_KEY"] = config.pop("SECRET_KEY")
# Replace WEB_PROXY by RUN_CADDY
if "WEB_PROXY" in config:
config["RUN_CADDY"] = not config.pop("WEB_PROXY")
# Rename ACTIVATE_HTTPS to ENABLE_HTTPS
if "ACTIVATE_HTTPS" in config:
config["ENABLE_HTTPS"] = config.pop("ACTIVATE_HTTPS")
# Replace RUN_* variables by RUN_*
for name in [
"ACTIVATE_LMS",
"ACTIVATE_CMS",
"ACTIVATE_FORUM",
"ACTIVATE_ELASTICSEARCH",
"ACTIVATE_MONGODB",
"ACTIVATE_MYSQL",
"ACTIVATE_REDIS",
"ACTIVATE_SMTP",
]:
if name in config:
config[name.replace("ACTIVATE_", "RUN_")] = config.pop(name)
def convert_json2yml(root: str) -> None:
"""
Older versions of tutor used to have json config files.
"""
json_path = os.path.join(root, "config.json")
if not os.path.exists(json_path):
return
if os.path.exists(config_path(root)):
raise exceptions.TutorError(
"Both config.json and config.yml exist in {}: only one of these files must exist to continue".format(
root
)
)
config = load_config_file(json_path)
save_config_file(root, config)
os.remove(json_path)
fmt.echo_info(
"File config.json detected in {} and converted to config.yml".format(root)
)
def save_config_file(root: str, config: Dict[str, str]) -> None:
path = config_path(root)
utils.ensure_file_directory_exists(path)
with open(path, "w") as of:
serialize.dump(config, of)
fmt.echo_info("Configuration saved to {}".format(path))
def check_existing_config(root: str) -> None:
"""
Check there is a configuration on disk and the current environment is up-to-date.
"""
if not os.path.exists(config_path(root)):
raise exceptions.TutorError(
"Project root does not exist. Make sure to generate the initial "
"configuration with `tutor config save --interactive` or `tutor local "
"quickstart` prior to running other commands."
)
env.check_is_up_to_date(root)
def config_path(root: str) -> str:
return os.path.join(root, "config.yml")