2016-12-16 15:07:14 +00:00
|
|
|
import os
|
|
|
|
import types
|
|
|
|
|
2018-10-03 13:27:59 +01:00
|
|
|
from sanic.exceptions import PyFileError
|
|
|
|
|
2016-12-16 15:07:14 +00:00
|
|
|
|
2017-09-13 07:42:42 +01:00
|
|
|
SANIC_PREFIX = 'SANIC_'
|
2017-03-29 04:57:58 +01:00
|
|
|
|
2017-04-12 10:44:47 +01:00
|
|
|
|
2016-12-16 15:07:14 +00:00
|
|
|
class Config(dict):
|
2017-04-17 06:43:49 +01:00
|
|
|
def __init__(self, defaults=None, load_env=True, keep_alive=True):
|
2016-12-16 17:46:07 +00:00
|
|
|
super().__init__(defaults or {})
|
2016-12-16 15:07:14 +00:00
|
|
|
self.LOGO = """
|
2016-10-15 20:59:00 +01:00
|
|
|
▄▄▄▄▄
|
|
|
|
▀▀▀██████▄▄▄ _______________
|
|
|
|
▄▄▄▄▄ █████████▄ / \\
|
|
|
|
▀▀▀▀█████▌ ▀▐▄ ▀▐█ | Gotta go fast! |
|
|
|
|
▀▀█████▄▄ ▀██████▄██ | _________________/
|
|
|
|
▀▄▄▄▄▄ ▀▀█▄▀█════█▀ |/
|
|
|
|
▀▀▀▄ ▀▀███ ▀ ▄▄
|
|
|
|
▄███▀▀██▄████████▄ ▄▀▀▀▀▀▀█▌
|
|
|
|
██▀▄▄▄██▀▄███▀ ▀▀████ ▄██
|
|
|
|
▄▀▀▀▄██▄▀▀▌████▒▒▒▒▒▒███ ▌▄▄▀
|
|
|
|
▌ ▐▀████▐███▒▒▒▒▒▐██▌
|
|
|
|
▀▄▄▄▄▀ ▀▀████▒▒▒▒▄██▀
|
|
|
|
▀▀█████████▀
|
|
|
|
▄▄██▀██████▀█
|
|
|
|
▄██▀ ▀▀▀ █
|
|
|
|
▄█ ▐▌
|
|
|
|
▄▄▄▄█▌ ▀█▄▄▄▄▀▀▄
|
|
|
|
▌ ▐ ▀▀▄▄▄▀
|
|
|
|
▀▀▄▄▀
|
|
|
|
"""
|
2017-05-26 04:11:26 +01:00
|
|
|
self.REQUEST_MAX_SIZE = 100000000 # 100 megabytes
|
2016-12-16 15:07:14 +00:00
|
|
|
self.REQUEST_TIMEOUT = 60 # 60 seconds
|
2017-09-11 08:17:33 +01:00
|
|
|
self.RESPONSE_TIMEOUT = 60 # 60 seconds
|
2017-04-17 06:43:49 +01:00
|
|
|
self.KEEP_ALIVE = keep_alive
|
2017-09-11 08:17:33 +01:00
|
|
|
self.KEEP_ALIVE_TIMEOUT = 5 # 5 seconds
|
2017-05-26 04:11:26 +01:00
|
|
|
self.WEBSOCKET_MAX_SIZE = 2 ** 20 # 1 megabytes
|
|
|
|
self.WEBSOCKET_MAX_QUEUE = 32
|
2018-02-10 04:44:02 +00:00
|
|
|
self.WEBSOCKET_READ_LIMIT = 2 ** 16
|
|
|
|
self.WEBSOCKET_WRITE_LIMIT = 2 ** 16
|
2017-06-25 18:22:40 +01:00
|
|
|
self.GRACEFUL_SHUTDOWN_TIMEOUT = 15.0 # 15 sec
|
2018-08-15 18:23:04 +01:00
|
|
|
self.ACCESS_LOG = True
|
2016-12-16 15:07:14 +00:00
|
|
|
|
2017-03-22 01:37:46 +00:00
|
|
|
if load_env:
|
2017-09-05 10:05:31 +01:00
|
|
|
prefix = SANIC_PREFIX if load_env is True else load_env
|
2017-09-05 09:58:48 +01:00
|
|
|
self.load_environment_vars(prefix=prefix)
|
2017-03-22 01:37:46 +00:00
|
|
|
|
2016-12-16 15:07:14 +00:00
|
|
|
def __getattr__(self, attr):
|
2016-12-16 17:46:07 +00:00
|
|
|
try:
|
|
|
|
return self[attr]
|
|
|
|
except KeyError as ke:
|
|
|
|
raise AttributeError("Config has no '{}'".format(ke.args[0]))
|
2016-12-16 15:07:14 +00:00
|
|
|
|
|
|
|
def __setattr__(self, attr, value):
|
|
|
|
self[attr] = value
|
|
|
|
|
|
|
|
def from_envvar(self, variable_name):
|
2017-02-14 19:10:19 +00:00
|
|
|
"""Load a configuration from an environment variable pointing to
|
2016-12-16 15:07:14 +00:00
|
|
|
a configuration file.
|
2017-02-14 19:10:19 +00:00
|
|
|
|
2016-12-16 15:07:14 +00:00
|
|
|
:param variable_name: name of the environment variable
|
|
|
|
:return: bool. ``True`` if able to load config, ``False`` otherwise.
|
|
|
|
"""
|
|
|
|
config_file = os.environ.get(variable_name)
|
|
|
|
if not config_file:
|
|
|
|
raise RuntimeError('The environment variable %r is not set and '
|
|
|
|
'thus configuration could not be loaded.' %
|
|
|
|
variable_name)
|
|
|
|
return self.from_pyfile(config_file)
|
|
|
|
|
|
|
|
def from_pyfile(self, filename):
|
2017-02-14 19:10:19 +00:00
|
|
|
"""Update the values in the config from a Python file.
|
|
|
|
Only the uppercase variables in that module are stored in the config.
|
|
|
|
|
2016-12-16 15:07:14 +00:00
|
|
|
:param filename: an absolute path to the config file
|
|
|
|
"""
|
|
|
|
module = types.ModuleType('config')
|
|
|
|
module.__file__ = filename
|
|
|
|
try:
|
|
|
|
with open(filename) as config_file:
|
|
|
|
exec(compile(config_file.read(), filename, 'exec'),
|
|
|
|
module.__dict__)
|
|
|
|
except IOError as e:
|
|
|
|
e.strerror = 'Unable to load configuration file (%s)' % e.strerror
|
|
|
|
raise
|
2018-10-03 13:27:59 +01:00
|
|
|
except Exception as e:
|
|
|
|
raise PyFileError(filename) from e
|
|
|
|
|
2016-12-16 15:07:14 +00:00
|
|
|
self.from_object(module)
|
|
|
|
return True
|
|
|
|
|
|
|
|
def from_object(self, obj):
|
2017-02-14 19:10:19 +00:00
|
|
|
"""Update the values from the given object.
|
2016-12-16 15:07:14 +00:00
|
|
|
Objects are usually either modules or classes.
|
|
|
|
|
|
|
|
Just the uppercase variables in that object are stored in the config.
|
|
|
|
Example usage::
|
|
|
|
|
|
|
|
from yourapplication import default_config
|
|
|
|
app.config.from_object(default_config)
|
|
|
|
|
|
|
|
You should not use this function to load the actual configuration but
|
|
|
|
rather configuration defaults. The actual config should be loaded
|
|
|
|
with :meth:`from_pyfile` and ideally from a location not within the
|
|
|
|
package because the package might be installed system wide.
|
|
|
|
|
2016-12-17 19:20:07 +00:00
|
|
|
:param obj: an object holding the configuration
|
2016-12-16 15:07:14 +00:00
|
|
|
"""
|
|
|
|
for key in dir(obj):
|
|
|
|
if key.isupper():
|
|
|
|
self[key] = getattr(obj, key)
|
2017-03-22 01:37:46 +00:00
|
|
|
|
2017-09-05 08:53:33 +01:00
|
|
|
def load_environment_vars(self, prefix=SANIC_PREFIX):
|
2017-04-17 05:39:18 +01:00
|
|
|
"""
|
2017-09-05 08:53:33 +01:00
|
|
|
Looks for prefixed environment variables and applies
|
2017-04-17 05:39:18 +01:00
|
|
|
them to the configuration if present.
|
|
|
|
"""
|
2017-03-22 01:37:46 +00:00
|
|
|
for k, v in os.environ.items():
|
2017-09-05 08:53:33 +01:00
|
|
|
if k.startswith(prefix):
|
|
|
|
_, config_key = k.split(prefix, 1)
|
2017-06-27 05:26:34 +01:00
|
|
|
try:
|
|
|
|
self[config_key] = int(v)
|
|
|
|
except ValueError:
|
|
|
|
try:
|
2017-06-27 04:58:31 +01:00
|
|
|
self[config_key] = float(v)
|
2017-06-27 05:26:34 +01:00
|
|
|
except ValueError:
|
|
|
|
self[config_key] = v
|