-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathconfig.py
141 lines (109 loc) · 3.89 KB
/
config.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
# Batteries
import json
import os
from functools import reduce
# Third-party Imports
from loguru import logger
class InvalidConfiguration(Exception):
"""
Thrown when the read configuration is missing
mandatory parameters or is an invalid JSON format.
Args:
builtins.Exception (class): Builtin exception class.
"""
...
class Config(object):
"""
Base configuration class. Reads data from config.json file
and allows retrieving that data.
Configurations are cached in memory for faster access.
Args:
builtins.object (class): Builtin object class.
"""
# Base directory
BASE_DIR = os.path.realpath(os.path.dirname(os.path.realpath(__file__)))
# Default configuration file
CONFIG_FILE_PATH = f'{BASE_DIR}/config.json'
# Supported record types
SUPPORTED_RECORD_TYPES = ('A', 'AAAA', 'CNAME', 'MX')
# Class parameters
_configfile = None
_configdict = None
# TODO: Validate configurations with JSON Schema package
@classmethod
def load(cls, path):
"""
Loads the configuration file into memory.
Args:
path (str): The path for the configuration file.
"""
try:
# Set configuration file
cls._configfile = path
# Open file and read configuration into memory
with open(cls._configfile, 'r') as configfile:
cls._configdict = json.load(configfile)
except FileNotFoundError:
raise InvalidConfiguration(f'Configuration file {path} does not exist.')
except json.JSONDecodeError:
raise InvalidConfiguration('Configuration file contains invalid JSON format.')
@classmethod
def reload(cls):
"""
Reloads the configuration file into memory.
"""
try:
cls.load(cls.CONFIG_FILE_PATH)
except InvalidConfiguration as e:
logger.error(f'Invalid configurations, not reloading. Error: {str(e)}')
return False
return True
@classmethod
def get(cls, key: str, default: object = None) -> object:
"""
Retrieves the value for a key from the configuration file.
Args:
key (str): The key from which to get the value. These
can be several splitted by a dot.
default (object): What to return if the key is not found.
Returns:
object: The configuration value.
"""
# Retrieve configurations if not loaded
if not cls._configdict:
cls.load(cls.CONFIG_FILE_PATH)
value = reduce(dict.get, key.split('.'), cls._configdict)
return default if value is None else value
@classmethod
def int(cls, key: str, default: int = None):
"""
Retrieves the value for a key from the configuration file.
Args:
key (str): The key from which to get the value. These
can be several splitted by a dot.
default (int): What to return if the key is not found.
Returns:
int: The configuration value.
"""
return int(cls.get(str, default))
@classmethod
def getpath(cls, key):
"""
Utility function to retrieve a key whose value is a
filesystem path. This will read both absolute and
relative paths and always return an absolute path.
If relative, the path will be prefixed with the
project's root.
Args:
key (str): The key from which to get the value. These
can be several splitted by a dot.
Returns:
str: The absolute path.
"""
# Read path from configuration
path = str(cls.get(key))
# Prevent empty value
if not path:
return None
# Return path or absolute path from base dir
return path if os.path.isabs(path) else f'{cls.BASE_DIR}/{path}'