Source code for mimesis.providers.date

"""Provider of data related to date and time."""

import typing as t
from calendar import monthrange
from datetime import date, datetime, time, timedelta

from mimesis.compat import pytz
from mimesis.datasets import GMT_OFFSETS, ROMAN_NUMS, TIMEZONES
from mimesis.enums import TimestampFormat, TimezoneRegion
from mimesis.providers.base import BaseDataProvider
from mimesis.types import Date, DateTime, Time

__all__ = ["Datetime"]


[docs] class Datetime(BaseDataProvider): """Class for generating data related to the date and time.""" # See: https://git.io/Jf15A _CURRENT_YEAR = datetime.now().year class Meta: name = "datetime" datafile = f"{name}.json"
[docs] @staticmethod def bulk_create_datetimes( date_start: DateTime, date_end: DateTime, **kwargs: t.Any, ) -> list[DateTime]: """Bulk create datetime objects. This method creates a list of datetime objects from ``date_start`` to ``date_end``. You can use the following keyword arguments: * ``days`` * ``hours`` * ``minutes`` * ``seconds`` * ``microseconds`` .. warning:: Empty ``**kwargs`` produces **timedelta(0)** which obviously cannot be used as step, so you have to pass valid ``**kwargs`` for :py:class:`datetime.timedelta` which will be used as a step by which ``date_start`` will be incremented until it reaches ``date_end`` to avoid infinite loop which eventually leads to ``OverflowError``. See :py:class:`datetime.timedelta` for more details. :param date_start: Begin of the range. :param date_end: End of the range. :param kwargs: Keyword arguments for :py:class:`datetime.timedelta` :return: List of datetime objects :raises: ValueError: When ``date_start``/``date_end`` not passed, when ``date_start`` larger than ``date_end`` or when the given keywords for `datetime.timedelta` represent a non-positive timedelta. """ dt_objects = [] if not date_start and not date_end: raise ValueError("You must pass date_start and date_end") if date_end < date_start: raise ValueError("date_start can not be larger than date_end") if timedelta(**kwargs) <= timedelta(): raise ValueError("timedelta must be positive") while date_start <= date_end: date_start += timedelta(**kwargs) dt_objects.append(date_start) return dt_objects
[docs] def week_date(self, start: int = 2017, end: int = _CURRENT_YEAR) -> str: """Generates week number with year. :param start: Starting year. :param end: Ending year. :return: Week number. """ year = self.year(start, end) week = self.random.randint(1, 52) return f"{year}-W{week}"
[docs] def day_of_week(self, abbr: bool = False) -> str: """Generates a random day of the week. :param abbr: Abbreviated day name. :return: Day of the week. """ key = "abbr" if abbr else "name" days: list[str] = self._extract(["day", key]) return self.random.choice(days)
[docs] def month(self, abbr: bool = False) -> str: """Generates a random month of the year. :param abbr: Abbreviated month name. :return: Month name. """ key = "abbr" if abbr else "name" months: list[str] = self._extract(["month", key]) return self.random.choice(months)
[docs] def year(self, minimum: int = 1990, maximum: int = _CURRENT_YEAR) -> int: """Generates a random year. :param minimum: Minimum value. :param maximum: Maximum value. :return: Year. """ return self.random.randint(minimum, maximum)
[docs] def century(self) -> str: """Generates a random century. :return: Century. """ return self.random.choice(ROMAN_NUMS)
[docs] def periodicity(self) -> str: """Generates a random periodicity string. :return: Periodicity. """ periodicity: list[str] = self._extract(["periodicity"]) return self.random.choice(periodicity)
[docs] def date(self, start: int = 2000, end: int = _CURRENT_YEAR) -> Date: """Generates a random date object. :param start: Minimum value of year. :param end: Maximum value of year. :return: Formatted date. """ year = self.random.randint(start, end) month = self.random.randint(1, 12) day = self.random.randint(1, monthrange(year, month)[1]) date_object = date(year, month, day) return date_object
[docs] def formatted_date(self, fmt: str = "", **kwargs: t.Any) -> str: """Generates random date as string. :param fmt: The format of date, if None then use standard accepted in the current locale. :param kwargs: Keyword arguments for :meth:`~.date()` :return: Formatted date. """ date_obj = self.date(**kwargs) if not fmt: fmt = self._extract(["formats", "date"]) return date_obj.strftime(fmt)
[docs] def time(self) -> Time: """Generates a random time object. :return: ``datetime.time`` object. """ random_time = time( self.random.randint(0, 23), self.random.randint(0, 59), self.random.randint(0, 59), self.random.randint(0, 999999), ) return random_time
[docs] def formatted_time(self, fmt: str = "") -> str: """Generates formatted time as string. :param fmt: The format of time, if None then use standard accepted in the current locale. :return: String formatted time. """ time_obj = self.time() if not fmt: fmt = self._extract(["formats", "time"]) return time_obj.strftime(fmt)
[docs] def day_of_month(self) -> int: """Generates a random day of the month, from 1 to 31. :return: Random value from 1 to 31. """ return self.random.randint(1, 31)
[docs] def timezone(self, region: TimezoneRegion | None = None) -> str: """Generates a random timezone. :param region: Timezone region. :return: Timezone. """ region_name = self.validate_enum(region, TimezoneRegion) return self.random.choice( [tz for tz in TIMEZONES if tz.startswith(region_name)] )
[docs] def gmt_offset(self) -> str: """Generates a random GMT offset value. :return: GMT Offset. """ return self.random.choice(GMT_OFFSETS)
[docs] def datetime( self, start: int = _CURRENT_YEAR, end: int = _CURRENT_YEAR, timezone: str | None = None, ) -> DateTime: """Generates random datetime. :param start: Minimum value of year. :param end: Maximum value of year. :param timezone: Set custom timezone (pytz required). :return: Datetime """ datetime_obj = datetime.combine( date=self.date(start, end), time=self.time(), ) if timezone: if not pytz: raise ImportError("Timezones are supported only with pytz") tz = pytz.timezone(timezone) datetime_obj = tz.localize(datetime_obj) return datetime_obj
[docs] def formatted_datetime(self, fmt: str = "", **kwargs: t.Any) -> str: """Generates datetime string in human-readable format. :param fmt: Custom format (default is format for current locale) :param kwargs: Keyword arguments for :meth:`~.datetime()` :return: Formatted datetime string. """ dt_obj = self.datetime(**kwargs) if not fmt: date_fmt = self._extract(["formats", "date"]) time_fmt = self._extract(["formats", "time"]) fmt = f"{date_fmt} {time_fmt}" return dt_obj.strftime(fmt)
[docs] def timestamp( self, fmt: TimestampFormat = TimestampFormat.POSIX, **kwargs: t.Any ) -> str | int: """Generates a random timestamp in given format. Supported formats are: - TimestampFormat.POSIX - TimestampFormat.RFC_3339 - TimestampFormat.ISO_8601 Example: >>> from mimesis import Datetime >>> from mimesis.enums import TimestampFormat >>> dt = Datetime() >>> dt.timestamp(fmt=TimestampFormat.POSIX) 1697322442 >>> dt.timestamp(fmt=TimestampFormat.RFC_3339) '2023-12-08T18:46:34' >>> dt.timestamp(fmt=TimestampFormat.ISO_8601) '2009-05-30T21:45:57.328600' :param fmt: Format of timestamp (Default is TimestampFormat.POSIX). :param kwargs: Kwargs for :meth:`~.datetime()`. :return: Timestamp. """ self.validate_enum(fmt, TimestampFormat) stamp = self.datetime(**kwargs) if fmt == TimestampFormat.RFC_3339: return stamp.strftime("%Y-%m-%dT%H:%M:%SZ") elif fmt == TimestampFormat.ISO_8601: return stamp.isoformat() else: return int(stamp.timestamp())