diff options
| author | 2023-05-14 23:54:51 +0530 | |
|---|---|---|
| committer | 2023-05-14 23:54:51 +0530 | |
| commit | a241a397f966a4265935dfd5c92a84fdf95c52c8 (patch) | |
| tree | 83f6eb572e26e64e6ca18642013abf60f1b23d8b /pydis_site/apps/api/migrations | |
| parent | Update pydis_site/apps/content/resources/guides/python-guides/subclassing_bot.md (diff) | |
| parent | Merge pull request #972 from python-discord/fix-psycopg3-compatibility-in-met... (diff) | |
Merge branch 'main' into subclassing_bot
Diffstat (limited to 'pydis_site/apps/api/migrations')
16 files changed, 534 insertions, 7 deletions
| diff --git a/pydis_site/apps/api/migrations/0013_specialsnake_image.py b/pydis_site/apps/api/migrations/0013_specialsnake_image.py index a0d0d318..8ba3432f 100644 --- a/pydis_site/apps/api/migrations/0013_specialsnake_image.py +++ b/pydis_site/apps/api/migrations/0013_specialsnake_image.py @@ -2,7 +2,6 @@  import datetime  from django.db import migrations, models -from django.utils.timezone import utc  class Migration(migrations.Migration): @@ -15,7 +14,7 @@ class Migration(migrations.Migration):          migrations.AddField(              model_name='specialsnake',              name='image', -            field=models.URLField(default=datetime.datetime(2018, 10, 23, 11, 51, 23, 703868, tzinfo=utc)), +            field=models.URLField(default=datetime.datetime(2018, 10, 23, 11, 51, 23, 703868, tzinfo=datetime.timezone.utc)),              preserve_default=False,          ),      ] diff --git a/pydis_site/apps/api/migrations/0019_deletedmessage.py b/pydis_site/apps/api/migrations/0019_deletedmessage.py index 6b848d64..25d04434 100644 --- a/pydis_site/apps/api/migrations/0019_deletedmessage.py +++ b/pydis_site/apps/api/migrations/0019_deletedmessage.py @@ -18,7 +18,7 @@ class Migration(migrations.Migration):                  ('id', models.BigIntegerField(help_text='The message ID as taken from Discord.', primary_key=True, serialize=False, validators=[django.core.validators.MinValueValidator(limit_value=0, message='Message IDs cannot be negative.')])),                  ('channel_id', models.BigIntegerField(help_text='The channel ID that this message was sent in, taken from Discord.', validators=[django.core.validators.MinValueValidator(limit_value=0, message='Channel IDs cannot be negative.')])),                  ('content', models.CharField(help_text='The content of this message, taken from Discord.', max_length=2000)), -                ('embeds', django.contrib.postgres.fields.ArrayField(base_field=django.contrib.postgres.fields.jsonb.JSONField(validators=[pydis_site.apps.api.models.utils.validate_embed]), help_text='Embeds attached to this message.', size=None)), +                ('embeds', django.contrib.postgres.fields.ArrayField(base_field=django.contrib.postgres.fields.jsonb.JSONField(validators=[]), help_text='Embeds attached to this message.', size=None)),                  ('author', models.ForeignKey(help_text='The author of this message.', on_delete=django.db.models.deletion.CASCADE, to='api.User')),                  ('deletion_context', models.ForeignKey(help_text='The deletion context this message is part of.', on_delete=django.db.models.deletion.CASCADE, to='api.MessageDeletionContext')),              ], diff --git a/pydis_site/apps/api/migrations/0051_allow_blank_message_embeds.py b/pydis_site/apps/api/migrations/0051_allow_blank_message_embeds.py index 124c6a57..622f21d1 100644 --- a/pydis_site/apps/api/migrations/0051_allow_blank_message_embeds.py +++ b/pydis_site/apps/api/migrations/0051_allow_blank_message_embeds.py @@ -3,7 +3,6 @@  import django.contrib.postgres.fields  import django.contrib.postgres.fields.jsonb  from django.db import migrations -import pydis_site.apps.api.models.utils  class Migration(migrations.Migration): @@ -16,6 +15,6 @@ class Migration(migrations.Migration):          migrations.AlterField(              model_name='deletedmessage',              name='embeds', -            field=django.contrib.postgres.fields.ArrayField(base_field=django.contrib.postgres.fields.jsonb.JSONField(validators=[pydis_site.apps.api.models.utils.validate_embed]), blank=True, help_text='Embeds attached to this message.', size=None), +            field=django.contrib.postgres.fields.ArrayField(base_field=django.contrib.postgres.fields.jsonb.JSONField(validators=[]), blank=True, help_text='Embeds attached to this message.', size=None),          ),      ] diff --git a/pydis_site/apps/api/migrations/0077_use_generic_jsonfield.py b/pydis_site/apps/api/migrations/0077_use_generic_jsonfield.py index 9e8f2fb9..95ef5850 100644 --- a/pydis_site/apps/api/migrations/0077_use_generic_jsonfield.py +++ b/pydis_site/apps/api/migrations/0077_use_generic_jsonfield.py @@ -2,7 +2,6 @@  import django.contrib.postgres.fields  from django.db import migrations, models -import pydis_site.apps.api.models.utils  class Migration(migrations.Migration): @@ -20,6 +19,6 @@ class Migration(migrations.Migration):          migrations.AlterField(              model_name='deletedmessage',              name='embeds', -            field=django.contrib.postgres.fields.ArrayField(base_field=models.JSONField(validators=[pydis_site.apps.api.models.utils.validate_embed]), blank=True, help_text='Embeds attached to this message.', size=None), +            field=django.contrib.postgres.fields.ArrayField(base_field=models.JSONField(validators=[]), blank=True, help_text='Embeds attached to this message.', size=None),          ),      ] diff --git a/pydis_site/apps/api/migrations/0080_add_aoc_tables.py b/pydis_site/apps/api/migrations/0080_add_aoc_tables.py new file mode 100644 index 00000000..2c0c689a --- /dev/null +++ b/pydis_site/apps/api/migrations/0080_add_aoc_tables.py @@ -0,0 +1,32 @@ +# Generated by Django 3.1.14 on 2022-03-06 16:07 + +from django.db import migrations, models +import django.db.models.deletion +import pydis_site.apps.api.models.mixins + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0079_merge_20220125_2022'), +    ] + +    operations = [ +        migrations.CreateModel( +            name='AocAccountLink', +            fields=[ +                ('user', models.OneToOneField(help_text='The user that is blocked from getting the AoC Completionist Role', on_delete=django.db.models.deletion.CASCADE, primary_key=True, serialize=False, to='api.user')), +                ('aoc_username', models.CharField(help_text='The AoC username associated with the Discord User.', max_length=120)), +            ], +            bases=(pydis_site.apps.api.models.mixins.ModelReprMixin, models.Model), +        ), +        migrations.CreateModel( +            name='AocCompletionistBlock', +            fields=[ +                ('user', models.OneToOneField(help_text='The user that is blocked from getting the AoC Completionist Role', on_delete=django.db.models.deletion.CASCADE, primary_key=True, serialize=False, to='api.user')), +                ('is_blocked', models.BooleanField(default=True, help_text='Whether this user is actively being blocked from getting the AoC Completionist Role', verbose_name='Blocked')), +                ('reason', models.TextField(help_text='The reason for the AoC Completionist Role Block.', null=True)), +            ], +            bases=(pydis_site.apps.api.models.mixins.ModelReprMixin, models.Model), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0081_bumpedthread.py b/pydis_site/apps/api/migrations/0081_bumpedthread.py new file mode 100644 index 00000000..03e66cc1 --- /dev/null +++ b/pydis_site/apps/api/migrations/0081_bumpedthread.py @@ -0,0 +1,22 @@ +# Generated by Django 3.1.14 on 2022-02-19 16:26 + +import django.core.validators +from django.db import migrations, models +import pydis_site.apps.api.models.mixins + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0080_add_aoc_tables'), +    ] + +    operations = [ +        migrations.CreateModel( +            name='BumpedThread', +            fields=[ +                ('thread_id', models.BigIntegerField(help_text='The thread ID that should be bumped.', primary_key=True, serialize=False, validators=[django.core.validators.MinValueValidator(limit_value=0, message='Thread IDs cannot be negative.')], verbose_name='Thread ID')), +            ], +            bases=(pydis_site.apps.api.models.mixins.ModelReprMixin, models.Model), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0082_otn_allow_big_solidus.py b/pydis_site/apps/api/migrations/0082_otn_allow_big_solidus.py new file mode 100644 index 00000000..abbb98ec --- /dev/null +++ b/pydis_site/apps/api/migrations/0082_otn_allow_big_solidus.py @@ -0,0 +1,19 @@ +# Generated by Django 3.1.14 on 2022-04-21 23:29 + +import django.core.validators +from django.db import migrations, models + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0081_bumpedthread'), +    ] + +    operations = [ +        migrations.AlterField( +            model_name='offtopicchannelname', +            name='name', +            field=models.CharField(help_text='The actual channel name that will be used on our Discord server.', max_length=96, primary_key=True, serialize=False, validators=[django.core.validators.RegexValidator(regex="^[a-z0-9\\U0001d5a0-\\U0001d5b9-ǃ?’'<>⧹⧸]+$")]), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0083_remove_embed_validation.py b/pydis_site/apps/api/migrations/0083_remove_embed_validation.py new file mode 100644 index 00000000..e835bb66 --- /dev/null +++ b/pydis_site/apps/api/migrations/0083_remove_embed_validation.py @@ -0,0 +1,19 @@ +# Generated by Django 3.1.14 on 2022-06-30 09:41 + +import django.contrib.postgres.fields +from django.db import migrations, models + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0082_otn_allow_big_solidus'), +    ] + +    operations = [ +        migrations.AlterField( +            model_name='deletedmessage', +            name='embeds', +            field=django.contrib.postgres.fields.ArrayField(base_field=models.JSONField(), blank=True, help_text='Embeds attached to this message.', size=None), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0084_infraction_last_applied.py b/pydis_site/apps/api/migrations/0084_infraction_last_applied.py new file mode 100644 index 00000000..7704ddb8 --- /dev/null +++ b/pydis_site/apps/api/migrations/0084_infraction_last_applied.py @@ -0,0 +1,26 @@ +# Generated by Django 4.0.6 on 2022-07-27 20:32 + +import django.utils.timezone +from django.db import migrations, models +from django.apps.registry import Apps + + +def set_last_applied_to_inserted_at(apps: Apps, schema_editor): +    Infractions = apps.get_model("api", "infraction") +    Infractions.objects.all().update(last_applied=models.F("inserted_at")) + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0083_remove_embed_validation'), +    ] + +    operations = [ +        migrations.AddField( +            model_name='infraction', +            name='last_applied', +            field=models.DateTimeField(default=django.utils.timezone.now, help_text='The date and time of when this infraction was last applied.'), +        ), +        migrations.RunPython(set_last_applied_to_inserted_at) +    ] diff --git a/pydis_site/apps/api/migrations/0085_add_thread_id_to_nominations.py b/pydis_site/apps/api/migrations/0085_add_thread_id_to_nominations.py new file mode 100644 index 00000000..56a24cc3 --- /dev/null +++ b/pydis_site/apps/api/migrations/0085_add_thread_id_to_nominations.py @@ -0,0 +1,18 @@ +# Generated by Django 4.1.2 on 2022-11-12 14:52 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0084_infraction_last_applied'), +    ] + +    operations = [ +        migrations.AddField( +            model_name='nomination', +            name='thread_id', +            field=models.BigIntegerField(help_text="The nomination vote's thread id.", null=True), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0086_infraction_jump_url.py b/pydis_site/apps/api/migrations/0086_infraction_jump_url.py new file mode 100644 index 00000000..7ae65751 --- /dev/null +++ b/pydis_site/apps/api/migrations/0086_infraction_jump_url.py @@ -0,0 +1,18 @@ +# Generated by Django 4.1.7 on 2023-03-10 17:25 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0085_add_thread_id_to_nominations'), +    ] + +    operations = [ +        migrations.AddField( +            model_name='infraction', +            name='jump_url', +            field=models.URLField(default=None, help_text='The jump url to message invoking the infraction.', max_length=88, null=True), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0087_alter_mute_to_timeout.py b/pydis_site/apps/api/migrations/0087_alter_mute_to_timeout.py new file mode 100644 index 00000000..8a826ba5 --- /dev/null +++ b/pydis_site/apps/api/migrations/0087_alter_mute_to_timeout.py @@ -0,0 +1,25 @@ +from django.apps.registry import Apps +from django.db import migrations, models + +import pydis_site.apps.api.models + + +def rename_type(apps: Apps, _) -> None: +    infractions: pydis_site.apps.api.models.Infraction = apps.get_model("api", "Infraction") +    infractions.objects.filter(type="mute").update(type="timeout") + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0086_infraction_jump_url'), +    ] + +    operations = [ +        migrations.AlterField( +            model_name='infraction', +            name='type', +            field=models.CharField(choices=[('note', 'Note'), ('warning', 'Warning'), ('watch', 'Watch'), ('timeout', 'Timeout'), ('kick', 'Kick'), ('ban', 'Ban'), ('superstar', 'Superstar'), ('voice_ban', 'Voice Ban'), ('voice_mute', 'Voice Mute')], help_text='The type of the infraction.', max_length=10), +        ), +        migrations.RunPython(rename_type, migrations.RunPython.noop) +    ] diff --git a/pydis_site/apps/api/migrations/0088_new_filter_schema.py b/pydis_site/apps/api/migrations/0088_new_filter_schema.py new file mode 100644 index 00000000..9bc40779 --- /dev/null +++ b/pydis_site/apps/api/migrations/0088_new_filter_schema.py @@ -0,0 +1,171 @@ +"""Modified migration file to migrate existing filters to the new system.""" +from datetime import timedelta + +import django.contrib.postgres.fields +from django.apps.registry import Apps +from django.core.validators import MinValueValidator +from django.db import migrations, models +import django.db.models.deletion +from django.db.backends.base.schema import BaseDatabaseSchemaEditor + +import pydis_site.apps.api.models + +OLD_LIST_NAMES = (('GUILD_INVITE', True), ('GUILD_INVITE', False), ('FILE_FORMAT', True), ('DOMAIN_NAME', False), ('FILTER_TOKEN', False), ('REDIRECT', False)) +change_map = { +    "FILTER_TOKEN": "token", +    "DOMAIN_NAME": "domain", +    "GUILD_INVITE": "invite", +    "FILE_FORMAT": "extension", +    "REDIRECT": "redirect" +} + + +def forward(apps: Apps, schema_editor: BaseDatabaseSchemaEditor) -> None: +    filter_: pydis_site.apps.api.models.Filter = apps.get_model("api", "Filter") +    filter_list: pydis_site.apps.api.models.FilterList = apps.get_model("api", "FilterList") +    filter_list_old = apps.get_model("api", "FilterListOld") + +    for name, type_ in OLD_LIST_NAMES: +        objects = filter_list_old.objects.filter(type=name, allowed=type_) +        if name == "DOMAIN_NAME": +            dm_content = "Your message has been removed because it contained a blocked domain: `{domain}`." +        elif name == "GUILD_INVITE": +            dm_content = "Per Rule 6, your invite link has been removed. " \ +                         "Our server rules can be found here: https://pythondiscord.com/pages/rules" +        else: +            dm_content = "" + +        list_ = filter_list.objects.create( +            name=change_map[name], +            list_type=int(type_), +            guild_pings=(["Moderators"] if name != "FILE_FORMAT" else []), +            filter_dm=True, +            dm_pings=[], +            remove_context=(True if name != "FILTER_TOKEN" else False), +            bypass_roles=["Helpers"], +            enabled=True, +            dm_content=dm_content, +            dm_embed="" if name != "FILE_FORMAT" else "*Defined at runtime.*", +            infraction_type="NONE", +            infraction_reason="", +            infraction_duration=timedelta(seconds=0), +            infraction_channel=0, +            disabled_channels=[], +            disabled_categories=(["CODE JAM"] if name in ("FILE_FORMAT", "GUILD_INVITE") else []), +            enabled_channels=[], +            enabled_categories=[], +            send_alert=(name in ('GUILD_INVITE', 'DOMAIN_NAME', 'FILTER_TOKEN')) +        ) + +        for object_ in objects: +            new_object = filter_.objects.create( +                content=object_.content, +                created_at=object_.created_at, +                updated_at=object_.updated_at, +                filter_list=list_, +                description=object_.comment, +                additional_settings={}, +                guild_pings=None, +                filter_dm=None, +                dm_pings=None, +                remove_context=None, +                bypass_roles=None, +                enabled=None, +                dm_content=None, +                dm_embed=None, +                infraction_type=None, +                infraction_reason=None, +                infraction_duration=None, +                infraction_channel=None, +                disabled_channels=None, +                disabled_categories=None, +                enabled_channels=None, +                enabled_categories=None, +                send_alert=None, +            ) +            new_object.save() + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0087_alter_mute_to_timeout'), +    ] + +    operations = [ +        migrations.RenameModel( +            old_name='FilterList', +            new_name='FilterListOld' +        ), +        migrations.CreateModel( +            name='Filter', +            fields=[ +                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), +                ('created_at', models.DateTimeField(auto_now_add=True)), +                ('updated_at', models.DateTimeField(auto_now=True)), +                ('content', models.TextField(help_text='The definition of this filter.')), +                ('description', models.TextField(help_text='Why this filter has been added.', null=True)), +                ('additional_settings', models.JSONField(help_text='Additional settings which are specific to this filter.', default=dict)), +                ('guild_pings', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text='Who to ping when this filter triggers.', size=None, null=True)), +                ('filter_dm', models.BooleanField(help_text='Whether DMs should be filtered.', null=True)), +                ('dm_pings', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text='Who to ping when this filter triggers on a DM.', size=None, null=True)), +                ('remove_context', models.BooleanField(help_text='Whether this filter should remove the context (such as a message) triggering it.', null=True)), +                ('bypass_roles', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text='Roles and users who can bypass this filter.', size=None, null=True)), +                ('enabled', models.BooleanField(help_text='Whether this filter is currently enabled.', null=True)), +                ('dm_content', models.CharField(help_text='The DM to send to a user triggering this filter.', max_length=1000, null=True, blank=True)), +                ('dm_embed', models.CharField(help_text='The content of the DM embed', max_length=2000, null=True, blank=True)), +                ('infraction_type', models.CharField(choices=[('NONE', 'None'), ('NOTE', 'Note'), ('WARNING', 'Warning'), ('WATCH', 'Watch'), ('TIMEOUT', 'Timeout'), ('KICK', 'Kick'), ('BAN', 'Ban'), ('SUPERSTAR', 'Superstar'), ('VOICE_BAN', 'Voice Ban'), ('VOICE_MUTE', 'Voice Mute')], help_text='The infraction to apply to this user.', max_length=10, null=True)), +                ('infraction_reason', models.CharField(help_text='The reason to give for the infraction.', max_length=1000, null=True, blank=True)), +                ('infraction_duration', models.DurationField(help_text='The duration of the infraction. 0 for permanent.', null=True)), +                ('infraction_channel', models.BigIntegerField(validators=(MinValueValidator(limit_value=0, message="Channel IDs cannot be negative."),), help_text="Channel in which to send the infraction.", null=True)), +                ('disabled_channels', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="Channels in which to not run the filter even if it's enabled in the category.", null=True, size=None)), +                ('disabled_categories', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="Categories in which to not run the filter.", null=True, size=None)), +                ('enabled_channels', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="Channels in which to run the filter even if it's disabled in the category.", null=True, size=None)), +                ('enabled_categories', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="The only categories in which to run the filter.", null=True, size=None)), +                ('send_alert', models.BooleanField(help_text='Whether an alert should be sent.', null=True)), +            ], +        ), +        migrations.CreateModel( +            name='FilterList', +            fields=[ +                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), +                ('created_at', models.DateTimeField(auto_now_add=True)), +                ('updated_at', models.DateTimeField(auto_now=True)), +                ('name', models.CharField(help_text='The unique name of this list.', max_length=50)), +                ('list_type', models.IntegerField(choices=[(1, 'Allow'), (0, 'Deny')], help_text='Whether this list is an allowlist or denylist')), +                ('guild_pings', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text='Who to ping when this filter triggers.', size=None)), +                ('filter_dm', models.BooleanField(help_text='Whether DMs should be filtered.')), +                ('dm_pings', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text='Who to ping when this filter triggers on a DM.', size=None)), +                ('remove_context', models.BooleanField(help_text='Whether this filter should remove the context (such as a message) triggering it.')), +                ('bypass_roles', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text='Roles and users who can bypass this filter.', size=None)), +                ('enabled', models.BooleanField(help_text='Whether this filter is currently enabled.')), +                ('dm_content', models.CharField(help_text='The DM to send to a user triggering this filter.', max_length=1000, blank=True)), +                ('dm_embed', models.CharField(help_text='The content of the DM embed', max_length=2000, blank=True)), +                ('infraction_type', models.CharField(choices=[('NONE', 'None'), ('NOTE', 'Note'), ('WARNING', 'Warning'), ('WATCH', 'Watch'), ('TIMEOUT', 'Timeout'), ('KICK', 'Kick'), ('BAN', 'Ban'), ('SUPERSTAR', 'Superstar'), ('VOICE_BAN', 'Voice Ban'), ('VOICE_MUTE', 'Voice Mute')], help_text='The infraction to apply to this user.', max_length=10)), +                ('infraction_reason', models.CharField(help_text='The reason to give for the infraction.', max_length=1000, blank=True)), +                ('infraction_duration', models.DurationField(help_text='The duration of the infraction. 0 for permanent.')), +                ('infraction_channel', models.BigIntegerField(validators=(MinValueValidator(limit_value=0, message="Channel IDs cannot be negative."),), help_text="Channel in which to send the infraction.")), +                ('disabled_channels', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="Channels in which to not run the filter even if it's enabled in the category.", size=None)), +                ('disabled_categories', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="Categories in which to not run the filter.", size=None)), +                ('enabled_channels', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="Channels in which to run the filter even if it's disabled in the category.", size=None)), +                ('enabled_categories', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=100), help_text="The only categories in which to run the filter.", size=None)), +                ('send_alert', models.BooleanField(help_text='Whether an alert should be sent.')), +            ], +        ), +        migrations.AddField( +            model_name='filter', +            name='filter_list', +            field=models.ForeignKey(help_text='The filter list containing this filter.', on_delete=django.db.models.deletion.CASCADE, related_name='filters', to='api.FilterList'), +        ), +        migrations.AddConstraint( +            model_name='filterlist', +            constraint=models.UniqueConstraint(fields=('name', 'list_type'), name='unique_name_type'), +        ), +        migrations.RunPython( +            code=forward,  # Core of the migration +            reverse_code=lambda *_: None +        ), +        migrations.DeleteModel( +            name='FilterListOld' +        ) +    ] diff --git a/pydis_site/apps/api/migrations/0089_unique_constraint_filters.py b/pydis_site/apps/api/migrations/0089_unique_constraint_filters.py new file mode 100644 index 00000000..cb230a27 --- /dev/null +++ b/pydis_site/apps/api/migrations/0089_unique_constraint_filters.py @@ -0,0 +1,26 @@ +from django.db import migrations, models + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0088_new_filter_schema'), +    ] + +    operations = [ +        migrations.RunSQL( +            "ALTER TABLE api_filter " +            "ADD CONSTRAINT unique_filters UNIQUE NULLS NOT DISTINCT " +            "(content, additional_settings, filter_list_id, dm_content, dm_embed, infraction_type, infraction_reason, infraction_duration, infraction_channel, guild_pings, filter_dm, dm_pings, remove_context, bypass_roles, enabled, send_alert, enabled_channels, disabled_channels, enabled_categories, disabled_categories)", +            reverse_sql="ALTER TABLE api_filter DROP CONSTRAINT unique_filters", +            state_operations=[ +                migrations.AddConstraint( +                    model_name='filter', +                    constraint=models.UniqueConstraint( +                        fields=('content', 'additional_settings', 'filter_list', 'dm_content', 'dm_embed', 'infraction_type', 'infraction_reason', 'infraction_duration', 'infraction_channel', 'guild_pings', 'filter_dm', 'dm_pings', 'remove_context', 'bypass_roles', 'enabled', 'send_alert', 'enabled_channels', 'disabled_channels', 'enabled_categories', 'disabled_categories'), +                        name='unique_filters' +                    ), +                ), +            ], +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0090_unique_filter_list.py b/pydis_site/apps/api/migrations/0090_unique_filter_list.py new file mode 100644 index 00000000..cef2faa3 --- /dev/null +++ b/pydis_site/apps/api/migrations/0090_unique_filter_list.py @@ -0,0 +1,102 @@ +from datetime import timedelta + +from django.apps.registry import Apps +from django.db import migrations + +import pydis_site.apps.api.models.bot.filters + + +def create_unique_list(apps: Apps, _): +    """Create the 'unique' FilterList and its related Filters.""" +    filter_list: pydis_site.apps.api.models.FilterList = apps.get_model("api", "FilterList") +    filter_: pydis_site.apps.api.models.Filter = apps.get_model("api", "Filter") + +    list_ = filter_list.objects.create( +        name="unique", +        list_type=0, +        guild_pings=[], +        filter_dm=True, +        dm_pings=[], +        remove_context=False, +        bypass_roles=[], +        enabled=True, +        dm_content="", +        dm_embed="", +        infraction_type="NONE", +        infraction_reason="", +        infraction_duration=timedelta(seconds=0), +        infraction_channel=0, +        disabled_channels=[], +        disabled_categories=[], +        enabled_channels=[], +        enabled_categories=[], +        send_alert=True +    ) + +    everyone = filter_.objects.create( +        content="everyone", +        filter_list=list_, +        description="", +        remove_context=True, +        bypass_roles=["Helpers"], +        dm_content=( +            "Please don't try to ping `@everyone` or `@here`. Your message has been removed. " +            "If you believe this was a mistake, please let staff know!" +        ), +        disabled_categories=["CODE JAM"] +    ) +    everyone.save() + +    webhook = filter_.objects.create( +        content="webhook", +        filter_list=list_, +        description="", +        remove_context=True, +        dm_content=( +            "Looks like you posted a Discord webhook URL. " +            "Therefore, your message has been removed, and your webhook has been deleted. " +            "You can re-create it if you wish to. " +            "If you believe this was a mistake, please let us know." +        ), +    ) +    webhook.save() + +    rich_embed = filter_.objects.create( +        content="rich_embed", +        filter_list=list_, +        description="", +        guild_pings=["Moderators"], +        dm_pings=["Moderators"] +    ) +    rich_embed.save() + +    discord_token = filter_.objects.create( +        content="discord_token", +        filter_list=list_, +        filter_dm=False, +        remove_context=True, +        dm_content=( +            "I noticed you posted a seemingly valid Discord API " +            "token in your message and have removed your message. " +            "This means that your token has been **compromised**. " +            "Please change your token **immediately** at: " +            "<https://discord.com/developers/applications>\n\n" +            "Feel free to re-post it with the token removed. " +            "If you believe this was a mistake, please let us know!" +        ) +    ) +    discord_token.save() + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0089_unique_constraint_filters'), +    ] + +    operations = [ +        migrations.RunPython( +            code=create_unique_list, +            reverse_code=None +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0091_antispam_filter_list.py b/pydis_site/apps/api/migrations/0091_antispam_filter_list.py new file mode 100644 index 00000000..7c233142 --- /dev/null +++ b/pydis_site/apps/api/migrations/0091_antispam_filter_list.py @@ -0,0 +1,52 @@ +from datetime import timedelta + +from django.apps.registry import Apps +from django.db import migrations + +import pydis_site.apps.api.models.bot.filters + + +def create_antispam_list(apps: Apps, _): +    """Create the 'antispam' FilterList and its related Filters.""" +    filter_list: pydis_site.apps.api.models.FilterList = apps.get_model("api", "FilterList") +    filter_: pydis_site.apps.api.models.Filter = apps.get_model("api", "Filter") + +    list_ = filter_list.objects.create( +        name="antispam", +        list_type=0, +        guild_pings=["Moderators"], +        filter_dm=False, +        dm_pings=[], +        remove_context=True, +        bypass_roles=["Helpers"], +        enabled=True, +        dm_content="", +        dm_embed="", +        infraction_type="TIMEOUT", +        infraction_reason="", +        infraction_duration=timedelta(seconds=600), +        infraction_channel=0, +        disabled_channels=[], +        disabled_categories=["CODE JAM"], +        enabled_channels=[], +        enabled_categories=[], +        send_alert=True +    ) + +    rules = ("duplicates", "attachments", "burst", "chars", "emoji", "links", "mentions", "newlines", "role_mentions") + +    filter_.objects.bulk_create([filter_(content=rule, filter_list=list_) for rule in rules]) + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0090_unique_filter_list'), +    ] + +    operations = [ +        migrations.RunPython( +            code=create_antispam_list, +            reverse_code=None +        ), +    ] | 
