2019-02-25 16:21:56 +00:00
|
|
|
"""passbook HIBP Models"""
|
|
|
|
from hashlib import sha1
|
2020-07-20 14:58:48 +01:00
|
|
|
from typing import Type
|
2019-02-25 16:21:56 +00:00
|
|
|
|
|
|
|
from django.db import models
|
2020-07-20 14:58:48 +01:00
|
|
|
from django.forms import ModelForm
|
2019-02-25 16:21:56 +00:00
|
|
|
from django.utils.translation import gettext as _
|
|
|
|
from requests import get
|
2020-08-21 23:42:15 +01:00
|
|
|
from rest_framework.serializers import BaseSerializer
|
2019-10-01 09:24:10 +01:00
|
|
|
from structlog import get_logger
|
2019-02-25 16:21:56 +00:00
|
|
|
|
2020-05-16 17:07:00 +01:00
|
|
|
from passbook.policies.models import Policy, PolicyResult
|
2020-07-10 19:57:15 +01:00
|
|
|
from passbook.policies.types import PolicyRequest
|
2019-02-25 16:21:56 +00:00
|
|
|
|
2019-10-04 09:08:53 +01:00
|
|
|
LOGGER = get_logger()
|
2019-02-25 16:21:56 +00:00
|
|
|
|
2019-12-31 11:51:16 +00:00
|
|
|
|
2019-02-25 16:21:56 +00:00
|
|
|
class HaveIBeenPwendPolicy(Policy):
|
2020-07-01 17:40:52 +01:00
|
|
|
"""Check if password is on HaveIBeenPwned's list by uploading the first
|
2019-02-25 16:21:56 +00:00
|
|
|
5 characters of the SHA1 Hash."""
|
|
|
|
|
2020-07-10 19:57:15 +01:00
|
|
|
password_field = models.TextField(
|
|
|
|
default="password",
|
|
|
|
help_text=_(
|
|
|
|
"Field key to check, field keys defined in Prompt stages are available."
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
2019-02-25 16:21:56 +00:00
|
|
|
allowed_count = models.IntegerField(default=0)
|
|
|
|
|
2020-08-21 23:42:15 +01:00
|
|
|
@property
|
|
|
|
def serializer(self) -> BaseSerializer:
|
|
|
|
from passbook.policies.hibp.api import HaveIBeenPwendPolicySerializer
|
|
|
|
|
|
|
|
return HaveIBeenPwendPolicySerializer
|
|
|
|
|
2020-07-20 14:58:48 +01:00
|
|
|
def form(self) -> Type[ModelForm]:
|
|
|
|
from passbook.policies.hibp.forms import HaveIBeenPwnedPolicyForm
|
|
|
|
|
|
|
|
return HaveIBeenPwnedPolicyForm
|
2019-02-25 16:21:56 +00:00
|
|
|
|
2020-07-10 19:57:15 +01:00
|
|
|
def passes(self, request: PolicyRequest) -> PolicyResult:
|
2019-02-25 16:21:56 +00:00
|
|
|
"""Check if password is in HIBP DB. Hashes given Password with SHA1, uses the first 5
|
|
|
|
characters of Password in request and checks if full hash is in response. Returns 0
|
|
|
|
if Password is not in result otherwise the count of how many times it was used."""
|
2020-07-10 19:57:15 +01:00
|
|
|
if self.password_field not in request.context:
|
|
|
|
LOGGER.warning(
|
|
|
|
"Password field not set in Policy Request",
|
|
|
|
field=self.password_field,
|
|
|
|
fields=request.context.keys(),
|
|
|
|
)
|
|
|
|
password = request.context[self.password_field]
|
|
|
|
|
2019-12-31 11:51:16 +00:00
|
|
|
pw_hash = sha1(password.encode("utf-8")).hexdigest() # nosec
|
2020-07-10 19:57:15 +01:00
|
|
|
url = f"https://api.pwnedpasswords.com/range/{pw_hash[:5]}"
|
2019-02-25 16:21:56 +00:00
|
|
|
result = get(url).text
|
|
|
|
final_count = 0
|
2019-12-31 11:51:16 +00:00
|
|
|
for line in result.split("\r\n"):
|
|
|
|
full_hash, count = line.split(":")
|
2019-02-25 16:21:56 +00:00
|
|
|
if pw_hash[5:] == full_hash.lower():
|
|
|
|
final_count = int(count)
|
2020-02-18 20:35:58 +00:00
|
|
|
LOGGER.debug("got hibp result", count=final_count, hash=pw_hash[:5])
|
2019-02-25 16:21:56 +00:00
|
|
|
if final_count > self.allowed_count:
|
2019-12-31 11:51:16 +00:00
|
|
|
message = _(
|
|
|
|
"Password exists on %(count)d online lists." % {"count": final_count}
|
|
|
|
)
|
2019-10-01 09:17:39 +01:00
|
|
|
return PolicyResult(False, message)
|
|
|
|
return PolicyResult(True)
|
2019-02-25 16:21:56 +00:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
2019-12-31 11:51:16 +00:00
|
|
|
verbose_name = _("Have I Been Pwned Policy")
|
|
|
|
verbose_name_plural = _("Have I Been Pwned Policies")
|