mirror of
https://github.com/etesync/server
synced 2025-01-28 09:30:56 +00:00
fac36aae11
This commit adds code to allow users that prefer to do user management via LDAP to do so. Since Etebase does not store the password (proof) in a LDAP compatible fashion, we compromise and instead two checks: One while registering an account to see if the username is the LDAP directory and one whenever the API is accessed. To prevent too many LDAP requests, the result of the request is cached for an configurable amount of hours. Make sure you have python-ldap installed and can successfully import ldap. Then, if you use the easy config, add the following to your config: ``` ; [...] ; Regular etesync configuration [ldap] bind_dn = <Your LDAP "user" to bind as. See Note 1> bind_pw = <The password to authenticate as your bind user> ; Or if you have the password in a file: ; bind_pw_file = /path/to/the/file.txt server = <The URL to your LDAP server> search_base = <Your search base> filter = <Your LDAP filter query. See Note 2> ; In case a cache TTL of 1 hour is too short for you, set `cache_ttl` to the preferred ; amount of hours a cache entry should be viewed as valid: ; cache_ttl = 5 ``` With this config, I am able to make the EteSync server check with my LDAP server if a user should be able to login or register. Note that if a user is allowed to login or register, the password of the LDAP user will be ignored. This LDAP patch is nothing more than an additional check before the actual authentication. A successful LDAP check will be cached, if not configured (correctly), for one hour, after which the LDAP query will be performed again. Note 1: This commit only works with a bind user Note 2: The query must be specified. If an LDAP query returns more than one or no result, then the authentication fails. If your query needs to include the username that currently tries to perform a login or registration, you can use %%s, which will be subsituted for the used username.
204 lines
6.2 KiB
Python
204 lines
6.2 KiB
Python
"""
|
|
Django settings for etebase_server project.
|
|
|
|
Generated by 'django-admin startproject' using Django 3.0.3.
|
|
|
|
For more information on this file, see
|
|
https://docs.djangoproject.com/en/3.0/topics/settings/
|
|
|
|
For the full list of settings and their values, see
|
|
https://docs.djangoproject.com/en/3.0/ref/settings/
|
|
"""
|
|
|
|
import os
|
|
import configparser
|
|
from .utils import get_secret_from_file
|
|
|
|
# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
|
|
SOURCE_DIR = os.path.dirname(os.path.abspath(__file__))
|
|
BASE_DIR = os.path.dirname(SOURCE_DIR)
|
|
|
|
AUTH_USER_MODEL = "myauth.User"
|
|
|
|
|
|
# Quick-start development settings - unsuitable for production
|
|
# See https://docs.djangoproject.com/en/3.0/howto/deployment/checklist/
|
|
|
|
# SECURITY WARNING: keep the secret key used in production secret!
|
|
# See secret.py for how this is generated; uses a file 'secret.txt' in the root
|
|
# directory
|
|
SECRET_FILE = os.path.join(BASE_DIR, "secret.txt")
|
|
|
|
# SECURITY WARNING: don't run with debug turned on in production!
|
|
DEBUG = True
|
|
|
|
ALLOWED_HOSTS = []
|
|
|
|
# Database
|
|
# https://docs.djangoproject.com/en/2.0/ref/settings/#databases
|
|
|
|
DATABASES = {
|
|
"default": {
|
|
"ENGINE": "django.db.backends.sqlite3",
|
|
"NAME": os.environ.get("ETEBASE_DB_PATH", os.path.join(BASE_DIR, "db.sqlite3")),
|
|
}
|
|
}
|
|
|
|
DEFAULT_AUTO_FIELD = 'django.db.models.AutoField'
|
|
|
|
# Application definition
|
|
|
|
INSTALLED_APPS = [
|
|
"django.contrib.admin",
|
|
"django.contrib.auth",
|
|
"django.contrib.contenttypes",
|
|
"django.contrib.sessions",
|
|
"django.contrib.messages",
|
|
"django.contrib.staticfiles",
|
|
"etebase_server.myauth.apps.MyauthConfig",
|
|
"etebase_server.django.apps.DjangoEtebaseConfig",
|
|
"etebase_server.django.token_auth.apps.TokenAuthConfig",
|
|
]
|
|
|
|
MIDDLEWARE = [
|
|
"django.middleware.security.SecurityMiddleware",
|
|
"django.contrib.sessions.middleware.SessionMiddleware",
|
|
"django.middleware.common.CommonMiddleware",
|
|
"django.middleware.csrf.CsrfViewMiddleware",
|
|
"django.contrib.auth.middleware.AuthenticationMiddleware",
|
|
"django.contrib.messages.middleware.MessageMiddleware",
|
|
"django.middleware.clickjacking.XFrameOptionsMiddleware",
|
|
]
|
|
|
|
ROOT_URLCONF = "etebase_server.urls"
|
|
|
|
TEMPLATES = [
|
|
{
|
|
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
|
"DIRS": [os.path.join(SOURCE_DIR, "templates")],
|
|
"APP_DIRS": True,
|
|
"OPTIONS": {
|
|
"context_processors": [
|
|
"django.template.context_processors.debug",
|
|
"django.template.context_processors.request",
|
|
"django.contrib.auth.context_processors.auth",
|
|
"django.contrib.messages.context_processors.messages",
|
|
],
|
|
},
|
|
},
|
|
]
|
|
|
|
WSGI_APPLICATION = "etebase_server.wsgi.application"
|
|
|
|
|
|
# Password validation
|
|
# https://docs.djangoproject.com/en/3.0/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",
|
|
},
|
|
]
|
|
|
|
|
|
# Internationalization
|
|
# https://docs.djangoproject.com/en/3.0/topics/i18n/
|
|
|
|
LANGUAGE_CODE = "en-us"
|
|
|
|
TIME_ZONE = "UTC"
|
|
|
|
USE_I18N = True
|
|
|
|
USE_L10N = True
|
|
|
|
USE_TZ = True
|
|
|
|
# Static files (CSS, JavaScript, Images)
|
|
# https://docs.djangoproject.com/en/3.0/howto/static-files/
|
|
|
|
STATIC_URL = "/static/"
|
|
STATIC_ROOT = os.environ.get("DJANGO_STATIC_ROOT", os.path.join(BASE_DIR, "static"))
|
|
|
|
MEDIA_ROOT = os.environ.get("DJANGO_MEDIA_ROOT", os.path.join(BASE_DIR, "media"))
|
|
MEDIA_URL = "/user-media/"
|
|
|
|
|
|
# Define where to find configuration files
|
|
config_locations = [
|
|
os.environ.get("ETEBASE_EASY_CONFIG_PATH", ""),
|
|
"etebase-server.ini",
|
|
"/etc/etebase-server/etebase-server.ini",
|
|
]
|
|
|
|
ETEBASE_CREATE_USER_FUNC = "etebase_server.django.utils.create_user_blocked"
|
|
|
|
# Use config file if present
|
|
if any(os.path.isfile(x) for x in config_locations):
|
|
config = configparser.ConfigParser()
|
|
config.read(config_locations)
|
|
|
|
section = config["global"]
|
|
|
|
SECRET_FILE = section.get("secret_file", SECRET_FILE)
|
|
STATIC_ROOT = section.get("static_root", STATIC_ROOT)
|
|
STATIC_URL = section.get("static_url", STATIC_URL)
|
|
MEDIA_ROOT = section.get("media_root", MEDIA_ROOT)
|
|
MEDIA_URL = section.get("media_url", MEDIA_URL)
|
|
LANGUAGE_CODE = section.get("language_code", LANGUAGE_CODE)
|
|
TIME_ZONE = section.get("time_zone", TIME_ZONE)
|
|
DEBUG = section.getboolean("debug", DEBUG)
|
|
|
|
if "redis_uri" in section:
|
|
ETEBASE_REDIS_URI = section.get("redis_uri")
|
|
|
|
if "allowed_hosts" in config:
|
|
ALLOWED_HOSTS = [y for x, y in config.items("allowed_hosts")]
|
|
|
|
if "database" in config:
|
|
DATABASES = {"default": {x.upper(): y for x, y in config.items("database")}}
|
|
|
|
if "database-options" in config:
|
|
DATABASES["default"]["OPTIONS"] = config["database-options"]
|
|
|
|
if "ldap" in config:
|
|
ldap = config["ldap"]
|
|
LDAP_SERVER = ldap.get("server", "")
|
|
LDAP_SEARCH_BASE = ldap.get("search_base", "")
|
|
LDAP_FILTER = ldap.get("filter", "")
|
|
LDAP_BIND_DN = ldap.get("bind_dn", "")
|
|
LDAP_BIND_PW = ldap.get("bind_pw", "")
|
|
LDAP_BIND_PW_FILE = ldap.get("bind_pw_file", "")
|
|
LDAP_CACHE_TTL = ldap.get("cache_ttl", "")
|
|
|
|
if not LDAP_BIND_DN:
|
|
raise Exception("LDAP enabled but bind_dn is not set!")
|
|
if not LDAP_BIND_PW and not LDAP_BIND_PW_FILE:
|
|
raise Exception("LDAP enabled but both bind_pw and bind_pw_file are not set!")
|
|
|
|
# Configure EteBase to use LDAP
|
|
ETEBASE_CREATE_USER_FUNC = "etebase_server.myauth.ldap.create_user"
|
|
ETEBASE_API_PERMISSIONS_READ = ["etebase_server.myauth.ldap.is_user_in_ldap"]
|
|
|
|
# Efficient file streaming (for large files)
|
|
SENDFILE_BACKEND = "etebase_server.fastapi.sendfile.backends.simple"
|
|
SENDFILE_ROOT = MEDIA_ROOT
|
|
|
|
# Make an `etebase_server_settings` module available to override settings.
|
|
try:
|
|
from etebase_server_settings import *
|
|
except ImportError:
|
|
pass
|
|
|
|
if "SECRET_KEY" not in locals():
|
|
SECRET_KEY = get_secret_from_file(SECRET_FILE)
|