settings.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. import os
  2. import sys
  3. from pathlib import Path
  4. import dj_database_url
  5. from dotenv import load_dotenv
  6. TRUTHY = ("true", "1", "t")
  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").lower() in TRUTHY
  24. TESTING = len(sys.argv) > 1 and sys.argv[1] == "test"
  25. TAGGIT_CASE_INSENSITIVE = True
  26. KEEP_DETAILED_SCROBBLE_LOGS = os.getenv(
  27. "VROBBLER_KEEP_DETAILED_SCROBBLE_LOGS", False
  28. )
  29. # Key must be 16, 24 or 32 bytes long and will be converted to a byte stream
  30. ENCRYPTED_FIELD_KEY = os.getenv(
  31. "VROBBLER_ENCRYPTED_FIELD_KEY", "12345678901234567890123456789012"
  32. )
  33. DJANGO_ENCRYPTED_FIELD_KEY = bytes(ENCRYPTED_FIELD_KEY, "utf-8")
  34. # Should we cull old in-progress scrobbles that are beyond the wait period for resuming?
  35. DELETE_STALE_SCROBBLES = (
  36. os.getenv("VROBBLER_DELETE_STALE_SCROBBLES", "true").lower() in TRUTHY
  37. )
  38. # Used to dump data coming from srobbling sources, helpful for building new inputs
  39. DUMP_REQUEST_DATA = (
  40. os.getenv("VROBBLER_DUMP_REQUEST_DATA", "false").lower() in TRUTHY
  41. )
  42. THESPORTSDB_API_KEY = os.getenv("VROBBLER_THESPORTSDB_API_KEY", "2")
  43. THEAUDIODB_API_KEY = os.getenv("VROBBLER_THEAUDIODB_API_KEY", "2")
  44. PODCASTINDEX_API_KEY = os.getenv("VROBBLER_PODCASTINDEX_API_KEY", "")
  45. PODCASTINDEX_API_SECRET = os.getenv("VROBBLER_PODCASTINDEX_API_SECRET", "")
  46. TMDB_API_KEY = os.getenv("VROBBLER_TMDB_API_KEY", "")
  47. LASTFM_API_KEY = os.getenv("VROBBLER_LASTFM_API_KEY")
  48. LASTFM_SECRET_KEY = os.getenv("VROBBLER_LASTFM_SECRET_KEY")
  49. IGDB_CLIENT_ID = os.getenv("VROBBLER_IGDB_CLIENT_ID")
  50. IGDB_CLIENT_SECRET = os.getenv("VROBBLER_IGDB_CLIENT_SECRET")
  51. COMICVINE_API_KEY = os.getenv("VROBBLER_COMICVINE_API_KEY")
  52. GEOLOC_ACCURACY = os.getenv("VROBBLER_GEOLOC_ACCURACY", 3)
  53. GEOLOC_PROXIMITY = os.getenv("VROBBLER_GEOLOC_PROXIMITY", "0.0001")
  54. POINTS_FOR_MOVEMENT_HISTORY = os.getenv(
  55. "VROBBLER_POINTS_FOR_MOVEMENT_HISTORY", 3
  56. )
  57. TODOIST_CLIENT_ID = os.getenv("VROBBLER_TODOIST_CLIENT_ID", "")
  58. TODOIST_CLIENT_SECRET = os.getenv("VROBBLER_TODOIST_CLIENT_SECRET", "")
  59. GOOGLE_API_KEY = os.getenv("VROBBLER_GOOGLE_API_KEY", "")
  60. LICHESS_API_KEY = os.getenv("VROBBLER_LICHESS_API_KEY", "")
  61. DEFAULT_TASK_CONTEXT_TAGS = [
  62. "Dev",
  63. "Home",
  64. "Errand",
  65. "Blog",
  66. "Mail",
  67. "Call",
  68. "Meeting",
  69. "Exercise",
  70. ]
  71. DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField"
  72. TIME_ZONE = os.getenv("VROBBLER_TIME_ZONE", "America/New_York")
  73. ALLOWED_HOSTS = ["*"]
  74. CSRF_TRUSTED_ORIGINS = [
  75. os.getenv("VROBBLER_TRUSTED_ORIGINS", "http://localhost:8000")
  76. ]
  77. X_FRAME_OPTIONS = "SAMEORIGIN"
  78. REDIS_URL = os.getenv("VROBBLER_REDIS_URL", None)
  79. if REDIS_URL:
  80. print(f"Sending tasks to redis@{REDIS_URL.split('@')[-1]}")
  81. else:
  82. print("Eagerly running all tasks")
  83. CELERY_TASK_ALWAYS_EAGER = (
  84. os.getenv("VROBBLER_SKIP_CELERY", "false").lower() in TRUTHY
  85. )
  86. CELERY_BROKER_URL = REDIS_URL if REDIS_URL else "memory://localhost/"
  87. CELERY_RESULT_BACKEND = "django-db"
  88. CELERY_TIMEZONE = os.getenv("VROBBLER_TIME_ZONE", "America/New_York")
  89. CELERY_TASK_TRACK_STARTED = True
  90. INSTALLED_APPS = [
  91. "django.contrib.admin",
  92. "django.contrib.auth",
  93. "django.contrib.contenttypes",
  94. "django.contrib.sessions",
  95. "django.contrib.messages",
  96. "django.contrib.staticfiles",
  97. "django.contrib.sites",
  98. "django.contrib.humanize",
  99. "django_filters",
  100. "django_extensions",
  101. "markdownify",
  102. "imagekit",
  103. "storages",
  104. "taggit",
  105. "rest_framework.authtoken",
  106. "oauth2_provider",
  107. "encrypted_field",
  108. "profiles",
  109. "scrobbles",
  110. "people",
  111. "videos",
  112. "music",
  113. "podcasts",
  114. "sports",
  115. "books",
  116. "boardgames",
  117. "bricksets",
  118. "videogames",
  119. "locations",
  120. "webpages",
  121. "tasks",
  122. "trails",
  123. "beers",
  124. "puzzles",
  125. "foods",
  126. "lifeevents",
  127. "moods",
  128. "mathfilters",
  129. "rest_framework",
  130. "allauth",
  131. "allauth.account",
  132. "allauth.socialaccount",
  133. "django_celery_results",
  134. ]
  135. SITE_ID = 1
  136. MIDDLEWARE = [
  137. "vrobbler.health_check.HealthCheckMiddleware",
  138. "vrobbler.apps.scrobbles.middleware.TimezoneMiddleware",
  139. "django.middleware.security.SecurityMiddleware",
  140. "django.contrib.sessions.middleware.SessionMiddleware",
  141. "django.middleware.common.CommonMiddleware",
  142. "django.middleware.csrf.CsrfViewMiddleware",
  143. "django.contrib.auth.middleware.AuthenticationMiddleware",
  144. "django.contrib.messages.middleware.MessageMiddleware",
  145. "django.middleware.clickjacking.XFrameOptionsMiddleware",
  146. "django.middleware.gzip.GZipMiddleware",
  147. ]
  148. ROOT_URLCONF = "vrobbler.urls"
  149. TEMPLATES = [
  150. {
  151. "BACKEND": "django.template.backends.django.DjangoTemplates",
  152. "DIRS": [str(PROJECT_ROOT.joinpath("templates"))],
  153. "APP_DIRS": True,
  154. "OPTIONS": {
  155. "context_processors": [
  156. "django.template.context_processors.debug",
  157. "django.template.context_processors.request",
  158. "django.contrib.auth.context_processors.auth",
  159. "django.contrib.messages.context_processors.messages",
  160. "videos.context_processors.video_lists",
  161. "music.context_processors.music_lists",
  162. "scrobbles.context_processors.now_playing",
  163. ],
  164. },
  165. },
  166. ]
  167. MESSAGE_STORAGE = "django.contrib.messages.storage.session.SessionStorage"
  168. WSGI_APPLICATION = "vrobbler.wsgi.application"
  169. DATABASES = {
  170. "default": dj_database_url.config(
  171. default=os.getenv("VROBBLER_DATABASE_URL", "sqlite:///db.sqlite3"),
  172. conn_max_age=600,
  173. ),
  174. }
  175. if TESTING:
  176. DATABASES = {
  177. "default": dj_database_url.config(default="sqlite:///testdb.sqlite3")
  178. }
  179. db_str = ""
  180. if "sqlite" in DATABASES["default"]["ENGINE"]:
  181. db_str = f"Connected to sqlite@{DATABASES['default']['NAME']}"
  182. if "postgresql" in DATABASES["default"]["ENGINE"]:
  183. db_str = f"Connected to postgres@{DATABASES['default']['HOST']}/{DATABASES['default']['NAME']}"
  184. if db_str:
  185. print(db_str)
  186. CACHES = {
  187. "default": {
  188. "BACKEND": "django.core.cache.backends.locmem.LocMemCache",
  189. "LOCATION": "unique-snowflake",
  190. }
  191. }
  192. if REDIS_URL:
  193. CACHES["default"]["BACKEND"] = "django_redis.cache.RedisCache"
  194. CACHES["default"]["LOCATION"] = REDIS_URL
  195. SESSION_ENGINE = "django.contrib.sessions.backends.cached_db"
  196. AUTHENTICATION_BACKENDS = [
  197. "django.contrib.auth.backends.ModelBackend",
  198. "allauth.account.auth_backends.AuthenticationBackend",
  199. ]
  200. # We have to ignore content negotiation because Jellyfin is a bad actor
  201. REST_FRAMEWORK = {
  202. "DEFAULT_PERMISSION_CLASSES": ("rest_framework.permissions.AllowAny",),
  203. "DEFAULT_AUTHENTICATION_CLASSES": [
  204. "rest_framework.authentication.TokenAuthentication",
  205. "rest_framework.authentication.SessionAuthentication",
  206. ],
  207. "DEFAULT_CONTENT_NEGOTIATION_CLASS": "vrobbler.negotiation.IgnoreClientContentNegotiation",
  208. "DEFAULT_FILTER_BACKENDS": [
  209. "django_filters.rest_framework.DjangoFilterBackend"
  210. ],
  211. "DEFAULT_PAGINATION_CLASS": "rest_framework.pagination.PageNumberPagination",
  212. "PAGE_SIZE": 200,
  213. }
  214. LOGIN_REDIRECT_URL = "/"
  215. AUTH_PASSWORD_VALIDATORS = [
  216. {
  217. "NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
  218. },
  219. {
  220. "NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",
  221. },
  222. {
  223. "NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",
  224. },
  225. {
  226. "NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",
  227. },
  228. ]
  229. # Internationalization
  230. # https://docs.djangoproject.com/en/3.1/topics/i18n/
  231. LANGUAGE_CODE = "en-us"
  232. TIME_ZONE = os.getenv("VROBBLER_TIME_ZONE", "America/New_York")
  233. USE_I18N = True
  234. USE_TZ = True
  235. # Static files (CSS, JavaScript, Images)
  236. # https://docs.djangoproject.com/en/3.1/howto/static-files/
  237. #
  238. from storages.backends import s3boto3
  239. USE_S3_STORAGE = os.getenv("VROBBLER_USE_S3", "False").lower() in TRUTHY
  240. if USE_S3_STORAGE:
  241. AWS_S3_ENDPOINT_URL = os.getenv("AWS_S3_ENDPOINT_URL", "")
  242. AWS_STORAGE_BUCKET_NAME = os.getenv("AWS_STORAGE_BUCKET_NAME", "")
  243. AWS_S3_ACCESS_KEY_ID = os.getenv("AWS_S3_ACCESS_KEY_ID")
  244. AWS_S3_SECRET_ACCESS_KEY = os.getenv("AWS_S3_SECRET_ACCESS_KEY")
  245. S3_ROOT = "/".join([AWS_S3_ENDPOINT_URL, AWS_STORAGE_BUCKET_NAME])
  246. print(f"Storing media on S3 at {S3_ROOT}")
  247. DEFAULT_FILE_STORAGE = "vrobbler.storages.MediaStorage"
  248. STATICFILES_STORAGE = "vrobbler.storages.StaticStorage"
  249. STATIC_URL = S3_ROOT + "/static/"
  250. MEDIA_URL = S3_ROOT + "/media/"
  251. else:
  252. STATIC_ROOT = os.getenv(
  253. "VROBBLER_STATIC_ROOT", os.path.join(PROJECT_ROOT, "static")
  254. )
  255. MEDIA_ROOT = os.getenv(
  256. "VROBBLER_MEDIA_ROOT", os.path.join(PROJECT_ROOT, "media")
  257. )
  258. STATIC_URL = os.getenv("VROBBLER_STATIC_URL", "/static/")
  259. MEDIA_URL = os.getenv("VROBBLER_MEDIA_URL", "/media/")
  260. JSON_LOGGING = os.getenv("VROBBLER_JSON_LOGGING", "false").lower() in TRUTHY
  261. LOG_TYPE = "json" if JSON_LOGGING else "log"
  262. default_level = "INFO"
  263. if DEBUG:
  264. default_level = "DEBUG"
  265. LOG_LEVEL = os.getenv("VROBBLER_LOG_LEVEL", default_level)
  266. LOG_FILE_PATH = os.getenv("VROBBLER_LOG_FILE_PATH", "/tmp/")
  267. LOGGING = {
  268. "version": 1,
  269. "disable_existing_loggers": False,
  270. "root": {
  271. "handlers": ["console", "file"],
  272. "level": LOG_LEVEL,
  273. "propagate": True,
  274. },
  275. "formatters": {
  276. "color": {
  277. "()": "colorlog.ColoredFormatter",
  278. # \r returns caret to line beginning, in tests this eats the silly dot that removes
  279. # the beautiful alignment produced below
  280. "format": "\r"
  281. "{log_color}{levelname:8s}{reset} "
  282. "{bold_cyan}{name}{reset}:"
  283. "{fg_bold_red}{lineno}{reset} "
  284. "{thin_yellow}{funcName} "
  285. "{thin_white}{message}"
  286. "{reset}",
  287. "style": "{",
  288. },
  289. "log": {"format": "%(asctime)s %(levelname)s %(message)s"},
  290. "json": {
  291. "()": "pythonjsonlogger.jsonlogger.JsonFormatter",
  292. "format": "%(levelname)s %(name) %(funcName) %(lineno) %(asctime)s %(message)s",
  293. },
  294. },
  295. "handlers": {
  296. "console": {
  297. "class": "logging.StreamHandler",
  298. "formatter": "color",
  299. "level": LOG_LEVEL,
  300. },
  301. "null": {
  302. "class": "logging.NullHandler",
  303. "level": LOG_LEVEL,
  304. },
  305. "sql": {
  306. "class": "logging.handlers.RotatingFileHandler",
  307. "filename": "".join([LOG_FILE_PATH, "vrobbler_sql.", LOG_TYPE]),
  308. "formatter": LOG_TYPE,
  309. "level": LOG_LEVEL,
  310. },
  311. "file": {
  312. "class": "logging.handlers.RotatingFileHandler",
  313. "filename": "".join([LOG_FILE_PATH, "vrobbler.", LOG_TYPE]),
  314. "formatter": LOG_TYPE,
  315. "level": LOG_LEVEL,
  316. },
  317. },
  318. "loggers": {
  319. # Quiet down our console a little
  320. "django": {
  321. "handlers": ["null"],
  322. "propagate": True,
  323. },
  324. "django.db.backends": {"handlers": ["null"]},
  325. "django.server": {"handlers": ["null"]},
  326. "pylast": {"handlers": ["null"], "propagate": False},
  327. "musicbrainzngs": {"handlers": ["null"], "propagate": False},
  328. "httpx": {"handlers": ["null"], "propagate": False},
  329. "vrobbler": {
  330. "handlers": ["console"],
  331. "propagate": False,
  332. },
  333. },
  334. }
  335. LOG_TO_CONSOLE = (
  336. os.getenv("VROBBLER_LOG_TO_CONSOLE", "false").lower() in TRUTHY
  337. )
  338. if LOG_TO_CONSOLE:
  339. LOGGING["loggers"]["django"]["handlers"] = ["console"]
  340. LOGGING["loggers"]["vrobbler"]["handlers"] = ["console"]