2014-09-13 02:50:36 +08:00
|
|
|
from django.conf import settings
|
|
|
|
|
2019-03-22 05:33:41 +08:00
|
|
|
from .. import Error, Tags, Warning, register
|
|
|
|
|
|
|
|
REFERRER_POLICY_VALUES = {
|
|
|
|
'no-referrer', 'no-referrer-when-downgrade', 'origin',
|
|
|
|
'origin-when-cross-origin', 'same-origin', 'strict-origin',
|
|
|
|
'strict-origin-when-cross-origin', 'unsafe-url',
|
|
|
|
}
|
2014-09-13 02:50:36 +08:00
|
|
|
|
|
|
|
SECRET_KEY_MIN_LENGTH = 50
|
|
|
|
SECRET_KEY_MIN_UNIQUE_CHARACTERS = 5
|
|
|
|
|
|
|
|
W001 = Warning(
|
|
|
|
"You do not have 'django.middleware.security.SecurityMiddleware' "
|
2015-11-07 23:12:37 +08:00
|
|
|
"in your MIDDLEWARE so the SECURE_HSTS_SECONDS, "
|
2019-03-22 05:33:41 +08:00
|
|
|
"SECURE_CONTENT_TYPE_NOSNIFF, SECURE_BROWSER_XSS_FILTER, "
|
|
|
|
"SECURE_REFERRER_POLICY, and SECURE_SSL_REDIRECT settings will have no "
|
|
|
|
"effect.",
|
2014-09-13 02:50:36 +08:00
|
|
|
id='security.W001',
|
|
|
|
)
|
|
|
|
|
|
|
|
W002 = Warning(
|
|
|
|
"You do not have "
|
|
|
|
"'django.middleware.clickjacking.XFrameOptionsMiddleware' in your "
|
2015-11-07 23:12:37 +08:00
|
|
|
"MIDDLEWARE, so your pages will not be served with an "
|
2014-09-13 02:50:36 +08:00
|
|
|
"'x-frame-options' header. Unless there is a good reason for your "
|
|
|
|
"site to be served in a frame, you should consider enabling this "
|
|
|
|
"header to help prevent clickjacking attacks.",
|
|
|
|
id='security.W002',
|
|
|
|
)
|
|
|
|
|
|
|
|
W004 = Warning(
|
|
|
|
"You have not set a value for the SECURE_HSTS_SECONDS setting. "
|
|
|
|
"If your entire site is served only over SSL, you may want to consider "
|
|
|
|
"setting a value and enabling HTTP Strict Transport Security. "
|
|
|
|
"Be sure to read the documentation first; enabling HSTS carelessly "
|
|
|
|
"can cause serious, irreversible problems.",
|
|
|
|
id='security.W004',
|
|
|
|
)
|
|
|
|
|
|
|
|
W005 = Warning(
|
|
|
|
"You have not set the SECURE_HSTS_INCLUDE_SUBDOMAINS setting to True. "
|
|
|
|
"Without this, your site is potentially vulnerable to attack "
|
|
|
|
"via an insecure connection to a subdomain. Only set this to True if "
|
|
|
|
"you are certain that all subdomains of your domain should be served "
|
|
|
|
"exclusively via SSL.",
|
|
|
|
id='security.W005',
|
|
|
|
)
|
|
|
|
|
|
|
|
W006 = Warning(
|
|
|
|
"Your SECURE_CONTENT_TYPE_NOSNIFF setting is not set to True, "
|
|
|
|
"so your pages will not be served with an "
|
2018-10-30 06:19:04 +08:00
|
|
|
"'X-Content-Type-Options: nosniff' header. "
|
2014-09-13 02:50:36 +08:00
|
|
|
"You should consider enabling this header to prevent the "
|
|
|
|
"browser from identifying content types incorrectly.",
|
|
|
|
id='security.W006',
|
|
|
|
)
|
|
|
|
|
|
|
|
W008 = Warning(
|
|
|
|
"Your SECURE_SSL_REDIRECT setting is not set to True. "
|
|
|
|
"Unless your site should be available over both SSL and non-SSL "
|
|
|
|
"connections, you may want to either set this setting True "
|
|
|
|
"or configure a load balancer or reverse-proxy server "
|
|
|
|
"to redirect all connections to HTTPS.",
|
|
|
|
id='security.W008',
|
|
|
|
)
|
|
|
|
|
|
|
|
W009 = Warning(
|
|
|
|
"Your SECRET_KEY has less than %(min_length)s characters or less than "
|
|
|
|
"%(min_unique_chars)s unique characters. Please generate a long and random "
|
|
|
|
"SECRET_KEY, otherwise many of Django's security-critical features will be "
|
|
|
|
"vulnerable to attack." % {
|
|
|
|
'min_length': SECRET_KEY_MIN_LENGTH,
|
|
|
|
'min_unique_chars': SECRET_KEY_MIN_UNIQUE_CHARACTERS,
|
|
|
|
},
|
|
|
|
id='security.W009',
|
|
|
|
)
|
|
|
|
|
|
|
|
W018 = Warning(
|
|
|
|
"You should not have DEBUG set to True in deployment.",
|
|
|
|
id='security.W018',
|
|
|
|
)
|
|
|
|
|
|
|
|
W019 = Warning(
|
|
|
|
"You have "
|
|
|
|
"'django.middleware.clickjacking.XFrameOptionsMiddleware' in your "
|
2015-11-07 23:12:37 +08:00
|
|
|
"MIDDLEWARE, but X_FRAME_OPTIONS is not set to 'DENY'. "
|
2019-09-07 15:52:10 +08:00
|
|
|
"Unless there is a good reason for your site to serve other parts of "
|
|
|
|
"itself in a frame, you should change it to 'DENY'.",
|
2014-09-13 02:50:36 +08:00
|
|
|
id='security.W019',
|
|
|
|
)
|
|
|
|
|
2015-06-17 04:08:03 +08:00
|
|
|
W020 = Warning(
|
|
|
|
"ALLOWED_HOSTS must not be empty in deployment.",
|
|
|
|
id='security.W020',
|
|
|
|
)
|
|
|
|
|
2016-07-26 20:05:27 +08:00
|
|
|
W021 = Warning(
|
|
|
|
"You have not set the SECURE_HSTS_PRELOAD setting to True. Without this, "
|
|
|
|
"your site cannot be submitted to the browser preload list.",
|
|
|
|
id='security.W021',
|
|
|
|
)
|
|
|
|
|
2019-03-22 05:33:41 +08:00
|
|
|
W022 = Warning(
|
|
|
|
'You have not set the SECURE_REFERRER_POLICY setting. Without this, your '
|
|
|
|
'site will not send a Referrer-Policy header. You should consider '
|
|
|
|
'enabling this header to protect user privacy.',
|
|
|
|
id='security.W022',
|
|
|
|
)
|
|
|
|
|
|
|
|
E023 = Error(
|
|
|
|
'You have set the SECURE_REFERRER_POLICY setting to an invalid value.',
|
|
|
|
hint='Valid values are: {}.'.format(', '.join(sorted(REFERRER_POLICY_VALUES))),
|
|
|
|
id='security.E023',
|
|
|
|
)
|
|
|
|
|
2014-09-13 02:50:36 +08:00
|
|
|
|
|
|
|
def _security_middleware():
|
2017-01-01 02:24:00 +08:00
|
|
|
return 'django.middleware.security.SecurityMiddleware' in settings.MIDDLEWARE
|
2014-09-13 02:50:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
def _xframe_middleware():
|
2017-01-01 02:24:00 +08:00
|
|
|
return 'django.middleware.clickjacking.XFrameOptionsMiddleware' in settings.MIDDLEWARE
|
2014-09-13 02:50:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_security_middleware(app_configs, **kwargs):
|
|
|
|
passed_check = _security_middleware()
|
2017-01-01 02:24:00 +08:00
|
|
|
return [] if passed_check else [W001]
|
2014-09-13 02:50:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_xframe_options_middleware(app_configs, **kwargs):
|
|
|
|
passed_check = _xframe_middleware()
|
2017-01-01 02:24:00 +08:00
|
|
|
return [] if passed_check else [W002]
|
2014-09-13 02:50:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_sts(app_configs, **kwargs):
|
|
|
|
passed_check = not _security_middleware() or settings.SECURE_HSTS_SECONDS
|
|
|
|
return [] if passed_check else [W004]
|
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_sts_include_subdomains(app_configs, **kwargs):
|
|
|
|
passed_check = (
|
|
|
|
not _security_middleware() or
|
|
|
|
not settings.SECURE_HSTS_SECONDS or
|
|
|
|
settings.SECURE_HSTS_INCLUDE_SUBDOMAINS is True
|
|
|
|
)
|
|
|
|
return [] if passed_check else [W005]
|
|
|
|
|
|
|
|
|
2016-07-26 20:05:27 +08:00
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_sts_preload(app_configs, **kwargs):
|
|
|
|
passed_check = (
|
|
|
|
not _security_middleware() or
|
|
|
|
not settings.SECURE_HSTS_SECONDS or
|
|
|
|
settings.SECURE_HSTS_PRELOAD is True
|
|
|
|
)
|
|
|
|
return [] if passed_check else [W021]
|
|
|
|
|
|
|
|
|
2014-09-13 02:50:36 +08:00
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_content_type_nosniff(app_configs, **kwargs):
|
|
|
|
passed_check = (
|
|
|
|
not _security_middleware() or
|
|
|
|
settings.SECURE_CONTENT_TYPE_NOSNIFF is True
|
|
|
|
)
|
|
|
|
return [] if passed_check else [W006]
|
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_ssl_redirect(app_configs, **kwargs):
|
|
|
|
passed_check = (
|
|
|
|
not _security_middleware() or
|
|
|
|
settings.SECURE_SSL_REDIRECT is True
|
|
|
|
)
|
|
|
|
return [] if passed_check else [W008]
|
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_secret_key(app_configs, **kwargs):
|
|
|
|
passed_check = (
|
|
|
|
getattr(settings, 'SECRET_KEY', None) and
|
|
|
|
len(set(settings.SECRET_KEY)) >= SECRET_KEY_MIN_UNIQUE_CHARACTERS and
|
|
|
|
len(settings.SECRET_KEY) >= SECRET_KEY_MIN_LENGTH
|
|
|
|
)
|
|
|
|
return [] if passed_check else [W009]
|
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_debug(app_configs, **kwargs):
|
|
|
|
passed_check = not settings.DEBUG
|
|
|
|
return [] if passed_check else [W018]
|
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_xframe_deny(app_configs, **kwargs):
|
|
|
|
passed_check = (
|
|
|
|
not _xframe_middleware() or
|
|
|
|
settings.X_FRAME_OPTIONS == 'DENY'
|
|
|
|
)
|
2017-01-01 02:24:00 +08:00
|
|
|
return [] if passed_check else [W019]
|
2015-06-17 04:08:03 +08:00
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_allowed_hosts(app_configs, **kwargs):
|
|
|
|
return [] if settings.ALLOWED_HOSTS else [W020]
|
2019-03-22 05:33:41 +08:00
|
|
|
|
|
|
|
|
|
|
|
@register(Tags.security, deploy=True)
|
|
|
|
def check_referrer_policy(app_configs, **kwargs):
|
|
|
|
if _security_middleware():
|
|
|
|
if settings.SECURE_REFERRER_POLICY is None:
|
|
|
|
return [W022]
|
|
|
|
# Support a comma-separated string or iterable of values to allow fallback.
|
|
|
|
if isinstance(settings.SECURE_REFERRER_POLICY, str):
|
|
|
|
values = {v.strip() for v in settings.SECURE_REFERRER_POLICY.split(',')}
|
|
|
|
else:
|
|
|
|
values = set(settings.SECURE_REFERRER_POLICY)
|
|
|
|
if not values <= REFERRER_POLICY_VALUES:
|
|
|
|
return [E023]
|
|
|
|
return []
|