diff options
Diffstat (limited to 'pydis_site/apps/api/models')
| -rw-r--r-- | pydis_site/apps/api/models/bot/infraction.py | 6 | ||||
| -rw-r--r-- | pydis_site/apps/api/models/bot/message.py | 16 | ||||
| -rw-r--r-- | pydis_site/apps/api/models/bot/metricity.py | 28 | ||||
| -rw-r--r-- | pydis_site/apps/api/models/utils.py | 172 | 
4 files changed, 40 insertions, 182 deletions
| diff --git a/pydis_site/apps/api/models/bot/infraction.py b/pydis_site/apps/api/models/bot/infraction.py index c9303024..218ee5ec 100644 --- a/pydis_site/apps/api/models/bot/infraction.py +++ b/pydis_site/apps/api/models/bot/infraction.py @@ -23,6 +23,12 @@ class Infraction(ModelReprMixin, models.Model):          default=timezone.now,          help_text="The date and time of the creation of this infraction."      ) +    last_applied = models.DateTimeField( +        # This default is for backwards compatibility with bot versions +        # that don't explicitly give a value. +        default=timezone.now, +        help_text="The date and time of when this infraction was last applied." +    )      expires_at = models.DateTimeField(          null=True,          help_text=( diff --git a/pydis_site/apps/api/models/bot/message.py b/pydis_site/apps/api/models/bot/message.py index bab3368d..89ae27e4 100644 --- a/pydis_site/apps/api/models/bot/message.py +++ b/pydis_site/apps/api/models/bot/message.py @@ -1,13 +1,11 @@ -from datetime import datetime +import datetime  from django.contrib.postgres import fields as pgfields  from django.core.validators import MinValueValidator  from django.db import models -from django.utils import timezone  from pydis_site.apps.api.models.bot.user import User  from pydis_site.apps.api.models.mixins import ModelReprMixin -from pydis_site.apps.api.models.utils import validate_embed  class Message(ModelReprMixin, models.Model): @@ -48,9 +46,7 @@ class Message(ModelReprMixin, models.Model):          blank=True      )      embeds = pgfields.ArrayField( -        models.JSONField( -            validators=(validate_embed,) -        ), +        models.JSONField(),          blank=True,          help_text="Embeds attached to this message."      ) @@ -63,11 +59,11 @@ class Message(ModelReprMixin, models.Model):      )      @property -    def timestamp(self) -> datetime: +    def timestamp(self) -> datetime.datetime:          """Attribute that represents the message timestamp as derived from the snowflake id.""" -        tz_naive_datetime = datetime.utcfromtimestamp(((self.id >> 22) + 1420070400000) / 1000) -        tz_aware_datetime = timezone.make_aware(tz_naive_datetime, timezone=timezone.utc) -        return tz_aware_datetime +        return datetime.datetime.utcfromtimestamp( +            ((self.id >> 22) + 1420070400000) / 1000 +        ).replace(tzinfo=datetime.timezone.utc)      class Meta:          """Metadata provided for Django's ORM.""" diff --git a/pydis_site/apps/api/models/bot/metricity.py b/pydis_site/apps/api/models/bot/metricity.py index abd25ef0..f53dd33c 100644 --- a/pydis_site/apps/api/models/bot/metricity.py +++ b/pydis_site/apps/api/models/bot/metricity.py @@ -130,3 +130,31 @@ class Metricity:              raise NotFoundError()          return values + +    def total_messages_in_past_n_days( +        self, +        user_ids: list[str], +        days: int +    ) -> list[tuple[str, int]]: +        """ +        Query activity by a list of users in the past `days` days. + +        Returns a list of (user_id, message_count) tuples. +        """ +        self.cursor.execute( +            """ +            SELECT +                author_id, COUNT(*) +            FROM messages +            WHERE +                author_id IN %s +                AND NOT is_deleted +                AND channel_id NOT IN %s +                AND created_at > now() - interval '%s days' +            GROUP BY author_id +            """, +            [tuple(user_ids), EXCLUDE_CHANNELS, days] +        ) +        values = self.cursor.fetchall() + +        return values diff --git a/pydis_site/apps/api/models/utils.py b/pydis_site/apps/api/models/utils.py deleted file mode 100644 index 859394d2..00000000 --- a/pydis_site/apps/api/models/utils.py +++ /dev/null @@ -1,172 +0,0 @@ -from collections.abc import Mapping -from typing import Any, Dict - -from django.core.exceptions import ValidationError -from django.core.validators import MaxLengthValidator, MinLengthValidator - - -def is_bool_validator(value: Any) -> None: -    """Validates if a given value is of type bool.""" -    if not isinstance(value, bool): -        raise ValidationError(f"This field must be of type bool, not {type(value)}.") - - -def validate_embed_fields(fields: dict) -> None: -    """Raises a ValidationError if any of the given embed fields is invalid.""" -    field_validators = { -        'name': (MaxLengthValidator(limit_value=256),), -        'value': (MaxLengthValidator(limit_value=1024),), -        'inline': (is_bool_validator,), -    } - -    required_fields = ('name', 'value') - -    for field in fields: -        if not isinstance(field, Mapping): -            raise ValidationError("Embed fields must be a mapping.") - -        if not all(required_field in field for required_field in required_fields): -            raise ValidationError( -                f"Embed fields must contain the following fields: {', '.join(required_fields)}." -            ) - -        for field_name, value in field.items(): -            if field_name not in field_validators: -                raise ValidationError(f"Unknown embed field field: {field_name!r}.") - -            for validator in field_validators[field_name]: -                validator(value) - - -def validate_embed_footer(footer: Dict[str, str]) -> None: -    """Raises a ValidationError if the given footer is invalid.""" -    field_validators = { -        'text': ( -            MinLengthValidator( -                limit_value=1, -                message="Footer text must not be empty." -            ), -            MaxLengthValidator(limit_value=2048) -        ), -        'icon_url': (), -        'proxy_icon_url': () -    } - -    if not isinstance(footer, Mapping): -        raise ValidationError("Embed footer must be a mapping.") - -    for field_name, value in footer.items(): -        if field_name not in field_validators: -            raise ValidationError(f"Unknown embed footer field: {field_name!r}.") - -        for validator in field_validators[field_name]: -            validator(value) - - -def validate_embed_author(author: Any) -> None: -    """Raises a ValidationError if the given author is invalid.""" -    field_validators = { -        'name': ( -            MinLengthValidator( -                limit_value=1, -                message="Embed author name must not be empty." -            ), -            MaxLengthValidator(limit_value=256) -        ), -        'url': (), -        'icon_url': (), -        'proxy_icon_url': () -    } - -    if not isinstance(author, Mapping): -        raise ValidationError("Embed author must be a mapping.") - -    for field_name, value in author.items(): -        if field_name not in field_validators: -            raise ValidationError(f"Unknown embed author field: {field_name!r}.") - -        for validator in field_validators[field_name]: -            validator(value) - - -def validate_embed(embed: Any) -> None: -    """ -    Validate a JSON document containing an embed as possible to send on Discord. - -    This attempts to rebuild the validation used by Discord -    as well as possible by checking for various embed limits so we can -    ensure that any embed we store here will also be accepted as a -    valid embed by the Discord API. - -    Using this directly is possible, although not intended - you usually -    stick this onto the `validators` keyword argument of model fields. - -    Example: - -        >>> from django.db import models -        >>> from pydis_site.apps.api.models.utils import validate_embed -        >>> class MyMessage(models.Model): -        ...     embed = models.JSONField( -        ...         validators=( -        ...             validate_embed, -        ...         ) -        ...     ) -        ...     # ... -        ... - -    Args: -        embed (Any): -            A dictionary describing the contents of this embed. -            See the official documentation for a full reference -            of accepted keys by this dictionary: -                https://discordapp.com/developers/docs/resources/channel#embed-object - -    Raises: -        ValidationError: -            In case the given embed is deemed invalid, a `ValidationError` -            is raised which in turn will allow Django to display errors -            as appropriate. -    """ -    all_keys = { -        'title', 'type', 'description', 'url', 'timestamp', -        'color', 'footer', 'image', 'thumbnail', 'video', -        'provider', 'author', 'fields' -    } -    one_required_of = {'description', 'fields', 'image', 'title', 'video'} -    field_validators = { -        'title': ( -            MinLengthValidator( -                limit_value=1, -                message="Embed title must not be empty." -            ), -            MaxLengthValidator(limit_value=256) -        ), -        'description': (MaxLengthValidator(limit_value=4096),), -        'fields': ( -            MaxLengthValidator(limit_value=25), -            validate_embed_fields -        ), -        'footer': (validate_embed_footer,), -        'author': (validate_embed_author,) -    } - -    if not embed: -        raise ValidationError("Tag embed must not be empty.") - -    elif not isinstance(embed, Mapping): -        raise ValidationError("Tag embed must be a mapping.") - -    elif not any(field in embed for field in one_required_of): -        raise ValidationError(f"Tag embed must contain one of the fields {one_required_of}.") - -    for required_key in one_required_of: -        if required_key in embed and not embed[required_key]: -            raise ValidationError(f"Key {required_key!r} must not be empty.") - -    for field_name, value in embed.items(): -        if field_name not in all_keys: -            raise ValidationError(f"Unknown field name: {field_name!r}") - -        if field_name in field_validators: -            for validator in field_validators[field_name]: -                validator(value) | 
