2017-06-23 21:00:42 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
|
|
Django settings for funkwhale_api project.
|
|
|
|
|
|
|
|
For more information on this file, see
|
|
|
|
https://docs.djangoproject.com/en/dev/topics/settings/
|
|
|
|
|
|
|
|
For the full list of settings and their values, see
|
|
|
|
https://docs.djangoproject.com/en/dev/ref/settings/
|
|
|
|
"""
|
|
|
|
from __future__ import absolute_import, unicode_literals
|
|
|
|
|
2018-06-10 09:29:24 +00:00
|
|
|
import datetime
|
|
|
|
from urllib.parse import urlparse, urlsplit
|
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
import environ
|
2018-04-17 19:26:28 +00:00
|
|
|
from celery.schedules import crontab
|
|
|
|
|
2018-02-17 20:25:21 +00:00
|
|
|
from funkwhale_api import __version__
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
ROOT_DIR = environ.Path(__file__) - 3 # (/a/b/myfile.py - 3 = /)
|
2018-06-09 13:36:16 +00:00
|
|
|
APPS_DIR = ROOT_DIR.path("funkwhale_api")
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
env = environ.Env()
|
|
|
|
try:
|
2018-06-09 13:36:16 +00:00
|
|
|
env.read_env(ROOT_DIR.file(".env"))
|
2017-06-23 21:00:42 +00:00
|
|
|
except FileNotFoundError:
|
|
|
|
pass
|
2018-02-18 13:28:44 +00:00
|
|
|
|
2018-04-08 16:19:32 +00:00
|
|
|
FUNKWHALE_HOSTNAME = None
|
2018-06-09 13:36:16 +00:00
|
|
|
FUNKWHALE_HOSTNAME_SUFFIX = env("FUNKWHALE_HOSTNAME_SUFFIX", default=None)
|
|
|
|
FUNKWHALE_HOSTNAME_PREFIX = env("FUNKWHALE_HOSTNAME_PREFIX", default=None)
|
2018-04-08 16:19:32 +00:00
|
|
|
if FUNKWHALE_HOSTNAME_PREFIX and FUNKWHALE_HOSTNAME_SUFFIX:
|
|
|
|
# We're in traefik case, in development
|
2018-06-09 13:36:16 +00:00
|
|
|
FUNKWHALE_HOSTNAME = "{}.{}".format(
|
|
|
|
FUNKWHALE_HOSTNAME_PREFIX, FUNKWHALE_HOSTNAME_SUFFIX
|
|
|
|
)
|
|
|
|
FUNKWHALE_PROTOCOL = env("FUNKWHALE_PROTOCOL", default="https")
|
2018-04-08 16:19:32 +00:00
|
|
|
else:
|
|
|
|
try:
|
2018-06-09 13:36:16 +00:00
|
|
|
FUNKWHALE_HOSTNAME = env("FUNKWHALE_HOSTNAME")
|
|
|
|
FUNKWHALE_PROTOCOL = env("FUNKWHALE_PROTOCOL", default="https")
|
2018-04-08 16:19:32 +00:00
|
|
|
except Exception:
|
2018-06-09 13:36:16 +00:00
|
|
|
FUNKWHALE_URL = env("FUNKWHALE_URL")
|
2018-04-08 16:19:32 +00:00
|
|
|
_parsed = urlsplit(FUNKWHALE_URL)
|
|
|
|
FUNKWHALE_HOSTNAME = _parsed.netloc
|
|
|
|
FUNKWHALE_PROTOCOL = _parsed.scheme
|
|
|
|
|
2018-06-09 13:36:16 +00:00
|
|
|
FUNKWHALE_URL = "{}://{}".format(FUNKWHALE_PROTOCOL, FUNKWHALE_HOSTNAME)
|
2018-04-08 16:19:32 +00:00
|
|
|
|
2018-03-28 22:00:01 +00:00
|
|
|
|
2018-04-28 04:11:50 +00:00
|
|
|
# XXX: deprecated, see #186
|
2018-06-09 13:36:16 +00:00
|
|
|
FEDERATION_ENABLED = env.bool("FEDERATION_ENABLED", default=True)
|
|
|
|
FEDERATION_HOSTNAME = env("FEDERATION_HOSTNAME", default=FUNKWHALE_HOSTNAME)
|
2018-04-28 04:11:50 +00:00
|
|
|
# XXX: deprecated, see #186
|
2018-06-09 13:36:16 +00:00
|
|
|
FEDERATION_COLLECTION_PAGE_SIZE = env.int("FEDERATION_COLLECTION_PAGE_SIZE", default=50)
|
2018-04-28 04:11:50 +00:00
|
|
|
# XXX: deprecated, see #186
|
2018-04-02 20:43:59 +00:00
|
|
|
FEDERATION_MUSIC_NEEDS_APPROVAL = env.bool(
|
2018-06-09 13:36:16 +00:00
|
|
|
"FEDERATION_MUSIC_NEEDS_APPROVAL", default=True
|
2018-04-02 20:43:59 +00:00
|
|
|
)
|
2018-04-28 04:11:50 +00:00
|
|
|
# XXX: deprecated, see #186
|
2018-06-09 13:36:16 +00:00
|
|
|
FEDERATION_ACTOR_FETCH_DELAY = env.int("FEDERATION_ACTOR_FETCH_DELAY", default=60 * 12)
|
|
|
|
ALLOWED_HOSTS = env.list("DJANGO_ALLOWED_HOSTS")
|
2018-02-18 13:28:44 +00:00
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
# APP CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
DJANGO_APPS = (
|
2018-06-09 13:36:16 +00:00
|
|
|
"channels",
|
2017-06-23 21:00:42 +00:00
|
|
|
# Default Django apps:
|
2018-06-09 13:36:16 +00:00
|
|
|
"django.contrib.auth",
|
|
|
|
"django.contrib.contenttypes",
|
|
|
|
"django.contrib.sessions",
|
|
|
|
"django.contrib.sites",
|
|
|
|
"django.contrib.messages",
|
|
|
|
"django.contrib.staticfiles",
|
|
|
|
"django.contrib.postgres",
|
2017-06-23 21:00:42 +00:00
|
|
|
# Useful template tags:
|
|
|
|
# 'django.contrib.humanize',
|
|
|
|
# Admin
|
2018-06-09 13:36:16 +00:00
|
|
|
"django.contrib.admin",
|
2017-06-23 21:00:42 +00:00
|
|
|
)
|
|
|
|
THIRD_PARTY_APPS = (
|
|
|
|
# 'crispy_forms', # Form layouts
|
2018-06-09 13:36:16 +00:00
|
|
|
"allauth", # registration
|
|
|
|
"allauth.account", # registration
|
|
|
|
"allauth.socialaccount", # registration
|
|
|
|
"corsheaders",
|
|
|
|
"rest_framework",
|
|
|
|
"rest_framework.authtoken",
|
|
|
|
"taggit",
|
|
|
|
"rest_auth",
|
|
|
|
"rest_auth.registration",
|
|
|
|
"dynamic_preferences",
|
|
|
|
"django_filters",
|
|
|
|
"django_cleanup",
|
2018-07-13 12:10:39 +00:00
|
|
|
"versatileimagefield",
|
2017-06-23 21:00:42 +00:00
|
|
|
)
|
|
|
|
|
2018-02-17 20:25:21 +00:00
|
|
|
|
|
|
|
# Sentry
|
|
|
|
RAVEN_ENABLED = env.bool("RAVEN_ENABLED", default=False)
|
2018-06-09 13:36:16 +00:00
|
|
|
RAVEN_DSN = env("RAVEN_DSN", default="")
|
2018-02-17 20:25:21 +00:00
|
|
|
|
|
|
|
if RAVEN_ENABLED:
|
|
|
|
RAVEN_CONFIG = {
|
2018-06-09 13:36:16 +00:00
|
|
|
"dsn": RAVEN_DSN,
|
2018-02-17 20:25:21 +00:00
|
|
|
# If you are using git, you can also automatically configure the
|
|
|
|
# release based on the git info.
|
2018-06-09 13:36:16 +00:00
|
|
|
"release": __version__,
|
2018-02-17 20:25:21 +00:00
|
|
|
}
|
2018-06-09 13:36:16 +00:00
|
|
|
THIRD_PARTY_APPS += ("raven.contrib.django.raven_compat",)
|
2018-02-17 20:25:21 +00:00
|
|
|
|
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
# Apps specific for this project go here.
|
|
|
|
LOCAL_APPS = (
|
2018-06-09 13:36:16 +00:00
|
|
|
"funkwhale_api.common",
|
|
|
|
"funkwhale_api.activity.apps.ActivityConfig",
|
|
|
|
"funkwhale_api.users", # custom users app
|
2017-06-23 21:00:42 +00:00
|
|
|
# Your stuff: custom apps go here
|
2018-06-09 13:36:16 +00:00
|
|
|
"funkwhale_api.instance",
|
|
|
|
"funkwhale_api.music",
|
|
|
|
"funkwhale_api.requests",
|
|
|
|
"funkwhale_api.favorites",
|
|
|
|
"funkwhale_api.federation",
|
|
|
|
"funkwhale_api.radios",
|
|
|
|
"funkwhale_api.history",
|
|
|
|
"funkwhale_api.playlists",
|
|
|
|
"funkwhale_api.providers.acoustid",
|
|
|
|
"funkwhale_api.subsonic",
|
2017-06-23 21:00:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#installed-apps
|
2018-02-17 20:25:21 +00:00
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
INSTALLED_APPS = DJANGO_APPS + THIRD_PARTY_APPS + LOCAL_APPS
|
|
|
|
|
|
|
|
# MIDDLEWARE CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
2017-12-15 22:48:45 +00:00
|
|
|
MIDDLEWARE = (
|
2017-06-23 21:00:42 +00:00
|
|
|
# Make sure djangosecure.middleware.SecurityMiddleware is listed first
|
2018-06-09 13:36:16 +00:00
|
|
|
"django.contrib.sessions.middleware.SessionMiddleware",
|
|
|
|
"corsheaders.middleware.CorsMiddleware",
|
|
|
|
"django.middleware.common.CommonMiddleware",
|
|
|
|
"django.middleware.csrf.CsrfViewMiddleware",
|
|
|
|
"django.contrib.auth.middleware.AuthenticationMiddleware",
|
|
|
|
"django.contrib.messages.middleware.MessageMiddleware",
|
|
|
|
"django.middleware.clickjacking.XFrameOptionsMiddleware",
|
2018-06-17 15:53:40 +00:00
|
|
|
"funkwhale_api.users.middleware.RecordActivityMiddleware",
|
2017-06-23 21:00:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# MIGRATIONS CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
2018-06-09 13:36:16 +00:00
|
|
|
MIGRATION_MODULES = {"sites": "funkwhale_api.contrib.sites.migrations"}
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# DEBUG
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#debug
|
|
|
|
DEBUG = env.bool("DJANGO_DEBUG", False)
|
|
|
|
|
|
|
|
# FIXTURE CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#std:setting-FIXTURE_DIRS
|
2018-06-09 13:36:16 +00:00
|
|
|
FIXTURE_DIRS = (str(APPS_DIR.path("fixtures")),)
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# EMAIL CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
2018-05-06 14:23:23 +00:00
|
|
|
|
|
|
|
# EMAIL
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
DEFAULT_FROM_EMAIL = env(
|
2018-06-09 13:36:16 +00:00
|
|
|
"DEFAULT_FROM_EMAIL", default="Funkwhale <noreply@{}>".format(FUNKWHALE_HOSTNAME)
|
|
|
|
)
|
2018-05-06 14:23:23 +00:00
|
|
|
|
2018-06-09 13:36:16 +00:00
|
|
|
EMAIL_SUBJECT_PREFIX = env("EMAIL_SUBJECT_PREFIX", default="[Funkwhale] ")
|
|
|
|
SERVER_EMAIL = env("SERVER_EMAIL", default=DEFAULT_FROM_EMAIL)
|
2018-05-06 14:23:23 +00:00
|
|
|
|
|
|
|
|
2018-06-09 13:36:16 +00:00
|
|
|
EMAIL_CONFIG = env.email_url("EMAIL_CONFIG", default="consolemail://")
|
2018-05-06 11:48:23 +00:00
|
|
|
|
|
|
|
vars().update(EMAIL_CONFIG)
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# DATABASE CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#databases
|
|
|
|
DATABASES = {
|
|
|
|
# Raises ImproperlyConfigured exception if DATABASE_URL not in os.environ
|
2018-06-09 13:36:16 +00:00
|
|
|
"default": env.db("DATABASE_URL")
|
2017-06-23 21:00:42 +00:00
|
|
|
}
|
2018-06-09 13:36:16 +00:00
|
|
|
DATABASES["default"]["ATOMIC_REQUESTS"] = True
|
2017-06-23 21:00:42 +00:00
|
|
|
#
|
|
|
|
# DATABASES = {
|
|
|
|
# 'default': {
|
|
|
|
# 'ENGINE': 'django.db.backends.sqlite3',
|
|
|
|
# 'NAME': 'db.sqlite3',
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# GENERAL CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Local time zone for this installation. Choices can be found here:
|
|
|
|
# http://en.wikipedia.org/wiki/List_of_tz_zones_by_name
|
|
|
|
# although not all choices may be available on all operating systems.
|
|
|
|
# In a Windows environment this must be set to your system time zone.
|
2018-06-09 13:36:16 +00:00
|
|
|
TIME_ZONE = "UTC"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#language-code
|
2018-06-09 13:36:16 +00:00
|
|
|
LANGUAGE_CODE = "en-us"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#site-id
|
|
|
|
SITE_ID = 1
|
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#use-i18n
|
|
|
|
USE_I18N = True
|
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#use-l10n
|
|
|
|
USE_L10N = True
|
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#use-tz
|
|
|
|
USE_TZ = True
|
|
|
|
|
|
|
|
# TEMPLATE CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#templates
|
|
|
|
TEMPLATES = [
|
|
|
|
{
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#std:setting-TEMPLATES-BACKEND
|
2018-06-09 13:36:16 +00:00
|
|
|
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
2017-06-23 21:00:42 +00:00
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#template-dirs
|
2018-06-09 13:36:16 +00:00
|
|
|
"DIRS": [str(APPS_DIR.path("templates"))],
|
|
|
|
"OPTIONS": {
|
2017-06-23 21:00:42 +00:00
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#template-debug
|
2018-06-09 13:36:16 +00:00
|
|
|
"debug": DEBUG,
|
2017-06-23 21:00:42 +00:00
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#template-loaders
|
|
|
|
# https://docs.djangoproject.com/en/dev/ref/templates/api/#loader-types
|
2018-06-09 13:36:16 +00:00
|
|
|
"loaders": [
|
|
|
|
"django.template.loaders.filesystem.Loader",
|
|
|
|
"django.template.loaders.app_directories.Loader",
|
2017-06-23 21:00:42 +00:00
|
|
|
],
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#template-context-processors
|
2018-06-09 13:36:16 +00:00
|
|
|
"context_processors": [
|
|
|
|
"django.template.context_processors.debug",
|
|
|
|
"django.template.context_processors.request",
|
|
|
|
"django.contrib.auth.context_processors.auth",
|
|
|
|
"django.template.context_processors.i18n",
|
|
|
|
"django.template.context_processors.media",
|
|
|
|
"django.template.context_processors.static",
|
|
|
|
"django.template.context_processors.tz",
|
|
|
|
"django.contrib.messages.context_processors.messages",
|
2017-06-23 21:00:42 +00:00
|
|
|
# Your stuff: custom template context processors go here
|
|
|
|
],
|
|
|
|
},
|
2018-06-09 13:36:16 +00:00
|
|
|
}
|
2017-06-23 21:00:42 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
# See: http://django-crispy-forms.readthedocs.org/en/latest/install.html#template-packs
|
2018-06-09 13:36:16 +00:00
|
|
|
CRISPY_TEMPLATE_PACK = "bootstrap3"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# STATIC FILE CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#static-root
|
2018-06-09 13:36:16 +00:00
|
|
|
STATIC_ROOT = env("STATIC_ROOT", default=str(ROOT_DIR("staticfiles")))
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#static-url
|
2018-06-09 13:36:16 +00:00
|
|
|
STATIC_URL = env("STATIC_URL", default="/staticfiles/")
|
|
|
|
DEFAULT_FILE_STORAGE = "funkwhale_api.common.storage.ASCIIFileSystemStorage"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/contrib/staticfiles/#std:setting-STATICFILES_DIRS
|
2018-06-09 13:36:16 +00:00
|
|
|
STATICFILES_DIRS = (str(APPS_DIR.path("static")),)
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/contrib/staticfiles/#staticfiles-finders
|
|
|
|
STATICFILES_FINDERS = (
|
2018-06-09 13:36:16 +00:00
|
|
|
"django.contrib.staticfiles.finders.FileSystemFinder",
|
|
|
|
"django.contrib.staticfiles.finders.AppDirectoriesFinder",
|
2017-06-23 21:00:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# MEDIA CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#media-root
|
2018-06-09 13:36:16 +00:00
|
|
|
MEDIA_ROOT = env("MEDIA_ROOT", default=str(APPS_DIR("media")))
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#media-url
|
2018-06-09 13:36:16 +00:00
|
|
|
MEDIA_URL = env("MEDIA_URL", default="/media/")
|
2018-09-06 18:35:02 +00:00
|
|
|
FILE_UPLOAD_PERMISSIONS = 0o644
|
2017-06-23 21:00:42 +00:00
|
|
|
# URL Configuration
|
|
|
|
# ------------------------------------------------------------------------------
|
2018-06-09 13:36:16 +00:00
|
|
|
ROOT_URLCONF = "config.urls"
|
2017-06-23 21:00:42 +00:00
|
|
|
# See: https://docs.djangoproject.com/en/dev/ref/settings/#wsgi-application
|
2018-06-09 13:36:16 +00:00
|
|
|
WSGI_APPLICATION = "config.wsgi.application"
|
2018-02-25 12:05:01 +00:00
|
|
|
ASGI_APPLICATION = "config.routing.application"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
2018-06-02 07:17:32 +00:00
|
|
|
# This ensures that Django will be able to detect a secure connection
|
2018-06-09 13:36:16 +00:00
|
|
|
SECURE_PROXY_SSL_HEADER = ("HTTP_X_FORWARDED_PROTO", "https")
|
2018-06-02 07:17:32 +00:00
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
# AUTHENTICATION CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
AUTHENTICATION_BACKENDS = (
|
2018-06-09 13:36:16 +00:00
|
|
|
"django.contrib.auth.backends.ModelBackend",
|
|
|
|
"allauth.account.auth_backends.AuthenticationBackend",
|
2017-06-23 21:00:42 +00:00
|
|
|
)
|
2018-05-06 09:20:18 +00:00
|
|
|
SESSION_COOKIE_HTTPONLY = False
|
2017-06-23 21:00:42 +00:00
|
|
|
# Some really nice defaults
|
2018-06-09 13:36:16 +00:00
|
|
|
ACCOUNT_AUTHENTICATION_METHOD = "username_email"
|
2017-06-23 21:00:42 +00:00
|
|
|
ACCOUNT_EMAIL_REQUIRED = True
|
2018-06-09 13:36:16 +00:00
|
|
|
ACCOUNT_EMAIL_VERIFICATION = "mandatory"
|
2018-07-22 11:05:43 +00:00
|
|
|
ACCOUNT_USERNAME_VALIDATORS = "funkwhale_api.users.serializers.username_validators"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
# Custom user app defaults
|
|
|
|
# Select the correct user model
|
2018-06-09 13:36:16 +00:00
|
|
|
AUTH_USER_MODEL = "users.User"
|
|
|
|
LOGIN_REDIRECT_URL = "users:redirect"
|
|
|
|
LOGIN_URL = "account_login"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
2018-08-22 18:10:39 +00:00
|
|
|
# LDAP AUTHENTICATION CONFIGURATION
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
AUTH_LDAP_ENABLED = env.bool("LDAP_ENABLED", default=False)
|
|
|
|
if AUTH_LDAP_ENABLED:
|
|
|
|
|
|
|
|
# Import the LDAP modules here; this way, we don't need the dependency unless someone
|
|
|
|
# actually enables the LDAP support
|
|
|
|
import ldap
|
|
|
|
from django_auth_ldap.config import LDAPSearch, LDAPSearchUnion, GroupOfNamesType
|
|
|
|
|
|
|
|
# Add LDAP to the authentication backends
|
|
|
|
AUTHENTICATION_BACKENDS += ("django_auth_ldap.backend.LDAPBackend",)
|
|
|
|
|
|
|
|
# Basic configuration
|
|
|
|
AUTH_LDAP_SERVER_URI = env("LDAP_SERVER_URI")
|
|
|
|
AUTH_LDAP_BIND_DN = env("LDAP_BIND_DN", default="")
|
|
|
|
AUTH_LDAP_BIND_PASSWORD = env("LDAP_BIND_PASSWORD", default="")
|
|
|
|
AUTH_LDAP_SEARCH_FILTER = env("LDAP_SEARCH_FILTER", default="(uid={0})").format(
|
|
|
|
"%(user)s"
|
|
|
|
)
|
|
|
|
AUTH_LDAP_START_TLS = env.bool("LDAP_START_TLS", default=False)
|
|
|
|
|
|
|
|
DEFAULT_USER_ATTR_MAP = [
|
|
|
|
"first_name:givenName",
|
|
|
|
"last_name:sn",
|
|
|
|
"username:cn",
|
|
|
|
"email:mail",
|
|
|
|
]
|
|
|
|
LDAP_USER_ATTR_MAP = env.list("LDAP_USER_ATTR_MAP", default=DEFAULT_USER_ATTR_MAP)
|
|
|
|
AUTH_LDAP_USER_ATTR_MAP = {}
|
|
|
|
for m in LDAP_USER_ATTR_MAP:
|
|
|
|
funkwhale_field, ldap_field = m.split(":")
|
|
|
|
AUTH_LDAP_USER_ATTR_MAP[funkwhale_field.strip()] = ldap_field.strip()
|
|
|
|
|
|
|
|
# Determine root DN supporting multiple root DNs
|
|
|
|
AUTH_LDAP_ROOT_DN = env("LDAP_ROOT_DN")
|
|
|
|
AUTH_LDAP_ROOT_DN_LIST = []
|
|
|
|
for ROOT_DN in AUTH_LDAP_ROOT_DN.split():
|
|
|
|
AUTH_LDAP_ROOT_DN_LIST.append(
|
|
|
|
LDAPSearch(ROOT_DN, ldap.SCOPE_SUBTREE, AUTH_LDAP_SEARCH_FILTER)
|
|
|
|
)
|
|
|
|
# Search for the user in all the root DNs
|
|
|
|
AUTH_LDAP_USER_SEARCH = LDAPSearchUnion(*AUTH_LDAP_ROOT_DN_LIST)
|
|
|
|
|
|
|
|
# Search for group types
|
|
|
|
LDAP_GROUP_DN = env("LDAP_GROUP_DN", default="")
|
|
|
|
if LDAP_GROUP_DN:
|
|
|
|
AUTH_LDAP_GROUP_DN = LDAP_GROUP_DN
|
|
|
|
# Get filter
|
|
|
|
AUTH_LDAP_GROUP_FILTER = env("LDAP_GROUP_FILER", default="")
|
|
|
|
# Search for the group in the specified DN
|
|
|
|
AUTH_LDAP_GROUP_SEARCH = LDAPSearch(
|
|
|
|
AUTH_LDAP_GROUP_DN, ldap.SCOPE_SUBTREE, AUTH_LDAP_GROUP_FILTER
|
|
|
|
)
|
|
|
|
AUTH_LDAP_GROUP_TYPE = GroupOfNamesType()
|
|
|
|
|
|
|
|
# Configure basic group support
|
|
|
|
LDAP_REQUIRE_GROUP = env("LDAP_REQUIRE_GROUP", default="")
|
|
|
|
if LDAP_REQUIRE_GROUP:
|
|
|
|
AUTH_LDAP_REQUIRE_GROUP = LDAP_REQUIRE_GROUP
|
|
|
|
LDAP_DENY_GROUP = env("LDAP_DENY_GROUP", default="")
|
|
|
|
if LDAP_DENY_GROUP:
|
|
|
|
AUTH_LDAP_DENY_GROUP = LDAP_DENY_GROUP
|
|
|
|
|
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
# SLUGLIFIER
|
2018-06-09 13:36:16 +00:00
|
|
|
AUTOSLUG_SLUGIFY_FUNCTION = "slugify.slugify"
|
2017-06-23 21:00:42 +00:00
|
|
|
|
2017-07-17 20:00:32 +00:00
|
|
|
CACHE_DEFAULT = "redis://127.0.0.1:6379/0"
|
2018-06-09 13:36:16 +00:00
|
|
|
CACHES = {"default": env.cache_url("CACHE_URL", default=CACHE_DEFAULT)}
|
2017-07-09 11:27:15 +00:00
|
|
|
|
2017-07-17 20:00:32 +00:00
|
|
|
CACHES["default"]["BACKEND"] = "django_redis.cache.RedisCache"
|
2018-06-09 13:36:16 +00:00
|
|
|
|
|
|
|
cache_url = urlparse(CACHES["default"]["LOCATION"])
|
2018-02-25 12:05:01 +00:00
|
|
|
CHANNEL_LAYERS = {
|
|
|
|
"default": {
|
|
|
|
"BACKEND": "channels_redis.core.RedisChannelLayer",
|
2018-06-09 13:36:16 +00:00
|
|
|
"CONFIG": {"hosts": [(cache_url.hostname, cache_url.port)]},
|
|
|
|
}
|
2018-02-25 12:05:01 +00:00
|
|
|
}
|
|
|
|
|
2017-07-17 20:00:32 +00:00
|
|
|
CACHES["default"]["OPTIONS"] = {
|
|
|
|
"CLIENT_CLASS": "django_redis.client.DefaultClient",
|
|
|
|
"IGNORE_EXCEPTIONS": True, # mimics memcache behavior.
|
2018-06-09 13:36:16 +00:00
|
|
|
# http://niwinz.github.io/django-redis/latest/#_memcached_exceptions_behavior
|
2017-07-17 20:00:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-10 10:06:46 +00:00
|
|
|
# CELERY
|
2018-06-09 13:36:16 +00:00
|
|
|
INSTALLED_APPS += ("funkwhale_api.taskapp.celery.CeleryConfig",)
|
2017-12-26 20:12:37 +00:00
|
|
|
CELERY_BROKER_URL = env(
|
2018-06-09 13:36:16 +00:00
|
|
|
"CELERY_BROKER_URL", default=env("CACHE_URL", default=CACHE_DEFAULT)
|
|
|
|
)
|
2018-06-10 10:06:46 +00:00
|
|
|
# END CELERY
|
2017-06-23 21:00:42 +00:00
|
|
|
# Location of root django.contrib.admin URL, use {% url 'admin:index' %}
|
2018-03-06 16:44:53 +00:00
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
# Your common stuff: Below this line define 3rd party library settings
|
2017-12-26 20:12:37 +00:00
|
|
|
CELERY_TASK_DEFAULT_RATE_LIMIT = 1
|
|
|
|
CELERY_TASK_TIME_LIMIT = 300
|
2018-08-19 14:22:14 +00:00
|
|
|
CELERY_BEAT_SCHEDULE = {
|
2018-06-09 13:36:16 +00:00
|
|
|
"federation.clean_music_cache": {
|
2018-08-19 14:22:14 +00:00
|
|
|
"task": "federation.clean_music_cache",
|
2018-06-09 13:36:16 +00:00
|
|
|
"schedule": crontab(hour="*/2"),
|
|
|
|
"options": {"expires": 60 * 2},
|
2018-10-24 17:44:31 +00:00
|
|
|
},
|
|
|
|
"music.clean_transcoding_cache": {
|
|
|
|
"task": "music.clean_transcoding_cache",
|
|
|
|
"schedule": crontab(hour="*"),
|
|
|
|
"options": {"expires": 60 * 2},
|
|
|
|
},
|
2018-04-17 19:26:28 +00:00
|
|
|
}
|
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
JWT_AUTH = {
|
2018-06-09 13:36:16 +00:00
|
|
|
"JWT_ALLOW_REFRESH": True,
|
|
|
|
"JWT_EXPIRATION_DELTA": datetime.timedelta(days=7),
|
|
|
|
"JWT_REFRESH_EXPIRATION_DELTA": datetime.timedelta(days=30),
|
|
|
|
"JWT_AUTH_HEADER_PREFIX": "JWT",
|
|
|
|
"JWT_GET_USER_SECRET_KEY": lambda user: user.secret_key,
|
2017-06-23 21:00:42 +00:00
|
|
|
}
|
2017-12-26 14:56:04 +00:00
|
|
|
OLD_PASSWORD_FIELD_ENABLED = True
|
2018-06-09 13:36:16 +00:00
|
|
|
ACCOUNT_ADAPTER = "funkwhale_api.users.adapters.FunkwhaleAccountAdapter"
|
2017-06-23 21:00:42 +00:00
|
|
|
CORS_ORIGIN_ALLOW_ALL = True
|
|
|
|
# CORS_ORIGIN_WHITELIST = (
|
|
|
|
# 'localhost',
|
|
|
|
# 'funkwhale.localhost',
|
|
|
|
# )
|
|
|
|
CORS_ALLOW_CREDENTIALS = True
|
2018-05-06 11:48:23 +00:00
|
|
|
|
2017-06-23 21:00:42 +00:00
|
|
|
REST_FRAMEWORK = {
|
2018-06-09 13:36:16 +00:00
|
|
|
"DEFAULT_PERMISSION_CLASSES": ("rest_framework.permissions.IsAuthenticated",),
|
|
|
|
"DEFAULT_PAGINATION_CLASS": "funkwhale_api.common.pagination.FunkwhalePagination",
|
|
|
|
"PAGE_SIZE": 25,
|
|
|
|
"DEFAULT_PARSER_CLASSES": (
|
|
|
|
"rest_framework.parsers.JSONParser",
|
|
|
|
"rest_framework.parsers.FormParser",
|
|
|
|
"rest_framework.parsers.MultiPartParser",
|
|
|
|
"funkwhale_api.federation.parsers.ActivityParser",
|
2017-06-23 21:00:42 +00:00
|
|
|
),
|
2018-06-09 13:36:16 +00:00
|
|
|
"DEFAULT_AUTHENTICATION_CLASSES": (
|
|
|
|
"funkwhale_api.common.authentication.JSONWebTokenAuthenticationQS",
|
|
|
|
"funkwhale_api.common.authentication.BearerTokenHeaderAuth",
|
2018-09-06 18:35:02 +00:00
|
|
|
"funkwhale_api.common.authentication.JSONWebTokenAuthentication",
|
2018-06-09 13:36:16 +00:00
|
|
|
"rest_framework.authentication.SessionAuthentication",
|
|
|
|
"rest_framework.authentication.BasicAuthentication",
|
2018-04-01 20:11:46 +00:00
|
|
|
),
|
2018-06-09 13:36:16 +00:00
|
|
|
"DEFAULT_FILTER_BACKENDS": (
|
|
|
|
"rest_framework.filters.OrderingFilter",
|
|
|
|
"django_filters.rest_framework.DjangoFilterBackend",
|
2017-06-23 21:00:42 +00:00
|
|
|
),
|
2018-06-09 13:36:16 +00:00
|
|
|
"DEFAULT_RENDERER_CLASSES": ("rest_framework.renderers.JSONRenderer",),
|
2017-06-23 21:00:42 +00:00
|
|
|
}
|
2018-05-15 17:06:08 +00:00
|
|
|
|
2018-06-09 13:36:16 +00:00
|
|
|
BROWSABLE_API_ENABLED = env.bool("BROWSABLE_API_ENABLED", default=False)
|
2018-05-15 17:06:08 +00:00
|
|
|
if BROWSABLE_API_ENABLED:
|
2018-06-09 13:36:16 +00:00
|
|
|
REST_FRAMEWORK["DEFAULT_RENDERER_CLASSES"] += (
|
|
|
|
"rest_framework.renderers.BrowsableAPIRenderer",
|
2018-05-15 17:06:08 +00:00
|
|
|
)
|
|
|
|
|
2018-05-06 09:30:41 +00:00
|
|
|
REST_AUTH_SERIALIZERS = {
|
2018-06-09 13:36:16 +00:00
|
|
|
"PASSWORD_RESET_SERIALIZER": "funkwhale_api.users.serializers.PasswordResetSerializer" # noqa
|
2018-05-06 09:30:41 +00:00
|
|
|
}
|
|
|
|
REST_SESSION_LOGIN = False
|
|
|
|
REST_USE_JWT = True
|
2017-06-23 21:00:42 +00:00
|
|
|
|
|
|
|
ATOMIC_REQUESTS = False
|
2017-07-11 07:14:54 +00:00
|
|
|
USE_X_FORWARDED_HOST = True
|
|
|
|
USE_X_FORWARDED_PORT = True
|
2018-04-22 13:11:04 +00:00
|
|
|
|
|
|
|
# Wether we should use Apache, Nginx (or other) headers when serving audio files
|
|
|
|
# Default to Nginx
|
2018-06-09 13:36:16 +00:00
|
|
|
REVERSE_PROXY_TYPE = env("REVERSE_PROXY_TYPE", default="nginx")
|
|
|
|
assert REVERSE_PROXY_TYPE in ["apache2", "nginx"], "Unsupported REVERSE_PROXY_TYPE"
|
2018-04-22 13:11:04 +00:00
|
|
|
|
2017-06-28 21:30:26 +00:00
|
|
|
# Which path will be used to process the internal redirection
|
|
|
|
# **DO NOT** put a slash at the end
|
2018-06-09 13:36:16 +00:00
|
|
|
PROTECT_FILES_PATH = env("PROTECT_FILES_PATH", default="/_protected")
|
2017-06-29 21:26:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
# use this setting to tweak for how long you want to cache
|
|
|
|
# musicbrainz results. (value is in seconds)
|
2018-06-09 13:36:16 +00:00
|
|
|
MUSICBRAINZ_CACHE_DURATION = env.int("MUSICBRAINZ_CACHE_DURATION", default=300)
|
2017-06-29 21:26:57 +00:00
|
|
|
|
2018-08-31 19:38:54 +00:00
|
|
|
# Use this setting to change the musicbrainz hostname, for instance to
|
|
|
|
# use a mirror. The hostname can also contain a port number (so, e.g.,
|
|
|
|
# "localhost:5000" is a valid name to set).
|
|
|
|
MUSICBRAINZ_HOSTNAME = env("MUSICBRAINZ_HOSTNAME", default="musicbrainz.org")
|
|
|
|
|
2017-07-17 20:00:32 +00:00
|
|
|
# Custom Admin URL, use {% url 'admin:index' %}
|
2018-06-09 13:36:16 +00:00
|
|
|
ADMIN_URL = env("DJANGO_ADMIN_URL", default="^api/admin/")
|
2017-12-14 22:06:21 +00:00
|
|
|
CSRF_USE_SESSIONS = True
|
2018-03-19 11:36:15 +00:00
|
|
|
|
|
|
|
# Playlist settings
|
2018-04-28 04:11:50 +00:00
|
|
|
# XXX: deprecated, see #186
|
2018-06-09 13:36:16 +00:00
|
|
|
PLAYLISTS_MAX_TRACKS = env.int("PLAYLISTS_MAX_TRACKS", default=250)
|
2018-03-24 19:31:36 +00:00
|
|
|
|
|
|
|
ACCOUNT_USERNAME_BLACKLIST = [
|
2018-06-09 13:36:16 +00:00
|
|
|
"funkwhale",
|
|
|
|
"library",
|
2018-07-22 11:05:43 +00:00
|
|
|
"instance",
|
2018-06-09 13:36:16 +00:00
|
|
|
"test",
|
|
|
|
"status",
|
|
|
|
"root",
|
|
|
|
"admin",
|
|
|
|
"owner",
|
|
|
|
"superuser",
|
|
|
|
"staff",
|
|
|
|
"service",
|
2018-07-13 12:10:39 +00:00
|
|
|
"me",
|
2018-07-19 10:49:01 +00:00
|
|
|
"ghost",
|
|
|
|
"_",
|
2018-09-22 12:29:30 +00:00
|
|
|
"-",
|
2018-07-19 10:49:01 +00:00
|
|
|
"hello",
|
|
|
|
"contact",
|
2018-09-22 12:29:30 +00:00
|
|
|
"inbox",
|
|
|
|
"outbox",
|
|
|
|
"shared-inbox",
|
|
|
|
"shared_inbox",
|
|
|
|
"actor",
|
2018-06-09 13:36:16 +00:00
|
|
|
] + env.list("ACCOUNT_USERNAME_BLACKLIST", default=[])
|
|
|
|
|
|
|
|
EXTERNAL_REQUESTS_VERIFY_SSL = env.bool("EXTERNAL_REQUESTS_VERIFY_SSL", default=True)
|
2018-04-28 04:11:50 +00:00
|
|
|
# XXX: deprecated, see #186
|
|
|
|
API_AUTHENTICATION_REQUIRED = env.bool("API_AUTHENTICATION_REQUIRED", True)
|
2018-04-21 16:15:29 +00:00
|
|
|
|
2018-06-09 13:36:16 +00:00
|
|
|
MUSIC_DIRECTORY_PATH = env("MUSIC_DIRECTORY_PATH", default=None)
|
2018-04-21 16:15:29 +00:00
|
|
|
# on Docker setup, the music directory may not match the host path,
|
|
|
|
# and we need to know it for it to serve stuff properly
|
|
|
|
MUSIC_DIRECTORY_SERVE_PATH = env(
|
2018-06-09 13:36:16 +00:00
|
|
|
"MUSIC_DIRECTORY_SERVE_PATH", default=MUSIC_DIRECTORY_PATH
|
|
|
|
)
|
2018-06-19 19:47:43 +00:00
|
|
|
|
|
|
|
USERS_INVITATION_EXPIRATION_DAYS = env.int(
|
|
|
|
"USERS_INVITATION_EXPIRATION_DAYS", default=14
|
|
|
|
)
|
2018-07-18 13:37:07 +00:00
|
|
|
|
|
|
|
VERSATILEIMAGEFIELD_RENDITION_KEY_SETS = {
|
|
|
|
"square": [
|
|
|
|
("original", "url"),
|
|
|
|
("square_crop", "crop__400x400"),
|
|
|
|
("medium_square_crop", "crop__200x200"),
|
|
|
|
("small_square_crop", "crop__50x50"),
|
|
|
|
]
|
|
|
|
}
|
|
|
|
VERSATILEIMAGEFIELD_SETTINGS = {"create_images_on_demand": False}
|