Source code for mimesis.builtins.da

# -*- coding: utf-8 -*-

"""Specific data provider for Denmark (da)."""
import operator
from typing import Optional, Tuple

from mimesis import Datetime
from mimesis.builtins.base import BaseSpecProvider
from mimesis.locales import Locale
from mimesis.typing import Seed

__all__ = ["DenmarkSpecProvider"]


[docs]class DenmarkSpecProvider(BaseSpecProvider): """Class that provides special data for Denmark (da)."""
[docs] def __init__(self, seed: Optional[Seed] = None) -> None: """Initialize attributes.""" super().__init__(locale=Locale.DA, seed=seed) self._datetime = Datetime(locale=Locale.DA, seed=seed) self._checksum_factors = (4, 3, 2, 7, 6, 5, 4, 3, 2)
[docs] class Meta: """The name of the provider.""" name = "denmark_provider"
def _calculate_century_selector(self, year: int) -> int: if 1858 <= year < 1900: return self.random.randint(5, 8) elif 1900 <= year < 1937: return self.random.randint(0, 3) elif 1937 <= year < 2000: return self.random.choice([4, 9]) elif 2000 <= year < 2037: return self.random.randint(4, 9) raise ValueError("Invalid year") def _calculate_checksum(self, cpr_nr_no_checksum: str) -> int: """Calculate the CPR number checksum. The CPR checksum can be checked by: 1. Multiplying each digit in the CPR number with a corresponding fixed factor (self._checksum_factors) to produce a list of products. 2. Summing up all the products, including the checksum, and checking that the resulting sum modulo 11 is 0. As such the checksum can be determined by reordering the formula, as: * 11 - (sum_without_checksum % 11) If the sum_without_checksum is 0, the resulting checksum is 11, but returned as 0 according to the official rules. If the sum_without_checksum is 1, the resulting checksum is 10, and thus invalid as the checksum is only 1 digit, hence this implies that the generated serial_number is invalid. Note: This method does not handle checksum == 10 case. It is handled by recursion in _generate_serial_checksum. """ cpr_digits = map(int, cpr_nr_no_checksum) cpr_digit_products = list( map( operator.mul, cpr_digits, self._checksum_factors, ) ) remainder = sum(cpr_digit_products) % 11 if remainder == 0: return 0 return 11 - remainder def _generate_serial_checksum(self, cpr_century: str) -> Tuple[str, int]: """Generate a serial number and checksum from cpr_century.""" serial_number = "{:02d}".format(self.random.randint(0, 99)) cpr_nr_no_checksum = "{}{}".format(cpr_century, serial_number) checksum = self._calculate_checksum(cpr_nr_no_checksum) if checksum == 10: return self._generate_serial_checksum(cpr_century) return serial_number, checksum
[docs] def cpr(self) -> str: """Generate a random CPR number (Central Person Registry). :return: CPR number. :Example: 0405420694 """ date = self._datetime.date(start=1858, end=2021) cpr_date = "{:02d}{:02d}{}".format( date.day, date.month, str(date.year)[-2:], ) century_selector = self._calculate_century_selector(date.year) cpr_century = "{}{}".format(cpr_date, century_selector) serial_number, checksum = self._generate_serial_checksum(cpr_century) cpr_nr = "{}{}{}".format(cpr_century, serial_number, checksum) return cpr_nr