Files
tradon/config.py
2025-12-26 13:11:43 +00:00

168 lines
6.2 KiB
Python
Executable File

# This file is part of Tryton. The COPYRIGHT file at the top level of
# this repository contains the full copyright notices and license terms.
import configparser
import logging
import os
import urllib.parse
from getpass import getuser
__all__ = ['config', 'get_hostname', 'get_port', 'split_netloc',
'parse_listen', 'parse_uri']
logger = logging.getLogger(__name__)
# Needed so urlunsplit to always set netloc
for backend_name in ['postgresql', 'sqlite']:
if backend_name not in urllib.parse.uses_netloc:
urllib.parse.uses_netloc.append(backend_name)
def get_hostname(netloc):
if '[' in netloc and ']' in netloc:
return netloc.split(']')[0][1:]
elif ':' in netloc:
return netloc.split(':')[0]
else:
return netloc
def get_port(netloc):
netloc = netloc.split(']')[-1]
return int(netloc.split(':')[1])
def split_netloc(netloc):
return get_hostname(netloc).replace('*', ''), get_port(netloc)
def parse_listen(value):
for netloc in value.split(','):
yield split_netloc(netloc)
def parse_uri(uri):
return urllib.parse.urlparse(uri)
class TrytonConfigParser(configparser.ConfigParser):
def __init__(self):
super().__init__(interpolation=None)
self.add_section('web')
self.set('web', 'listen', 'localhost:8000')
self.set('web', 'root', os.path.join(os.path.expanduser('~'), 'www'))
self.set('web', 'num_proxies', '0')
self.set('web', 'cache_timeout', str(60 * 60 * 12))
self.add_section('database')
self.set('database', 'uri',
os.environ.get('TRYTOND_DATABASE_URI', 'sqlite://'))
self.set('database', 'path', os.path.join(
os.path.expanduser('~'), 'db'))
self.set('database', 'list', 'True')
self.set('database', 'retry', '5')
self.set('database', 'language', 'en')
self.set('database', 'timeout', str(30 * 60))
self.set('database', 'subquery_threshold', str(1_000))
self.add_section('request')
self.set('request', 'max_size', str(2 * 1024 * 1024))
self.set('request', 'max_size_authenticated',
str(2 * 1024 * 1024 * 1024))
self.set('request', 'timeout', str(60))
self.add_section('cache')
self.set('cache', 'transaction', '10')
self.set('cache', 'model', '200')
self.set('cache', 'record', '2000')
self.set('cache', 'field', '100')
self.set('cache', 'default', '1024')
self.set('cache', 'ir.message', '10240')
self.set('cache', 'ir.translation', '10240')
self.add_section('queue')
self.set('queue', 'worker', 'False')
self.add_section('ssl')
self.add_section('email')
self.set('email', 'uri', 'smtp://localhost:25')
self.set('email', 'from', getuser())
self.add_section('session')
self.set('session', 'authentications', 'password')
self.set('session', 'max_age', str(60 * 60 * 24 * 30))
self.set('session', 'timeout', str(60 * 5))
self.set('session', 'max_attempt', '5')
self.set('session', 'max_attempt_ip_network', '300')
self.set('session', 'ip_network_4', '32')
self.set('session', 'ip_network_6', '56')
self.add_section('password')
self.set('password', 'length', '8')
self.set('password', 'reset_timeout', str(24 * 60 * 60))
self.add_section('bus')
self.set('bus', 'allow_subscribe', 'False')
self.set('bus', 'long_polling_timeout', str(5 * 60))
self.set('bus', 'cache_timeout', '5')
self.set('bus', 'select_timeout', '5')
self.add_section('html')
self.update_environ()
self.update_etc()
def update_environ(self):
for key, value in os.environ.items():
if not key.startswith('TRYTOND_'):
continue
try:
section, option = key[len('TRYTOND_'):].lower().split('__', 1)
except ValueError:
continue
if section.startswith('wsgi_'):
section = section.replace('wsgi_', 'wsgi ')
if not self.has_section(section):
self.add_section(section)
self.set(section, option, value)
def update_etc(self, configfile=os.environ.get('TRYTOND_CONFIG')):
if isinstance(configfile, str):
configfile = [configfile]
if not configfile or not [_f for _f in configfile if _f]:
return []
configfile = [os.path.expanduser(filename) for filename in configfile]
read_files = self.read(configfile)
logger.info('using %s as configuration files', ', '.join(read_files))
if configfile != read_files:
logger.error('could not load %s',
','.join(set(configfile) - set(read_files)))
return configfile
def get(self, section, option, *args, **kwargs):
default = kwargs.pop('default', None)
try:
return configparser.RawConfigParser.get(self, section, option,
*args, **kwargs)
except (configparser.NoOptionError, configparser.NoSectionError):
return default
def getint(self, section, option, *args, **kwargs):
default = kwargs.pop('default', None)
try:
return configparser.RawConfigParser.getint(self, section, option,
*args, **kwargs)
except (configparser.NoOptionError, configparser.NoSectionError,
TypeError):
return default
def getfloat(self, section, option, *args, **kwargs):
default = kwargs.pop('default', None)
try:
return configparser.RawConfigParser.getfloat(self, section, option,
*args, **kwargs)
except (configparser.NoOptionError, configparser.NoSectionError,
TypeError):
return default
def getboolean(self, section, option, *args, **kwargs):
default = kwargs.pop('default', None)
try:
return configparser.RawConfigParser.getboolean(
self, section, option, *args, **kwargs)
except (configparser.NoOptionError, configparser.NoSectionError,
AttributeError):
return default
config = TrytonConfigParser()