production.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. import os
  2. import random
  3. import string
  4. import dj_database_url
  5. from .base import * # noqa: F403
  6. DEBUG = False
  7. # DJANGO_SECRET_KEY *should* be specified in the environment. If it's not, generate an ephemeral key.
  8. if "DJANGO_SECRET_KEY" in os.environ:
  9. SECRET_KEY = os.environ["DJANGO_SECRET_KEY"]
  10. else:
  11. # Use if/else rather than a default value to avoid calculating this if we don't need it
  12. print( # noqa: T201
  13. "WARNING: DJANGO_SECRET_KEY not found in os.environ. Generating ephemeral SECRET_KEY."
  14. )
  15. SECRET_KEY = "".join(
  16. [random.SystemRandom().choice(string.printable) for i in range(50)]
  17. )
  18. # Make sure Django can detect a secure connection properly on Heroku:
  19. SECURE_PROXY_SSL_HEADER = ("HTTP_X_FORWARDED_PROTO", "https")
  20. # Redirect all requests to HTTPS
  21. SECURE_SSL_REDIRECT = os.getenv("DJANGO_SECURE_SSL_REDIRECT", "off") == "on"
  22. # Accept all hostnames, since we don't know in advance which hostname will be used for any given Heroku instance.
  23. # IMPORTANT: Set this to a real hostname when using this in production!
  24. # See https://docs.djangoproject.com/en/3.2/ref/settings/#allowed-hosts
  25. ALLOWED_HOSTS = os.getenv("DJANGO_ALLOWED_HOSTS", "*").split(";")
  26. EMAIL_BACKEND = "django.core.mail.backends.console.EmailBackend"
  27. # This is used by Wagtail's email notifications for constructing absolute
  28. # URLs. Please set to the domain that users will access the admin site.
  29. if "PRIMARY_HOST" in os.environ:
  30. WAGTAILADMIN_BASE_URL = "https://{}".format(os.environ["PRIMARY_HOST"])
  31. db_from_env = dj_database_url.config(conn_max_age=500)
  32. DATABASES["default"].update(db_from_env)
  33. # AWS creds may be used for S3 and/or Elasticsearch
  34. AWS_ACCESS_KEY_ID = os.getenv("AWS_ACCESS_KEY_ID", "")
  35. AWS_SECRET_ACCESS_KEY = os.getenv("AWS_SECRET_ACCESS_KEY", "")
  36. AWS_REGION = os.getenv("AWS_REGION", "")
  37. # Server-side cache settings. Do not confuse with front-end cache.
  38. # https://docs.djangoproject.com/en/stable/topics/cache/
  39. # If the server has a Redis instance exposed via a URL string in the REDIS_URL
  40. # environment variable, prefer that. Otherwise use the database backend. We
  41. # usually use Redis in production and database backend on staging and dev. In
  42. # order to use database cache backend you need to run
  43. # "./manage.py createcachetable" to create a table for the cache.
  44. #
  45. # Do not use the same Redis instance for other things like Celery!
  46. # Prefer the TLS connection URL over non
  47. REDIS_URL = os.environ.get("REDIS_TLS_URL", os.environ.get("REDIS_URL"))
  48. if REDIS_URL:
  49. connection_pool_kwargs = {}
  50. if REDIS_URL.startswith("rediss"):
  51. # Heroku Redis uses self-signed certificates for secure redis connections
  52. # When using TLS, we need to disable certificate validation checks.
  53. connection_pool_kwargs["ssl_cert_reqs"] = None
  54. redis_options = {
  55. "IGNORE_EXCEPTIONS": True,
  56. "SOCKET_CONNECT_TIMEOUT": 2, # seconds
  57. "SOCKET_TIMEOUT": 2, # seconds
  58. "CONNECTION_POOL_KWARGS": connection_pool_kwargs,
  59. }
  60. CACHES = {
  61. "default": {
  62. "BACKEND": "django_redis.cache.RedisCache",
  63. "LOCATION": REDIS_URL + "/0",
  64. "OPTIONS": redis_options,
  65. },
  66. "renditions": {
  67. "BACKEND": "django_redis.cache.RedisCache",
  68. "LOCATION": REDIS_URL + "/1",
  69. "OPTIONS": redis_options,
  70. },
  71. }
  72. DJANGO_REDIS_LOG_IGNORED_EXCEPTIONS = True
  73. else:
  74. CACHES = {
  75. "default": {
  76. "BACKEND": "django.core.cache.backends.locmem.LocMemCache",
  77. "LOCATION": "bakerydemo",
  78. }
  79. }
  80. # Configure Elasticsearch, if present in os.environ
  81. ELASTICSEARCH_ENDPOINT = os.getenv("ELASTICSEARCH_ENDPOINT", "")
  82. if ELASTICSEARCH_ENDPOINT:
  83. from elasticsearch import RequestsHttpConnection
  84. WAGTAILSEARCH_BACKENDS = {
  85. "default": {
  86. "BACKEND": "wagtail.search.backends.elasticsearch5",
  87. "HOSTS": [
  88. {
  89. "host": ELASTICSEARCH_ENDPOINT,
  90. "port": int(os.getenv("ELASTICSEARCH_PORT", "9200")),
  91. "use_ssl": os.getenv("ELASTICSEARCH_USE_SSL", "off") == "on",
  92. "verify_certs": os.getenv("ELASTICSEARCH_VERIFY_CERTS", "off")
  93. == "on",
  94. }
  95. ],
  96. "OPTIONS": {
  97. "connection_class": RequestsHttpConnection,
  98. },
  99. }
  100. }
  101. if AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY:
  102. from aws_requests_auth.aws_auth import AWSRequestsAuth
  103. WAGTAILSEARCH_BACKENDS["default"]["HOSTS"][0]["http_auth"] = AWSRequestsAuth(
  104. aws_access_key=AWS_ACCESS_KEY_ID,
  105. aws_secret_access_key=AWS_SECRET_ACCESS_KEY,
  106. aws_token=os.getenv("AWS_SESSION_TOKEN", ""),
  107. aws_host=ELASTICSEARCH_ENDPOINT,
  108. aws_region=AWS_REGION,
  109. aws_service="es",
  110. )
  111. elif AWS_REGION:
  112. # No API keys in the environ, so attempt to discover them with Boto instead, per:
  113. # https://boto3.amazonaws.com/v1/documentation/api/latest/guide/configuration.html#configuring-credentials
  114. # This may be useful if your credentials are obtained via EC2 instance meta data.
  115. from aws_requests_auth.boto_utils import BotoAWSRequestsAuth
  116. WAGTAILSEARCH_BACKENDS["default"]["HOSTS"][0][
  117. "http_auth"
  118. ] = BotoAWSRequestsAuth(
  119. aws_host=ELASTICSEARCH_ENDPOINT,
  120. aws_region=AWS_REGION,
  121. aws_service="es",
  122. )
  123. # Simplified static file serving.
  124. # https://warehouse.python.org/project/whitenoise/
  125. MIDDLEWARE.append("whitenoise.middleware.WhiteNoiseMiddleware")
  126. STATICFILES_STORAGE = "whitenoise.storage.CompressedManifestStaticFilesStorage"
  127. if "AWS_STORAGE_BUCKET_NAME" in os.environ:
  128. AWS_STORAGE_BUCKET_NAME = os.getenv("AWS_STORAGE_BUCKET_NAME")
  129. AWS_QUERYSTRING_AUTH = False
  130. INSTALLED_APPS.append("storages")
  131. DEFAULT_FILE_STORAGE = "storages.backends.s3boto3.S3Boto3Storage"
  132. AWS_S3_FILE_OVERWRITE = False
  133. AWS_DEFAULT_ACL = "private"
  134. if "AWS_S3_CUSTOM_DOMAIN" in os.environ:
  135. AWS_S3_CUSTOM_DOMAIN = os.environ["AWS_S3_CUSTOM_DOMAIN"]
  136. if "AWS_S3_REGION_NAME" in os.environ:
  137. AWS_S3_REGION_NAME = os.environ["AWS_S3_REGION_NAME"]
  138. if "GS_BUCKET_NAME" in os.environ:
  139. GS_BUCKET_NAME = os.getenv("GS_BUCKET_NAME")
  140. GS_PROJECT_ID = os.getenv("GS_PROJECT_ID")
  141. GS_DEFAULT_ACL = "publicRead"
  142. GS_AUTO_CREATE_BUCKET = True
  143. INSTALLED_APPS.append("storages")
  144. DEFAULT_FILE_STORAGE = "storages.backends.gcloud.GoogleCloudStorage"
  145. LOGGING = {
  146. "version": 1,
  147. "disable_existing_loggers": False,
  148. "handlers": {
  149. "console": {
  150. "class": "logging.StreamHandler",
  151. },
  152. },
  153. "loggers": {
  154. "django": {
  155. "handlers": ["console"],
  156. "level": os.getenv("DJANGO_LOG_LEVEL", "INFO"),
  157. },
  158. },
  159. }
  160. # Front-end cache
  161. # This configuration is used to allow purging pages from cache when they are
  162. # published.
  163. # These settings are usually used only on the production sites.
  164. # This is a configuration of the CDN/front-end cache that is used to cache the
  165. # production websites.
  166. # https://docs.wagtail.org/en/latest/reference/contrib/frontendcache.html
  167. # The backend can be configured to use an account-wide API key, or an API token with
  168. # restricted access.
  169. if (
  170. "FRONTEND_CACHE_CLOUDFLARE_TOKEN" in os.environ
  171. or "FRONTEND_CACHE_CLOUDFLARE_BEARER_TOKEN" in os.environ
  172. ):
  173. INSTALLED_APPS.append("wagtail.contrib.frontend_cache")
  174. WAGTAILFRONTENDCACHE = {
  175. "default": {
  176. "BACKEND": "wagtail.contrib.frontend_cache.backends.CloudflareBackend",
  177. "ZONEID": os.environ["FRONTEND_CACHE_CLOUDFLARE_ZONEID"],
  178. }
  179. }
  180. if "FRONTEND_CACHE_CLOUDFLARE_TOKEN" in os.environ:
  181. # To use an account-wide API key, set the following:
  182. # * $FRONTEND_CACHE_CLOUDFLARE_TOKEN
  183. # * $FRONTEND_CACHE_CLOUDFLARE_EMAIL
  184. # * $FRONTEND_CACHE_CLOUDFLARE_ZONEID
  185. # These can be obtained from a sysadmin.
  186. WAGTAILFRONTENDCACHE["default"].update(
  187. {
  188. "EMAIL": os.environ["FRONTEND_CACHE_CLOUDFLARE_EMAIL"],
  189. "TOKEN": os.environ["FRONTEND_CACHE_CLOUDFLARE_TOKEN"],
  190. }
  191. )
  192. else:
  193. # To use an API token with restricted access, set the following:
  194. # * $FRONTEND_CACHE_CLOUDFLARE_BEARER_TOKEN
  195. # * $FRONTEND_CACHE_CLOUDFLARE_ZONEID
  196. WAGTAILFRONTENDCACHE["default"].update(
  197. {"BEARER_TOKEN": os.environ["FRONTEND_CACHE_CLOUDFLARE_BEARER_TOKEN"]}
  198. )
  199. # Basic authentication settings
  200. # These are settings to configure the third-party library:
  201. # https://gitlab.com/tmkn/django-basic-auth-ip-whitelist
  202. if os.environ.get("BASIC_AUTH_ENABLED", "false").lower().strip() == "true":
  203. # Insert basic auth as a first middleware to be checked first, before
  204. # anything else.
  205. MIDDLEWARE.insert(0, "baipw.middleware.BasicAuthIPWhitelistMiddleware")
  206. # This is the credentials users will have to use to access the site.
  207. BASIC_AUTH_LOGIN = os.environ.get("BASIC_AUTH_LOGIN", "wagtail")
  208. BASIC_AUTH_PASSWORD = os.environ.get("BASIC_AUTH_PASSWORD", "wagtail")
  209. # Wagtail requires Authorization header to be present for the previews
  210. BASIC_AUTH_DISABLE_CONSUMING_AUTHORIZATION_HEADER = True
  211. # This is the list of hosts that website can be accessed without basic auth
  212. # check.
  213. if "BASIC_AUTH_WHITELISTED_HTTP_HOSTS" in os.environ:
  214. BASIC_AUTH_WHITELISTED_HTTP_HOSTS = os.environ[
  215. "BASIC_AUTH_WHITELISTED_HTTP_HOSTS"
  216. ].split(",")
  217. BASIC_AUTH_RESPONSE_TEMPLATE = "base/basic_auth.html"
  218. # Force HTTPS redirect (enabled by default!)
  219. # https://docs.djangoproject.com/en/stable/ref/settings/#secure-ssl-redirect
  220. SECURE_SSL_REDIRECT = True
  221. # This will allow the cache to swallow the fact that the website is behind TLS
  222. # and inform the Django using "X-Forwarded-Proto" HTTP header.
  223. # https://docs.djangoproject.com/en/stable/ref/settings/#secure-proxy-ssl-header
  224. SECURE_PROXY_SSL_HEADER = ("HTTP_X_FORWARDED_PROTO", "https")
  225. # This is a setting activating the HSTS header. This will enforce the visitors to use
  226. # HTTPS for an amount of time specified in the header. Since we are expecting our apps
  227. # to run via TLS by default, this header is activated by default.
  228. # The header can be deactivated by setting this setting to 0, as it is done in the
  229. # dev and testing settings.
  230. # https://docs.djangoproject.com/en/stable/ref/settings/#secure-hsts-seconds
  231. DEFAULT_HSTS_SECONDS = 30 * 24 * 60 * 60 # 30 days
  232. SECURE_HSTS_SECONDS = int(
  233. os.environ.get("SECURE_HSTS_SECONDS", DEFAULT_HSTS_SECONDS)
  234. ) # noqa
  235. # Do not use the `includeSubDomains` directive for HSTS. This needs to be prevented
  236. # because the apps are running on client domains (or our own for staging), that are
  237. # being used for other applications as well. We should therefore not impose any
  238. # restrictions on these unrelated applications.
  239. # https://docs.djangoproject.com/en/3.2/ref/settings/#secure-hsts-include-subdomains
  240. SECURE_HSTS_INCLUDE_SUBDOMAINS = False
  241. # https://docs.djangoproject.com/en/stable/ref/settings/#secure-browser-xss-filter
  242. SECURE_BROWSER_XSS_FILTER = True
  243. # https://docs.djangoproject.com/en/stable/ref/settings/#secure-content-type-nosniff
  244. SECURE_CONTENT_TYPE_NOSNIFF = True
  245. # Referrer-policy header settings.
  246. # https://django-referrer-policy.readthedocs.io/en/1.0/
  247. REFERRER_POLICY = os.environ.get( # noqa
  248. "SECURE_REFERRER_POLICY", "no-referrer-when-downgrade"
  249. ).strip()
  250. # Allow the redirect importer to work in load-balanced / cloud environments.
  251. # https://docs.wagtail.io/en/v2.13/reference/settings.html#redirects
  252. WAGTAIL_REDIRECTS_FILE_STORAGE = "cache"