settings.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. import os
  2. import sys
  3. from pathlib import Path
  4. import dj_database_url
  5. from django.utils.translation import gettext_lazy as _
  6. from dotenv import load_dotenv
  7. PROJECT_ROOT = Path(__file__).resolve().parent
  8. BASE_DIR = Path(__file__).resolve().parent.parent
  9. sys.path.insert(0, os.path.join(PROJECT_ROOT, 'apps'))
  10. # Tap vrobbler.conf if it's available
  11. if os.path.exists("vrobbler.conf"):
  12. load_dotenv("vrobbler.conf")
  13. elif os.path.exists("/etc/vrobbler.conf"):
  14. load_dotenv("/etc/vrobbler.conf")
  15. elif os.path.exists("/usr/local/etc/vrobbler.conf"):
  16. load_dotenv("/usr/local/etc/vrobbler.conf")
  17. # Build paths inside the project like this: BASE_DIR / 'subdir'.
  18. # Quick-start development settings - unsuitable for production
  19. # See https://docs.djangoproject.com/en/3.1/howto/deployment/checklist/
  20. # SECURITY WARNING: keep the secret key used in production secret!
  21. SECRET_KEY = os.getenv("VROBBLER_SECRET_KEY", "not-a-secret-234lkjasdflj132")
  22. # SECURITY WARNING: don't run with debug turned on in production!
  23. DEBUG = os.getenv("VROBBLER_DEBUG", False)
  24. TESTING = len(sys.argv) > 1 and sys.argv[1] == "test"
  25. KEEP_DETAILED_SCROBBLE_LOGS = os.getenv(
  26. "VROBBLER_KEEP_DETAILED_SCROBBLE_LOGS", False
  27. )
  28. # Key must be 16, 24 or 32 bytes long and will be converted to a byte stream
  29. ENCRYPTED_FIELD_KEY = os.getenv(
  30. "VROBBLER_ENCRYPTED_FIELD_KEY", "12345678901234567890123456789012"
  31. )
  32. DJANGO_ENCRYPTED_FIELD_KEY = bytes(ENCRYPTED_FIELD_KEY, "utf-8")
  33. # Should we cull old in-progress scrobbles that are beyond the wait period for resuming?
  34. DELETE_STALE_SCROBBLES = os.getenv("VROBBLER_DELETE_STALE_SCROBBLES", True)
  35. # Used to dump data coming from srobbling sources, helpful for building new inputs
  36. DUMP_REQUEST_DATA = os.getenv("VROBBLER_DUMP_REQUEST_DATA", False)
  37. THESPORTSDB_API_KEY = os.getenv("VROBBLER_THESPORTSDB_API_KEY", "2")
  38. THESPORTSDB_BASE_URL = os.getenv(
  39. "VROBBLER_THESPORTSDB_BASE_URL", "https://www.thesportsdb.com/api/v1/json/"
  40. )
  41. TMDB_API_KEY = os.getenv("VROBBLER_TMDB_API_KEY", "")
  42. LASTFM_API_KEY = os.getenv("VROBBLER_LASTFM_API_KEY")
  43. LASTFM_SECRET_KEY = os.getenv("VROBBLER_LASTFM_SECRET_KEY")
  44. DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
  45. TIME_ZONE = os.getenv("VROBBLER_TIME_ZONE", "US/Eastern")
  46. ALLOWED_HOSTS = ["*"]
  47. CSRF_TRUSTED_ORIGINS = [
  48. os.getenv("VROBBLER_TRUSTED_ORIGINS", "http://localhost:8000")
  49. ]
  50. X_FRAME_OPTIONS = "SAMEORIGIN"
  51. REDIS_URL = os.getenv("VROBBLER_REDIS_URL", None)
  52. if REDIS_URL:
  53. print(f"Sending tasks to redis@{REDIS_URL.split('@')[-1]}")
  54. else:
  55. print("Eagerly running all tasks")
  56. CELERY_TASK_ALWAYS_EAGER = os.getenv("VROBBLER_SKIP_CELERY", False)
  57. CELERY_BROKER_URL = REDIS_URL if REDIS_URL else "memory://localhost/"
  58. CELERY_RESULT_BACKEND = "django-db"
  59. CELERY_TIMEZONE = os.getenv("VROBBLER_TIME_ZONE", "US/Eastern")
  60. CELERY_TASK_TRACK_STARTED = True
  61. INSTALLED_APPS = [
  62. "django.contrib.admin",
  63. "django.contrib.auth",
  64. "django.contrib.contenttypes",
  65. "django.contrib.sessions",
  66. "django.contrib.messages",
  67. "django.contrib.staticfiles",
  68. "django.contrib.sites",
  69. "django.contrib.humanize",
  70. "django_filters",
  71. "django_extensions",
  72. "rest_framework.authtoken",
  73. "encrypted_field",
  74. "profiles",
  75. "scrobbles",
  76. "videos",
  77. "music",
  78. "podcasts",
  79. "sports",
  80. "books",
  81. "mathfilters",
  82. "rest_framework",
  83. "allauth",
  84. "allauth.account",
  85. "allauth.socialaccount",
  86. "django_celery_results",
  87. ]
  88. SITE_ID = 1
  89. MIDDLEWARE = [
  90. "django.middleware.security.SecurityMiddleware",
  91. "whitenoise.middleware.WhiteNoiseMiddleware",
  92. "django.contrib.sessions.middleware.SessionMiddleware",
  93. "django.middleware.common.CommonMiddleware",
  94. "django.middleware.csrf.CsrfViewMiddleware",
  95. "django.contrib.auth.middleware.AuthenticationMiddleware",
  96. "django.contrib.messages.middleware.MessageMiddleware",
  97. "django.middleware.clickjacking.XFrameOptionsMiddleware",
  98. "django.middleware.gzip.GZipMiddleware",
  99. ]
  100. ROOT_URLCONF = "vrobbler.urls"
  101. TEMPLATES = [
  102. {
  103. "BACKEND": "django.template.backends.django.DjangoTemplates",
  104. "DIRS": [str(PROJECT_ROOT.joinpath("templates"))],
  105. "APP_DIRS": True,
  106. "OPTIONS": {
  107. "context_processors": [
  108. "django.template.context_processors.debug",
  109. "django.template.context_processors.request",
  110. "django.contrib.auth.context_processors.auth",
  111. "django.contrib.messages.context_processors.messages",
  112. "videos.context_processors.video_lists",
  113. "music.context_processors.music_lists",
  114. "scrobbles.context_processors.now_playing",
  115. ],
  116. },
  117. },
  118. ]
  119. MESSAGE_STORAGE = "django.contrib.messages.storage.session.SessionStorage"
  120. WSGI_APPLICATION = "vrobbler.wsgi.application"
  121. DATABASES = {
  122. "default": dj_database_url.config(
  123. default=os.getenv("VROBBLER_DATABASE_URL", "sqlite:///db.sqlite3"),
  124. conn_max_age=600,
  125. ),
  126. }
  127. if TESTING:
  128. DATABASES = {
  129. "default": dj_database_url.config(default="sqlite:///testdb.sqlite3")
  130. }
  131. db_str = ""
  132. if 'sqlite' in DATABASES['default']['ENGINE']:
  133. db_str = f"Connected to sqlite@{DATABASES['default']['NAME']}"
  134. if 'postgresql' in DATABASES['default']['ENGINE']:
  135. db_str = f"Connected to postgres@{DATABASES['default']['HOST']}/{DATABASES['default']['NAME']}"
  136. if db_str:
  137. print(db_str)
  138. CACHES = {
  139. "default": {
  140. "BACKEND": "django.core.cache.backends.locmem.LocMemCache",
  141. "LOCATION": "unique-snowflake",
  142. }
  143. }
  144. if REDIS_URL:
  145. CACHES["default"]["BACKEND"] = "django_redis.cache.RedisCache"
  146. CACHES["default"]["LOCATION"] = REDIS_URL
  147. SESSION_ENGINE = "django.contrib.sessions.backends.cached_db"
  148. AUTHENTICATION_BACKENDS = [
  149. "django.contrib.auth.backends.ModelBackend",
  150. "allauth.account.auth_backends.AuthenticationBackend",
  151. ]
  152. # We have to ignore content negotiation because Jellyfin is a bad actor
  153. REST_FRAMEWORK = {
  154. "DEFAULT_PERMISSION_CLASSES": ("rest_framework.permissions.AllowAny",),
  155. 'DEFAULT_AUTHENTICATION_CLASSES': [
  156. 'rest_framework.authentication.TokenAuthentication',
  157. 'rest_framework.authentication.SessionAuthentication',
  158. ],
  159. 'DEFAULT_CONTENT_NEGOTIATION_CLASS': 'vrobbler.negotiation.IgnoreClientContentNegotiation',
  160. "DEFAULT_FILTER_BACKENDS": [
  161. "django_filters.rest_framework.DjangoFilterBackend"
  162. ],
  163. "DEFAULT_PAGINATION_CLASS": "rest_framework.pagination.PageNumberPagination",
  164. "PAGE_SIZE": 200,
  165. }
  166. LOGIN_REDIRECT_URL = "/"
  167. AUTH_PASSWORD_VALIDATORS = [
  168. {
  169. "NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
  170. },
  171. {
  172. "NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",
  173. },
  174. {
  175. "NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",
  176. },
  177. {
  178. "NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",
  179. },
  180. ]
  181. # Internationalization
  182. # https://docs.djangoproject.com/en/3.1/topics/i18n/
  183. LANGUAGE_CODE = "en-us"
  184. TIME_ZONE = os.getenv("VROBBLER_TIME_ZONE", "EST")
  185. USE_I18N = True
  186. USE_TZ = True
  187. # Static files (CSS, JavaScript, Images)
  188. # https://docs.djangoproject.com/en/3.1/howto/static-files/
  189. STATIC_URL = "static/"
  190. STATIC_ROOT = os.getenv(
  191. "VROBBLER_STATIC_ROOT", os.path.join(PROJECT_ROOT, "static")
  192. )
  193. MEDIA_URL = "/media/"
  194. MEDIA_ROOT = os.getenv(
  195. "VROBBLER_MEDIA_ROOT", os.path.join(PROJECT_ROOT, "media")
  196. )
  197. JSON_LOGGING = os.getenv("VROBBLER_JSON_LOGGING", False)
  198. LOG_TYPE = "json" if JSON_LOGGING else "log"
  199. default_level = "INFO"
  200. if DEBUG:
  201. default_level = "DEBUG"
  202. LOG_LEVEL = os.getenv("VROBBLER_LOG_LEVEL", default_level)
  203. LOG_FILE_PATH = os.getenv("VROBBLER_LOG_FILE_PATH", "/tmp/")
  204. LOGGING = {
  205. "version": 1,
  206. "disable_existing_loggers": False,
  207. "root": {
  208. "handlers": ["console", "file"],
  209. "level": LOG_LEVEL,
  210. "propagate": True,
  211. },
  212. "formatters": {
  213. "color": {
  214. "()": "colorlog.ColoredFormatter",
  215. # \r returns caret to line beginning, in tests this eats the silly dot that removes
  216. # the beautiful alignment produced below
  217. "format": "\r"
  218. "{log_color}{levelname:8s}{reset} "
  219. "{bold_cyan}{name}{reset}:"
  220. "{fg_bold_red}{lineno}{reset} "
  221. "{thin_yellow}{funcName} "
  222. "{thin_white}{message}"
  223. "{reset}",
  224. "style": "{",
  225. },
  226. "log": {"format": "%(asctime)s %(levelname)s %(message)s"},
  227. "json": {
  228. "()": "pythonjsonlogger.jsonlogger.JsonFormatter",
  229. "format": "%(levelname)s %(name) %(funcName) %(lineno) %(asctime)s %(message)s",
  230. },
  231. },
  232. "handlers": {
  233. "console": {
  234. "class": "logging.StreamHandler",
  235. "formatter": "color",
  236. "level": LOG_LEVEL,
  237. },
  238. "null": {
  239. "class": "logging.NullHandler",
  240. "level": LOG_LEVEL,
  241. },
  242. 'sql': {
  243. 'class': 'logging.handlers.RotatingFileHandler',
  244. 'filename': ''.join([LOG_FILE_PATH, 'vrobbler_sql.', LOG_TYPE]),
  245. 'formatter': LOG_TYPE,
  246. 'level': LOG_LEVEL,
  247. },
  248. 'file': {
  249. 'class': 'logging.handlers.RotatingFileHandler',
  250. 'filename': ''.join([LOG_FILE_PATH, 'vrobbler.', LOG_TYPE]),
  251. 'formatter': LOG_TYPE,
  252. 'level': LOG_LEVEL,
  253. },
  254. },
  255. "loggers": {
  256. # Quiet down our console a little
  257. "django": {
  258. "handlers": ["file"],
  259. "propagate": True,
  260. },
  261. "django.db.backends": {"handlers": ["null"]},
  262. "django.server": {"handlers": ["null"]},
  263. "pylast": {"handlers": ["null"], "propagate": False},
  264. "musicbrainzngs": {"handlers": ["null"], "propagate": False},
  265. "httpx": {"handlers": ["null"], "propagate": False},
  266. "vrobbler": {
  267. "handlers": ["console"],
  268. "propagate": False,
  269. },
  270. },
  271. }
  272. LOG_TO_CONSOLE = os.getenv("VROBBLER_LOG_TO_CONSOLE", False)
  273. if LOG_TO_CONSOLE:
  274. LOGGING['loggers']['django']['handlers'] = ["console"]
  275. LOGGING['loggers']['vrobbler']['handlers'] = ["console"]