mirror of
https://github.com/inventree/InvenTree
synced 2024-08-30 18:33:04 +00:00
- Provides an error log viewer in the admin interface at /admin/error_report/error/ - Allows viewing of error logs even in a remote production environment (i.e. no access to command line)
407 lines
12 KiB
Python
407 lines
12 KiB
Python
"""
|
|
Django settings for InvenTree project.
|
|
|
|
In practice the settings in this file should not be adjusted,
|
|
instead settings can be configured in the config.yaml file
|
|
located in the top level project directory.
|
|
|
|
This allows implementation configuration to be hidden from source control,
|
|
as well as separate configuration parameters from the more complex
|
|
database setup in this file.
|
|
|
|
"""
|
|
|
|
import sys
|
|
import os
|
|
import logging
|
|
import tempfile
|
|
import yaml
|
|
|
|
from datetime import datetime
|
|
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
|
|
def eprint(*args, **kwargs):
|
|
""" Print a warning message to stderr """
|
|
print(*args, file=sys.stderr, **kwargs)
|
|
|
|
|
|
# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
|
|
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
|
|
|
cfg_filename = os.path.join(BASE_DIR, 'config.yaml')
|
|
|
|
if not os.path.exists(cfg_filename):
|
|
CONFIG = {}
|
|
eprint("Warning: config.yaml not found - using default settings")
|
|
else:
|
|
with open(cfg_filename, 'r') as cfg:
|
|
CONFIG = yaml.safe_load(cfg)
|
|
|
|
# Read the autogenerated key-file
|
|
key_file = open(os.path.join(BASE_DIR, 'secret_key.txt'), 'r')
|
|
|
|
SECRET_KEY = key_file.read().strip()
|
|
|
|
# Default action is to run the system in Debug mode
|
|
# SECURITY WARNING: don't run with debug turned on in production!
|
|
DEBUG = CONFIG.get('debug', True)
|
|
|
|
# List of allowed hosts (default = allow all)
|
|
ALLOWED_HOSTS = CONFIG.get('allowed_hosts', ['*'])
|
|
|
|
# Cross Origin Resource Sharing (CORS) options
|
|
|
|
# Only allow CORS access to API
|
|
CORS_URLS_REGEX = r'^/api/.*$'
|
|
|
|
# Extract CORS options from configuration file
|
|
cors_opt = CONFIG.get('cors', None)
|
|
|
|
if cors_opt:
|
|
CORS_ORIGIN_ALLOW_ALL = cors_opt.get('allow_all', False)
|
|
|
|
if not CORS_ORIGIN_ALLOW_ALL:
|
|
CORS_ORIGIN_WHITELIST = cors_opt.get('whitelist', [])
|
|
|
|
if DEBUG:
|
|
# will output to your console
|
|
logging.basicConfig(
|
|
level=logging.DEBUG,
|
|
format='%(asctime)s %(levelname)s %(message)s',
|
|
)
|
|
|
|
# Web URL endpoint for served static files
|
|
STATIC_URL = '/static/'
|
|
|
|
# The filesystem location for served static files
|
|
STATIC_ROOT = os.path.abspath(CONFIG.get('static_root', os.path.join(BASE_DIR, 'static')))
|
|
|
|
STATICFILES_DIRS = [
|
|
os.path.join(BASE_DIR, 'InvenTree', 'static'),
|
|
]
|
|
|
|
# Color Themes Directory
|
|
STATIC_COLOR_THEMES_DIR = os.path.join(STATIC_ROOT, 'css', 'color-themes')
|
|
|
|
# Web URL endpoint for served media files
|
|
MEDIA_URL = '/media/'
|
|
|
|
# The filesystem location for served static files
|
|
MEDIA_ROOT = os.path.abspath(CONFIG.get('media_root', os.path.join(BASE_DIR, 'media')))
|
|
|
|
if DEBUG:
|
|
print("InvenTree running in DEBUG mode")
|
|
print("MEDIA_ROOT:", MEDIA_ROOT)
|
|
print("STATIC_ROOT:", STATIC_ROOT)
|
|
|
|
# Does the user wish to use the sentry.io integration?
|
|
sentry_opts = CONFIG.get('sentry', {})
|
|
|
|
if sentry_opts.get('enabled', False):
|
|
dsn = sentry_opts.get('dsn', None)
|
|
|
|
if dsn is not None:
|
|
# Try to import required modules (exit if not installed)
|
|
try:
|
|
import sentry_sdk
|
|
from sentry_sdk.integrations.django import DjangoIntegration
|
|
|
|
sentry_sdk.init(dsn=dsn, integrations=[DjangoIntegration()], send_default_pii=True)
|
|
|
|
except ModuleNotFoundError:
|
|
print("sentry_sdk module not found. Install using 'pip install sentry-sdk'")
|
|
sys.exit(-1)
|
|
|
|
else:
|
|
print("Warning: Sentry.io DSN not specified")
|
|
|
|
# Application definition
|
|
|
|
INSTALLED_APPS = [
|
|
|
|
# Core django modules
|
|
'django.contrib.admin',
|
|
'django.contrib.auth',
|
|
'django.contrib.contenttypes',
|
|
'django.contrib.sessions',
|
|
'django.contrib.messages',
|
|
'django.contrib.staticfiles',
|
|
|
|
# InvenTree apps
|
|
'build.apps.BuildConfig',
|
|
'common.apps.CommonConfig',
|
|
'company.apps.CompanyConfig',
|
|
'label.apps.LabelConfig',
|
|
'order.apps.OrderConfig',
|
|
'part.apps.PartConfig',
|
|
'report.apps.ReportConfig',
|
|
'stock.apps.StockConfig',
|
|
'users.apps.UsersConfig',
|
|
|
|
# Third part add-ons
|
|
'django_filters', # Extended filter functionality
|
|
'dbbackup', # Database backup / restore
|
|
'rest_framework', # DRF (Django Rest Framework)
|
|
'rest_framework.authtoken', # Token authentication for API
|
|
'corsheaders', # Cross-origin Resource Sharing for DRF
|
|
'crispy_forms', # Improved form rendering
|
|
'import_export', # Import / export tables to file
|
|
'django_cleanup', # Automatically delete orphaned MEDIA files
|
|
'qr_code', # Generate QR codes
|
|
'mptt', # Modified Preorder Tree Traversal
|
|
'markdownx', # Markdown editing
|
|
'markdownify', # Markdown template rendering
|
|
'django_tex', # LaTeX output
|
|
'django_admin_shell', # Python shell for the admin interface
|
|
'error_report', # Error reporting in the admin interface
|
|
|
|
]
|
|
|
|
LOGGING = {
|
|
'version': 1,
|
|
'disable_existing_loggers': False,
|
|
'handlers': {
|
|
'console': {
|
|
'level': 'DEBUG',
|
|
'class': 'logging.StreamHandler',
|
|
},
|
|
},
|
|
}
|
|
|
|
MIDDLEWARE = CONFIG.get('middleware', [
|
|
'django.middleware.security.SecurityMiddleware',
|
|
'django.contrib.sessions.middleware.SessionMiddleware',
|
|
'django.middleware.locale.LocaleMiddleware',
|
|
'django.middleware.common.CommonMiddleware',
|
|
'django.middleware.csrf.CsrfViewMiddleware',
|
|
'corsheaders.middleware.CorsMiddleware',
|
|
'django.contrib.auth.middleware.AuthenticationMiddleware',
|
|
'django.contrib.messages.middleware.MessageMiddleware',
|
|
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
|
'InvenTree.middleware.AuthRequiredMiddleware'
|
|
])
|
|
|
|
# Error reporting middleware
|
|
MIDDLEWARE.append('error_report.middleware.ExceptionProcessor')
|
|
|
|
AUTHENTICATION_BACKENDS = CONFIG.get('authentication_backends', [
|
|
'django.contrib.auth.backends.ModelBackend'
|
|
])
|
|
|
|
# If the debug toolbar is enabled, add the modules
|
|
if DEBUG and CONFIG.get('debug_toolbar', False):
|
|
print("Running with DEBUG_TOOLBAR enabled")
|
|
INSTALLED_APPS.append('debug_toolbar')
|
|
MIDDLEWARE.append('debug_toolbar.middleware.DebugToolbarMiddleware')
|
|
|
|
ROOT_URLCONF = 'InvenTree.urls'
|
|
|
|
TEMPLATES = [
|
|
{
|
|
'BACKEND': 'django.template.backends.django.DjangoTemplates',
|
|
'DIRS': [
|
|
os.path.join(BASE_DIR, 'templates'),
|
|
# Allow templates in the reporting directory to be accessed
|
|
os.path.join(MEDIA_ROOT, 'report'),
|
|
],
|
|
'APP_DIRS': True,
|
|
'OPTIONS': {
|
|
'context_processors': [
|
|
'django.template.context_processors.debug',
|
|
'django.template.context_processors.request',
|
|
'django.template.context_processors.i18n',
|
|
'django.contrib.auth.context_processors.auth',
|
|
'django.contrib.messages.context_processors.messages',
|
|
'InvenTree.context.status_codes',
|
|
'InvenTree.context.user_roles',
|
|
],
|
|
},
|
|
},
|
|
# Backend for LaTeX report rendering
|
|
{
|
|
'NAME': 'tex',
|
|
'BACKEND': 'django_tex.engine.TeXEngine',
|
|
'DIRS': [
|
|
os.path.join(MEDIA_ROOT, 'report'),
|
|
]
|
|
},
|
|
]
|
|
|
|
REST_FRAMEWORK = {
|
|
'EXCEPTION_HANDLER': 'rest_framework.views.exception_handler',
|
|
'DATETIME_FORMAT': '%Y-%m-%d %H:%M',
|
|
'DEFAULT_AUTHENTICATION_CLASSES': (
|
|
'rest_framework.authentication.BasicAuthentication',
|
|
'rest_framework.authentication.SessionAuthentication',
|
|
'rest_framework.authentication.TokenAuthentication',
|
|
),
|
|
'DEFAULT_PERMISSION_CLASSES': (
|
|
'rest_framework.permissions.IsAuthenticated',
|
|
'rest_framework.permissions.DjangoModelPermissions',
|
|
),
|
|
'DEFAULT_SCHEMA_CLASS': 'rest_framework.schemas.coreapi.AutoSchema'
|
|
}
|
|
|
|
WSGI_APPLICATION = 'InvenTree.wsgi.application'
|
|
|
|
# Markdownx configuration
|
|
# Ref: https://neutronx.github.io/django-markdownx/customization/
|
|
MARKDOWNX_MEDIA_PATH = datetime.now().strftime('markdownx/%Y/%m/%d')
|
|
|
|
# Markdownify configuration
|
|
# Ref: https://django-markdownify.readthedocs.io/en/latest/settings.html
|
|
|
|
MARKDOWNIFY_WHITELIST_TAGS = [
|
|
'a',
|
|
'abbr',
|
|
'b',
|
|
'blockquote',
|
|
'em',
|
|
'h1', 'h2', 'h3',
|
|
'i',
|
|
'img',
|
|
'li',
|
|
'ol',
|
|
'p',
|
|
'strong',
|
|
'ul'
|
|
]
|
|
|
|
MARKDOWNIFY_WHITELIST_ATTRS = [
|
|
'href',
|
|
'src',
|
|
'alt',
|
|
]
|
|
|
|
MARKDOWNIFY_BLEACH = False
|
|
|
|
DATABASES = {}
|
|
|
|
"""
|
|
When running unit tests, enforce usage of sqlite3 database,
|
|
so that the tests can be run in RAM without any setup requirements
|
|
"""
|
|
if 'test' in sys.argv:
|
|
eprint('InvenTree: Running tests - Using sqlite3 memory database')
|
|
DATABASES['default'] = {
|
|
# Ensure sqlite3 backend is being used
|
|
'ENGINE': 'django.db.backends.sqlite3',
|
|
# Doesn't matter what the database is called, it is executed in RAM
|
|
'NAME': 'ram_test_db.sqlite3',
|
|
}
|
|
|
|
# Database backend selection
|
|
else:
|
|
if 'database' in CONFIG:
|
|
DATABASES['default'] = CONFIG['database']
|
|
else:
|
|
eprint("Warning: Database backend not specified - using default (sqlite)")
|
|
DATABASES['default'] = {
|
|
'ENGINE': 'django.db.backends.sqlite3',
|
|
'NAME': os.path.join(BASE_DIR, 'inventree_db.sqlite3'),
|
|
}
|
|
|
|
CACHES = {
|
|
'default': {
|
|
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
|
|
},
|
|
'qr-code': {
|
|
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
|
|
'LOCATION': 'qr-code-cache',
|
|
'TIMEOUT': 3600
|
|
}
|
|
}
|
|
|
|
QR_CODE_CACHE_ALIAS = 'qr-code'
|
|
|
|
# Password validation
|
|
# https://docs.djangoproject.com/en/1.10/ref/settings/#auth-password-validators
|
|
|
|
AUTH_PASSWORD_VALIDATORS = [
|
|
{
|
|
'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',
|
|
},
|
|
{
|
|
'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',
|
|
},
|
|
{
|
|
'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',
|
|
},
|
|
{
|
|
'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',
|
|
},
|
|
]
|
|
|
|
# Extra (optional) URL validators
|
|
# See https://docs.djangoproject.com/en/2.2/ref/validators/#django.core.validators.URLValidator
|
|
|
|
EXTRA_URL_SCHEMES = CONFIG.get('extra_url_schemes', [])
|
|
|
|
if not type(EXTRA_URL_SCHEMES) in [list]:
|
|
eprint("Warning: extra_url_schemes not correctly formatted")
|
|
EXTRA_URL_SCHEMES = []
|
|
|
|
# Internationalization
|
|
# https://docs.djangoproject.com/en/1.10/topics/i18n/
|
|
|
|
LANGUAGE_CODE = CONFIG.get('language', 'en-us')
|
|
|
|
# If a new language translation is supported, it must be added here
|
|
LANGUAGES = [
|
|
('en', _('English')),
|
|
('de', _('German')),
|
|
('fr', _('French')),
|
|
('pk', _('Polish')),
|
|
]
|
|
|
|
LOCALE_PATHS = (
|
|
os.path.join(BASE_DIR, 'locale/'),
|
|
)
|
|
|
|
TIME_ZONE = CONFIG.get('timezone', 'UTC')
|
|
|
|
USE_I18N = True
|
|
|
|
USE_L10N = True
|
|
|
|
USE_TZ = True
|
|
|
|
DATE_INPUT_FORMATS = [
|
|
"%Y-%m-%d",
|
|
]
|
|
|
|
# LaTeX rendering settings (django-tex)
|
|
LATEX_SETTINGS = CONFIG.get('latex', {})
|
|
|
|
# Is LaTeX rendering enabled? (Off by default)
|
|
LATEX_ENABLED = LATEX_SETTINGS.get('enabled', False)
|
|
|
|
# Set the latex interpreter in the config.yaml settings file
|
|
LATEX_INTERPRETER = LATEX_SETTINGS.get('interpreter', 'pdflatex')
|
|
|
|
LATEX_INTERPRETER_OPTIONS = LATEX_SETTINGS.get('options', '')
|
|
|
|
LATEX_GRAPHICSPATH = [
|
|
# Allow LaTeX files to access the report assets directory
|
|
os.path.join(MEDIA_ROOT, "report", "assets"),
|
|
]
|
|
|
|
# crispy forms use the bootstrap templates
|
|
CRISPY_TEMPLATE_PACK = 'bootstrap3'
|
|
|
|
# Use database transactions when importing / exporting data
|
|
IMPORT_EXPORT_USE_TRANSACTIONS = True
|
|
|
|
# Settings for dbbsettings app
|
|
DBBACKUP_STORAGE = 'django.core.files.storage.FileSystemStorage'
|
|
DBBACKUP_STORAGE_OPTIONS = {
|
|
'location': CONFIG.get('backup_dir', tempfile.gettempdir()),
|
|
}
|
|
|
|
# Internal IP addresses allowed to see the debug toolbar
|
|
INTERNAL_IPS = [
|
|
'127.0.0.1',
|
|
]
|