settings.py 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  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. # Should we cull old in-progress scrobbles that are beyond the wait period for resuming?
  29. DELETE_STALE_SCROBBLES = os.getenv("VROBBLER_DELETE_STALE_SCROBBLES", True)
  30. # Used to dump data coming from srobbling sources, helpful for building new inputs
  31. DUMP_REQUEST_DATA = os.getenv("VROBBLER_DUMP_REQUEST_DATA", False)
  32. VIDEO_BACKOFF_MINUTES = os.getenv("VROBBLER_VIDEO_BACKOFF_MINUTES", 15)
  33. MUSIC_BACKOFF_SECONDS = os.getenv("VROBBLER_VIDEO_BACKOFF_SECONDS", 5)
  34. # If you stop waching or listening to a track, how long should we wait before we
  35. # give up on the old scrobble and start a new one? This could also be considered
  36. # a "continue in progress scrobble" time period. So if you pause the media and
  37. # start again, should it be a new scrobble.
  38. VIDEO_WAIT_PERIOD_DAYS = os.getenv("VROBBLER_VIDEO_WAIT_PERIOD_DAYS", 1)
  39. MUSIC_WAIT_PERIOD_MINUTES = os.getenv("VROBBLER_VIDEO_BACKOFF_MINUTES", 1)
  40. TMDB_API_KEY = os.getenv("VROBBLER_TMDB_API_KEY", "")
  41. DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
  42. ALLOWED_HOSTS = ["*"]
  43. CSRF_TRUSTED_ORIGINS = [
  44. os.getenv("VROBBLER_TRUSTED_ORIGINS", "http://localhost:8000")
  45. ]
  46. X_FRAME_OPTIONS = "SAMEORIGIN"
  47. REDIS_URL = os.getenv("VROBBLER_REDIS_URL", None)
  48. CELERY_TASK_ALWAYS_EAGER = os.getenv("VROBBLER_SKIP_CELERY", False)
  49. CELERY_BROKER_URL = REDIS_URL if REDIS_URL else "memory://localhost/"
  50. CELERY_RESULT_BACKEND = "django-db"
  51. CELERY_TIMEZONE = os.getenv("VROBBLER_TIME_ZONE", "EST")
  52. CELERY_TASK_TRACK_STARTED = True
  53. INSTALLED_APPS = [
  54. "django.contrib.admin",
  55. "django.contrib.auth",
  56. "django.contrib.contenttypes",
  57. "django.contrib.sessions",
  58. "django.contrib.messages",
  59. "django.contrib.staticfiles",
  60. "django.contrib.sites",
  61. "django.contrib.humanize",
  62. "django_filters",
  63. "django_extensions",
  64. 'rest_framework.authtoken',
  65. "scrobbles",
  66. "videos",
  67. "music",
  68. "rest_framework",
  69. "allauth",
  70. "allauth.account",
  71. "allauth.socialaccount",
  72. "django_celery_results",
  73. ]
  74. SITE_ID = 1
  75. MIDDLEWARE = [
  76. "django.middleware.security.SecurityMiddleware",
  77. "whitenoise.middleware.WhiteNoiseMiddleware",
  78. "django.contrib.sessions.middleware.SessionMiddleware",
  79. "django.middleware.common.CommonMiddleware",
  80. "django.middleware.csrf.CsrfViewMiddleware",
  81. "django.contrib.auth.middleware.AuthenticationMiddleware",
  82. "django.contrib.messages.middleware.MessageMiddleware",
  83. "django.middleware.clickjacking.XFrameOptionsMiddleware",
  84. "django.middleware.gzip.GZipMiddleware",
  85. ]
  86. ROOT_URLCONF = "vrobbler.urls"
  87. TEMPLATES = [
  88. {
  89. "BACKEND": "django.template.backends.django.DjangoTemplates",
  90. "DIRS": [str(PROJECT_ROOT.joinpath("templates"))],
  91. "APP_DIRS": True,
  92. "OPTIONS": {
  93. "context_processors": [
  94. "django.template.context_processors.debug",
  95. "django.template.context_processors.request",
  96. "django.contrib.auth.context_processors.auth",
  97. "django.contrib.messages.context_processors.messages",
  98. "videos.context_processors.video_lists",
  99. "music.context_processors.music_lists",
  100. ],
  101. },
  102. },
  103. ]
  104. WSGI_APPLICATION = "vrobbler.wsgi.application"
  105. DATABASES = {
  106. "default": dj_database_url.config(
  107. default=os.getenv("VROBBLER_DATABASE_URL", "sqlite:///db.sqlite3"),
  108. conn_max_age=600,
  109. ),
  110. }
  111. if TESTING:
  112. DATABASES = {
  113. "default": dj_database_url.config(default="sqlite:///testdb.sqlite3")
  114. }
  115. CACHES = {
  116. "default": {
  117. "BACKEND": "django.core.cache.backends.locmem.LocMemCache",
  118. "LOCATION": "unique-snowflake",
  119. }
  120. }
  121. if REDIS_URL:
  122. CACHES["default"][
  123. "BACKEND"
  124. ] = "django.core.cache.backends.redis.RedisCache"
  125. CACHES["default"]["LOCATION"] = REDIS_URL
  126. SESSION_ENGINE = "django.contrib.sessions.backends.cached_db"
  127. AUTHENTICATION_BACKENDS = [
  128. "django.contrib.auth.backends.ModelBackend",
  129. "allauth.account.auth_backends.AuthenticationBackend",
  130. ]
  131. REST_FRAMEWORK = {
  132. "DEFAULT_PERMISSION_CLASSES": ("rest_framework.permissions.AllowAny",),
  133. 'DEFAULT_AUTHENTICATION_CLASSES': [
  134. #'rest_framework.authentication.BasicAuthentication',
  135. #'rest_framework.authentication.TokenAuthentication',
  136. 'rest_framework.authentication.SessionAuthentication',
  137. ],
  138. "DEFAULT_PAGINATION_CLASS": "rest_framework.pagination.PageNumberPagination",
  139. "DEFAULT_FILTER_BACKENDS": [
  140. "django_filters.rest_framework.DjangoFilterBackend"
  141. ],
  142. 'DEFAULT_PARSER_CLASSES': [
  143. 'rest_framework.parsers.JSONParser',
  144. ],
  145. 'DEFAULT_CONTENT_NEGOTIATION_CLASS': 'vrobbler.negotiation.IgnoreClientContentNegotiation',
  146. "PAGE_SIZE": 100,
  147. }
  148. LOGIN_REDIRECT_URL = "/"
  149. AUTH_PASSWORD_VALIDATORS = [
  150. {
  151. "NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
  152. },
  153. {
  154. "NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",
  155. },
  156. {
  157. "NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",
  158. },
  159. {
  160. "NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",
  161. },
  162. ]
  163. # Internationalization
  164. # https://docs.djangoproject.com/en/3.1/topics/i18n/
  165. LANGUAGE_CODE = "en-us"
  166. TIME_ZONE = os.getenv("VROBBLER_TIME_ZONE", "EST")
  167. USE_I18N = True
  168. USE_L10N = True
  169. USE_TZ = True
  170. # Static files (CSS, JavaScript, Images)
  171. # https://docs.djangoproject.com/en/3.1/howto/static-files/
  172. STATIC_URL = "static/"
  173. STATIC_ROOT = os.getenv(
  174. "VROBBLER_STATIC_ROOT", os.path.join(PROJECT_ROOT, "static")
  175. )
  176. if not DEBUG:
  177. STATICFILES_STORAGE = (
  178. "whitenoise.storage.CompressedManifestStaticFilesStorage"
  179. )
  180. MEDIA_URL = "/media/"
  181. MEDIA_ROOT = os.getenv(
  182. "VROBBLER_MEDIA_ROOT", os.path.join(PROJECT_ROOT, "media")
  183. )
  184. JSON_LOGGING = os.getenv("VROBBLER_JSON_LOGGING", False)
  185. LOG_TYPE = "json" if JSON_LOGGING else "log"
  186. default_level = "INFO"
  187. if DEBUG:
  188. default_level = "DEBUG"
  189. LOG_LEVEL = os.getenv("VROBBLER_LOG_LEVEL", default_level)
  190. LOG_FILE_PATH = os.getenv("VROBBLER_LOG_FILE_PATH", "/tmp/")
  191. LOGGING = {
  192. "version": 1,
  193. "disable_existing_loggers": False,
  194. "root": {
  195. "handlers": ["console", "file"],
  196. "level": LOG_LEVEL,
  197. "propagate": True,
  198. },
  199. "formatters": {
  200. "color": {
  201. "()": "colorlog.ColoredFormatter",
  202. # \r returns caret to line beginning, in tests this eats the silly dot that removes
  203. # the beautiful alignment produced below
  204. "format": "\r"
  205. "{log_color}{levelname:8s}{reset} "
  206. "{bold_cyan}{name}{reset}:"
  207. "{fg_bold_red}{lineno}{reset} "
  208. "{thin_yellow}{funcName} "
  209. "{thin_white}{message}"
  210. "{reset}",
  211. "style": "{",
  212. },
  213. "log": {"format": "%(asctime)s %(levelname)s %(message)s"},
  214. "json": {
  215. "()": "pythonjsonlogger.jsonlogger.JsonFormatter",
  216. "format": "%(levelname)s %(name) %(funcName) %(lineno) %(asctime)s %(message)s",
  217. },
  218. },
  219. "handlers": {
  220. "console": {
  221. "class": "logging.StreamHandler",
  222. "formatter": "color",
  223. "level": LOG_LEVEL,
  224. },
  225. "null": {
  226. "class": "logging.NullHandler",
  227. "level": LOG_LEVEL,
  228. },
  229. "file": {
  230. "class": "logging.handlers.RotatingFileHandler",
  231. "filename": "".join([LOG_FILE_PATH, "vrobbler.log"]),
  232. "formatter": LOG_TYPE,
  233. "level": LOG_LEVEL,
  234. },
  235. "requests_file": {
  236. "class": "logging.handlers.RotatingFileHandler",
  237. "filename": "".join([LOG_FILE_PATH, "vrobbler_requests.log"]),
  238. "formatter": LOG_TYPE,
  239. "level": LOG_LEVEL,
  240. },
  241. },
  242. "loggers": {
  243. # Quiet down our console a little
  244. "django": {
  245. "handlers": ["file"],
  246. "propagate": True,
  247. },
  248. "django.db.backends": {"handlers": ["null"]},
  249. "vrobbler": {
  250. "handlers": ["console", "file"],
  251. "propagate": True,
  252. },
  253. },
  254. }
  255. if DEBUG:
  256. # We clear out a db with lots of games all the time in dev
  257. DATA_UPLOAD_MAX_NUMBER_FIELDS = 3000