diff options
Diffstat (limited to '')
35 files changed, 303 insertions, 281 deletions
| diff --git a/.coveragerc b/.coveragerc index a49af74e..f5ddf08d 100644 --- a/.coveragerc +++ b/.coveragerc @@ -14,6 +14,7 @@ omit =      */urls.py      pydis_site/wsgi.py      pydis_site/settings.py +    pydis_site/utils/resources.py  [report]  fail_under = 100 diff --git a/.dockerignore b/.dockerignore index 236295ca..61fa291a 100644 --- a/.dockerignore +++ b/.dockerignore @@ -1,7 +1,6 @@  .cache  .coverage  .coveragerc -.git  .github  .gitignore  .gitlab diff --git a/.github/workflows/sentry-release.yml b/.github/workflows/sentry-release.yml new file mode 100644 index 00000000..87c85277 --- /dev/null +++ b/.github/workflows/sentry-release.yml @@ -0,0 +1,23 @@ +name: Create Sentry release + +on: +  push: +    branches: +      - master + +jobs: +  createSentryRelease: +    runs-on: ubuntu-latest +    steps: +      - name: Checkout code +        uses: actions/checkout@master +      - name: Create a Sentry.io release +        uses: tclindner/[email protected] +        env: +          SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} +          SENTRY_ORG: python-discord +          SENTRY_PROJECT: site +        with: +          tagName: ${{ github.sha }} +          environment: production +          releaseNamePrefix: pydis-site@ @@ -19,6 +19,7 @@ pyyaml = "~=5.1"  pyuwsgi = {version = "~=2.0", sys_platform = "!='win32'"}  django-allauth = "~=0.41"  sentry-sdk = "~=0.14" +gitpython = "~=3.1.7"  [dev-packages]  coverage = "~=5.0" diff --git a/Pipfile.lock b/Pipfile.lock index d9348899..65b9c154 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -1,7 +1,7 @@  {      "_meta": {          "hash": { -            "sha256": "9f344f6ff220239656ea524c6b1b6e0db6dfcdbfec50433f7cab3af1eaf4f65f" +            "sha256": "4ecc64deaa82df654479986c9c9569721a66296725e51272608a8294ac562af2"          },          "pipfile-spec": 6,          "requires": { @@ -56,11 +56,11 @@          },          "django": {              "hashes": [ -                "sha256:96fbe04e8ba0df289171e7f6970e0ff8b472bf4f909ed9e0e5beccbac7e1dbbe", -                "sha256:c22b4cd8e388f8219dc121f091e53a8701f9f5bca9aa132b5254263cab516215" +                "sha256:2d14be521c3ae24960e5e83d4575e156a8c479a75c935224b671b1c6e66eddaf", +                "sha256:313d0b8f96685e99327785cc600a5178ca855f8e6f4ed162e671e8c3cf749739"              ],              "index": "pypi", -            "version": "==3.0.9" +            "version": "==3.0.10"          },          "django-allauth": {              "hashes": [ @@ -145,6 +145,22 @@              "index": "pypi",              "version": "==3.11.1"          }, +        "gitdb": { +            "hashes": [ +                "sha256:91f36bfb1ab7949b3b40e23736db18231bf7593edada2ba5c3a174a7b23657ac", +                "sha256:c9e1f2d0db7ddb9a704c2a0217be31214e91a4fe1dea1efad19ae42ba0c285c9" +            ], +            "markers": "python_version >= '3.4'", +            "version": "==4.0.5" +        }, +        "gitpython": { +            "hashes": [ +                "sha256:080bf8e2cf1a2b907634761c2eaefbe83b69930c94c66ad11b65a8252959f912", +                "sha256:1858f4fd089abe92ae465f01d5aaaf55e937eca565fb2c1fce35a51b5f85c910" +            ], +            "index": "pypi", +            "version": "==3.1.8" +        },          "idna": {              "hashes": [                  "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6", @@ -239,39 +255,41 @@          },          "psycopg2-binary": {              "hashes": [ -                "sha256:008da3ab51adc70a5f1cfbbe5db3a22607ab030eb44bcecf517ad11a0c2b3cac", -                "sha256:07cf82c870ec2d2ce94d18e70c13323c89f2f2a2628cbf1feee700630be2519a", -                "sha256:08507efbe532029adee21b8d4c999170a83760d38249936038bd0602327029b5", -                "sha256:107d9be3b614e52a192719c6bf32e8813030020ea1d1215daa86ded9a24d8b04", -                "sha256:17a0ea0b0eabf07035e5e0d520dabc7950aeb15a17c6d36128ba99b2721b25b1", -                "sha256:3286541b9d85a340ee4ed42732d15fc1bb441dc500c97243a768154ab8505bb5", -                "sha256:3939cf75fc89c5e9ed836e228c4a63604dff95ad19aed2bbf71d5d04c15ed5ce", -                "sha256:40abc319f7f26c042a11658bf3dd3b0b3bceccf883ec1c565d5c909a90204434", -                "sha256:51f7823f1b087d2020d8e8c9e6687473d3d239ba9afc162d9b2ab6e80b53f9f9", -                "sha256:6bb2dd006a46a4a4ce95201f836194eb6a1e863f69ee5bab506673e0ca767057", -                "sha256:702f09d8f77dc4794651f650828791af82f7c2efd8c91ae79e3d9fe4bb7d4c98", -                "sha256:7036ccf715925251fac969f4da9ad37e4b7e211b1e920860148a10c0de963522", -                "sha256:7b832d76cc65c092abd9505cc670c4e3421fd136fb6ea5b94efbe4c146572505", -                "sha256:8f74e631b67482d504d7e9cf364071fc5d54c28e79a093ff402d5f8f81e23bfa", -                "sha256:930315ac53dc65cbf52ab6b6d27422611f5fb461d763c531db229c7e1af6c0b3", -                "sha256:96d3038f5bd061401996614f65d27a4ecb62d843eb4f48e212e6d129171a721f", -                "sha256:a20299ee0ea2f9cca494396ac472d6e636745652a64a418b39522c120fd0a0a4", -                "sha256:a34826d6465c2e2bbe9d0605f944f19d2480589f89863ed5f091943be27c9de4", -                "sha256:a69970ee896e21db4c57e398646af9edc71c003bc52a3cc77fb150240fefd266", -                "sha256:b9a8b391c2b0321e0cd7ec6b4cfcc3dd6349347bd1207d48bcb752aa6c553a66", -                "sha256:ba13346ff6d3eb2dca0b6fa0d8a9d999eff3dcd9b55f3a890f12b0b6362b2b38", -                "sha256:bb0608694a91db1e230b4a314e8ed00ad07ed0c518f9a69b83af2717e31291a3", -                "sha256:c8830b7d5f16fd79d39b21e3d94f247219036b29b30c8270314c46bf8b732389", -                "sha256:cac918cd7c4c498a60f5d2a61d4f0a6091c2c9490d81bc805c963444032d0dab", -                "sha256:cc30cb900f42c8a246e2cb76539d9726f407330bc244ca7729c41a44e8d807fb", -                "sha256:ccdc6a87f32b491129ada4b87a43b1895cf2c20fdb7f98ad979647506ffc41b6", -                "sha256:d1a8b01f6a964fec702d6b6dac1f91f2b9f9fe41b310cbb16c7ef1fac82df06d", -                "sha256:e004db88e5a75e5fdab1620fb9f90c9598c2a195a594225ac4ed2a6f1c23e162", -                "sha256:eb2f43ae3037f1ef5e19339c41cf56947021ac892f668765cd65f8ab9814192e", -                "sha256:fa466306fcf6b39b8a61d003123d442b23707d635a5cb05ac4e1b62cc79105cd" -            ], -            "index": "pypi", -            "version": "==2.8.5" +                "sha256:0deac2af1a587ae12836aa07970f5cb91964f05a7c6cdb69d8425ff4c15d4e2c", +                "sha256:0e4dc3d5996760104746e6cfcdb519d9d2cd27c738296525d5867ea695774e67", +                "sha256:11b9c0ebce097180129e422379b824ae21c8f2a6596b159c7659e2e5a00e1aa0", +                "sha256:1fabed9ea2acc4efe4671b92c669a213db744d2af8a9fc5d69a8e9bc14b7a9db", +                "sha256:2dac98e85565d5688e8ab7bdea5446674a83a3945a8f416ad0110018d1501b94", +                "sha256:42ec1035841b389e8cc3692277a0bd81cdfe0b65d575a2c8862cec7a80e62e52", +                "sha256:6a32f3a4cb2f6e1a0b15215f448e8ce2da192fd4ff35084d80d5e39da683e79b", +                "sha256:7312e931b90fe14f925729cde58022f5d034241918a5c4f9797cac62f6b3a9dd", +                "sha256:7d92a09b788cbb1aec325af5fcba9fed7203897bbd9269d5691bb1e3bce29550", +                "sha256:833709a5c66ca52f1d21d41865a637223b368c0ee76ea54ca5bad6f2526c7679", +                "sha256:8cd0fb36c7412996859cb4606a35969dd01f4ea34d9812a141cd920c3b18be77", +                "sha256:950bc22bb56ee6ff142a2cb9ee980b571dd0912b0334aa3fe0fe3788d860bea2", +                "sha256:a0c50db33c32594305b0ef9abc0cb7db13de7621d2cadf8392a1d9b3c437ef77", +                "sha256:a0eb43a07386c3f1f1ebb4dc7aafb13f67188eab896e7397aa1ee95a9c884eb2", +                "sha256:aaa4213c862f0ef00022751161df35804127b78adf4a2755b9f991a507e425fd", +                "sha256:ac0c682111fbf404525dfc0f18a8b5f11be52657d4f96e9fcb75daf4f3984859", +                "sha256:ad20d2eb875aaa1ea6d0f2916949f5c08a19c74d05b16ce6ebf6d24f2c9f75d1", +                "sha256:b4afc542c0ac0db720cf516dd20c0846f71c248d2b3d21013aa0d4ef9c71ca25", +                "sha256:b8a3715b3c4e604bcc94c90a825cd7f5635417453b253499664f784fc4da0152", +                "sha256:ba28584e6bca48c59eecbf7efb1576ca214b47f05194646b081717fa628dfddf", +                "sha256:ba381aec3a5dc29634f20692349d73f2d21f17653bda1decf0b52b11d694541f", +                "sha256:bd1be66dde2b82f80afb9459fc618216753f67109b859a361cf7def5c7968729", +                "sha256:c2507d796fca339c8fb03216364cca68d87e037c1f774977c8fc377627d01c71", +                "sha256:cec7e622ebc545dbb4564e483dd20e4e404da17ae07e06f3e780b2dacd5cee66", +                "sha256:d14b140a4439d816e3b1229a4a525df917d6ea22a0771a2a78332273fd9528a4", +                "sha256:d1b4ab59e02d9008efe10ceabd0b31e79519da6fb67f7d8e8977118832d0f449", +                "sha256:d5227b229005a696cc67676e24c214740efd90b148de5733419ac9aaba3773da", +                "sha256:e1f57aa70d3f7cc6947fd88636a481638263ba04a742b4a37dd25c373e41491a", +                "sha256:e74a55f6bad0e7d3968399deb50f61f4db1926acf4a6d83beaaa7df986f48b1c", +                "sha256:e82aba2188b9ba309fd8e271702bd0d0fc9148ae3150532bbb474f4590039ffb", +                "sha256:ee69dad2c7155756ad114c02db06002f4cded41132cc51378e57aad79cc8e4f4", +                "sha256:f5ab93a2cb2d8338b1674be43b442a7f544a0971da062a5da774ed40587f18f5" +            ], +            "index": "pypi", +            "version": "==2.8.6"          },          "pygments": {              "hashes": [ @@ -286,7 +304,7 @@                  "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1",                  "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"              ], -            "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2, 3.3'", +            "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2'",              "version": "==2.4.7"          },          "python3-openid": { @@ -357,20 +375,28 @@          },          "sentry-sdk": {              "hashes": [ -                "sha256:09cbc253c827a88064c5ed548d24fb4294568bfe9b1816a857fa5a423d4ce762", -                "sha256:1d654ac57be9967dae67545fb759f6e7594de07f487c21a276e6466dd52e83f1" +                "sha256:0af429c221670e602f960fca85ca3f607c85510a91f11e8be8f742a978127f78", +                "sha256:a088a1054673c6a19ea590045c871c38da029ef743b61a07bfee95e9f3c060f7"              ],              "index": "pypi", -            "version": "==0.17.0" +            "version": "==0.17.3"          },          "six": {              "hashes": [                  "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259",                  "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"              ], -            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", +            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2'",              "version": "==1.15.0"          }, +        "smmap": { +            "hashes": [ +                "sha256:54c44c197c819d5ef1991799a7e30b662d1e520f2ac75c9efbeb54a742214cf4", +                "sha256:9c98bbd1f9786d22f14b3d4126894d56befb835ec90cef151af566c7e19b5d24" +            ], +            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", +            "version": "==3.0.4" +        },          "sorl-thumbnail": {              "hashes": [                  "sha256:66771521f3c0ed771e1ce8e1aaf1639ebff18f7f5a40cfd3083da8f0fe6c7c99", @@ -437,11 +463,11 @@          },          "attrs": {              "hashes": [ -                "sha256:0ef97238856430dcf9228e07f316aefc17e8939fc8507e18c6501b761ef1a42a", -                "sha256:2867b7b9f8326499ab5b0e2d12801fa5c98842d2cbd22b35112ae04bf85b4dff" +                "sha256:26b54ddbbb9ee1d34d5d3668dd37d6cf74990ab23c828c2888dccdceee395594", +                "sha256:fce7fc47dfc976152e82d53ff92fa0407700c21acd20886a13777a0d20e655dc"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", -            "version": "==20.1.0" +            "version": "==20.2.0"          },          "bandit": {              "hashes": [ @@ -522,11 +548,11 @@          },          "flake8-annotations": {              "hashes": [ -                "sha256:7816a5d8f65ffdf37b8e21e5b17e0fd1e492aa92638573276de066e889a22b26", -                "sha256:8d18db74a750dd97f40b483cc3ef80d07d03f687525bad8fd83365dcd3bfd414" +                "sha256:09fe1aa3f40cb8fef632a0ab3614050a7584bb884b6134e70cf1fc9eeee642fa", +                "sha256:5bda552f074fd6e34276c7761756fa07d824ffac91ce9c0a8555eb2bc5b92d7a"              ],              "index": "pypi", -            "version": "==2.3.0" +            "version": "==2.4.0"          },          "flake8-bandit": {              "hashes": [ @@ -599,19 +625,19 @@          },          "gitpython": {              "hashes": [ -                "sha256:2db287d71a284e22e5c2846042d0602465c7434d910406990d5b74df4afb0858", -                "sha256:fa3b92da728a457dd75d62bb5f3eb2816d99a7fe6c67398e260637a40e3fafb5" +                "sha256:080bf8e2cf1a2b907634761c2eaefbe83b69930c94c66ad11b65a8252959f912", +                "sha256:1858f4fd089abe92ae465f01d5aaaf55e937eca565fb2c1fce35a51b5f85c910"              ], -            "markers": "python_version >= '3.4'", -            "version": "==3.1.7" +            "index": "pypi", +            "version": "==3.1.8"          },          "identify": {              "hashes": [ -                "sha256:9f5fcf22b665eaece583bd395b103c2769772a0f646ffabb5b1f155901b07de2", -                "sha256:b1aa2e05863dc80242610d46a7b49105e2eafe00ef0c8ff311c1828680760c76" +                "sha256:009f92ba753c467a99f6fd3eb395412cbc34077dd5a64313b62ba04297f2ab8e", +                "sha256:0868312cb7402b48cf44fe3f568259f804ef4e983c143d11bf7a51ca311ebc34"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", -            "version": "==1.4.29" +            "version": "==1.5.0"          },          "importlib-metadata": {              "hashes": [ @@ -638,10 +664,11 @@          },          "pbr": {              "hashes": [ -                "sha256:07f558fece33b05caf857474a366dfcc00562bca13dd8b47b2b3e22d9f9bf55c", -                "sha256:579170e23f8e0c2f24b0de612f71f648eccb79fb1322c814ae6b3c07b5ba23e8" +                "sha256:14bfd98f51c78a3dd22a1ef45cf194ad79eee4a19e8e1a0d5c7f8e81ffe182ea", +                "sha256:5adc0f9fc64319d8df5ca1e4e06eea674c26b80e6f00c530b18ce6a6592ead15"              ], -            "version": "==5.4.5" +            "markers": "python_version >= '2.6'", +            "version": "==5.5.0"          },          "pep8-naming": {              "hashes": [ @@ -669,11 +696,11 @@          },          "pydocstyle": {              "hashes": [ -                "sha256:08374b9d4d2b7164bae50b71bb24eb0d74a56b309029d5d502264092fa7db0c3", -                "sha256:4ca3c7736d36f92bb215dd74ef84ac3d6c146edd795c7afc5154c10f1eb1f65a" +                "sha256:19b86fa8617ed916776a11cd8bc0197e5b9856d5433b777f51a3defe13075325", +                "sha256:aca749e190a01726a4fb472dd4ef23b5c9da7b9205c0a7857c06533de13fd678"              ],              "markers": "python_version >= '3.5'", -            "version": "==5.1.0" +            "version": "==5.1.1"          },          "pyflakes": {              "hashes": [ @@ -705,7 +732,7 @@                  "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259",                  "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"              ], -            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", +            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2'",              "version": "==1.15.0"          },          "smmap": { @@ -725,11 +752,11 @@          },          "stevedore": {              "hashes": [ -                "sha256:38791aa5bed922b0a844513c5f9ed37774b68edc609e5ab8ab8d8fe0ce4315e5", -                "sha256:c8f4f0ebbc394e52ddf49de8bcc3cf8ad2b4425ebac494106bbc5e3661ac7633" +                "sha256:a34086819e2c7a7f86d5635363632829dab8014e5fd7be2454c7cba84ac7514e", +                "sha256:ddc09a744dc224c84ec8e8efcb70595042d21c97c76df60daee64c9ad53bc7ee"              ],              "markers": "python_version >= '3.6'", -            "version": "==3.2.0" +            "version": "==3.2.1"          },          "toml": {              "hashes": [ @@ -1,5 +1,5 @@  # Python Discord: Site -[](https://discord.gg/2B963hn) +[](https://discord.gg/2B963hn)  [](https://dev.azure.com/python-discord/Python%20Discord/_build/latest?definitionId=2&branchName=master)  [](https://dev.azure.com/python-discord/Python%20Discord/_apis/build/status/Site?branchName=master)  [](https://dev.azure.com/python-discord/Python%20Discord/_apis/build/status/Site?branchName=master) diff --git a/docker/Dockerfile b/docker/Dockerfile index aa427947..97cb73d5 100644 --- a/docker/Dockerfile +++ b/docker/Dockerfile @@ -8,6 +8,12 @@ ENV PIP_NO_CACHE_DIR=false \      PIPENV_HIDE_EMOJIS=1 \      PIPENV_NOSPIN=1 +# Install git +RUN apt-get -y update \ +    && apt-get install -y \ +        git \ +    && rm -rf /var/lib/apt/lists/* +  # Create non-root user.  RUN useradd --system --shell /bin/false --uid 1500 pysite diff --git a/pydis_site/apps/api/admin.py b/pydis_site/apps/api/admin.py index 0333fefc..dd1291b8 100644 --- a/pydis_site/apps/api/admin.py +++ b/pydis_site/apps/api/admin.py @@ -14,7 +14,6 @@ from .models import (      OffTopicChannelName,      OffensiveMessage,      Role, -    Tag,      User  ) @@ -64,5 +63,4 @@ admin.site.register(Nomination)  admin.site.register(OffensiveMessage)  admin.site.register(OffTopicChannelName)  admin.site.register(Role) -admin.site.register(Tag)  admin.site.register(User) diff --git a/pydis_site/apps/api/migrations/0008_tag_embed_validator.py b/pydis_site/apps/api/migrations/0008_tag_embed_validator.py index d53ddb90..d92042d2 100644 --- a/pydis_site/apps/api/migrations/0008_tag_embed_validator.py +++ b/pydis_site/apps/api/migrations/0008_tag_embed_validator.py @@ -1,7 +1,5 @@  # Generated by Django 2.1.1 on 2018-09-23 10:07 -import pydis_site.apps.api.models.bot.tag -import django.contrib.postgres.fields.jsonb  from django.db import migrations @@ -12,9 +10,4 @@ class Migration(migrations.Migration):      ]      operations = [ -        migrations.AlterField( -            model_name='tag', -            name='embed', -            field=django.contrib.postgres.fields.jsonb.JSONField(help_text='The actual embed shown by this tag.', validators=[pydis_site.apps.api.models.bot.tag.validate_tag_embed]), -        ),      ] diff --git a/pydis_site/apps/api/migrations/0019_deletedmessage.py b/pydis_site/apps/api/migrations/0019_deletedmessage.py index 33746253..6b848d64 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.bot.tag.validate_tag_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=[pydis_site.apps.api.models.utils.validate_embed]), 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 e617e1c9..124c6a57 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,7 @@  import django.contrib.postgres.fields  import django.contrib.postgres.fields.jsonb  from django.db import migrations -import pydis_site.apps.api.models.bot.tag +import pydis_site.apps.api.models.utils  class Migration(migrations.Migration): @@ -16,6 +16,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.bot.tag.validate_tag_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=[pydis_site.apps.api.models.utils.validate_embed]), blank=True, help_text='Embeds attached to this message.', size=None),          ),      ] diff --git a/pydis_site/apps/api/migrations/0051_delete_tag.py b/pydis_site/apps/api/migrations/0051_delete_tag.py new file mode 100644 index 00000000..bada5788 --- /dev/null +++ b/pydis_site/apps/api/migrations/0051_delete_tag.py @@ -0,0 +1,16 @@ +# Generated by Django 2.2.11 on 2020-04-01 06:15 + +from django.db import migrations + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0050_remove_infractions_active_default_value'), +    ] + +    operations = [ +        migrations.DeleteModel( +            name='Tag', +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0052_offtopicchannelname_used.py b/pydis_site/apps/api/migrations/0052_offtopicchannelname_used.py new file mode 100644 index 00000000..dfdf3835 --- /dev/null +++ b/pydis_site/apps/api/migrations/0052_offtopicchannelname_used.py @@ -0,0 +1,18 @@ +# Generated by Django 2.2.11 on 2020-03-30 10:24 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0051_create_news_setting'), +    ] + +    operations = [ +        migrations.AddField( +            model_name='offtopicchannelname', +            name='used', +            field=models.BooleanField(default=False, help_text='Whether or not this name has already been used during this rotation'), +        ), +    ] diff --git a/pydis_site/apps/api/migrations/0061_merge_20200830_0526.py b/pydis_site/apps/api/migrations/0061_merge_20200830_0526.py new file mode 100644 index 00000000..f0668696 --- /dev/null +++ b/pydis_site/apps/api/migrations/0061_merge_20200830_0526.py @@ -0,0 +1,14 @@ +# Generated by Django 3.0.8 on 2020-08-30 05:26 + +from django.db import migrations + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0060_populate_filterlists_fix'), +        ('api', '0052_offtopicchannelname_used'), +    ] + +    operations = [ +    ] diff --git a/pydis_site/apps/api/migrations/0062_merge_20200901_1459.py b/pydis_site/apps/api/migrations/0062_merge_20200901_1459.py new file mode 100644 index 00000000..d162acf1 --- /dev/null +++ b/pydis_site/apps/api/migrations/0062_merge_20200901_1459.py @@ -0,0 +1,14 @@ +# Generated by Django 3.0.8 on 2020-09-01 14:59 + +from django.db import migrations + + +class Migration(migrations.Migration): + +    dependencies = [ +        ('api', '0051_delete_tag'), +        ('api', '0061_merge_20200830_0526'), +    ] + +    operations = [ +    ] diff --git a/pydis_site/apps/api/models/__init__.py b/pydis_site/apps/api/models/__init__.py index 1d0ab7ea..e3f928e1 100644 --- a/pydis_site/apps/api/models/__init__.py +++ b/pydis_site/apps/api/models/__init__.py @@ -12,7 +12,6 @@ from .bot import (      OffTopicChannelName,      Reminder,      Role, -    Tag,      User  )  from .log_entry import LogEntry diff --git a/pydis_site/apps/api/models/bot/__init__.py b/pydis_site/apps/api/models/bot/__init__.py index efd98184..1673b434 100644 --- a/pydis_site/apps/api/models/bot/__init__.py +++ b/pydis_site/apps/api/models/bot/__init__.py @@ -11,5 +11,4 @@ from .off_topic_channel_name import OffTopicChannelName  from .offensive_message import OffensiveMessage  from .reminder import Reminder  from .role import Role -from .tag import Tag  from .user import User diff --git a/pydis_site/apps/api/models/bot/message.py b/pydis_site/apps/api/models/bot/message.py index 78dcbf1d..f6ae55a5 100644 --- a/pydis_site/apps/api/models/bot/message.py +++ b/pydis_site/apps/api/models/bot/message.py @@ -5,9 +5,9 @@ from django.core.validators import MinValueValidator  from django.db import models  from django.utils import timezone -from pydis_site.apps.api.models.bot.tag import validate_tag_embed  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): @@ -47,7 +47,7 @@ class Message(ModelReprMixin, models.Model):      )      embeds = pgfields.ArrayField(          pgfields.JSONField( -            validators=(validate_tag_embed,) +            validators=(validate_embed,)          ),          blank=True,          help_text="Embeds attached to this message." diff --git a/pydis_site/apps/api/models/bot/off_topic_channel_name.py b/pydis_site/apps/api/models/bot/off_topic_channel_name.py index 20e77b9f..403c7465 100644 --- a/pydis_site/apps/api/models/bot/off_topic_channel_name.py +++ b/pydis_site/apps/api/models/bot/off_topic_channel_name.py @@ -16,6 +16,11 @@ class OffTopicChannelName(ModelReprMixin, models.Model):          help_text="The actual channel name that will be used on our Discord server."      ) +    used = models.BooleanField( +        default=False, +        help_text="Whether or not this name has already been used during this rotation", +    ) +      def __str__(self):          """Returns the current off-topic name, for display purposes."""          return self.name diff --git a/pydis_site/apps/api/models/bot/tag.py b/pydis_site/apps/api/models/utils.py index 5e53582f..107231ba 100644 --- a/pydis_site/apps/api/models/bot/tag.py +++ b/pydis_site/apps/api/models/utils.py @@ -1,12 +1,8 @@  from collections.abc import Mapping  from typing import Any, Dict -from django.contrib.postgres import fields as pgfields  from django.core.exceptions import ValidationError  from django.core.validators import MaxLengthValidator, MinLengthValidator -from django.db import models - -from pydis_site.apps.api.models.mixins import ModelReprMixin  def is_bool_validator(value: Any) -> None: @@ -15,7 +11,7 @@ def is_bool_validator(value: Any) -> None:          raise ValidationError(f"This field must be of type bool, not {type(value)}.") -def validate_tag_embed_fields(fields: dict) -> None: +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),), @@ -42,7 +38,7 @@ def validate_tag_embed_fields(fields: dict) -> None:                  validator(value) -def validate_tag_embed_footer(footer: Dict[str, str]) -> None: +def validate_embed_footer(footer: Dict[str, str]) -> None:      """Raises a ValidationError if the given footer is invalid."""      field_validators = {          'text': ( @@ -67,7 +63,7 @@ def validate_tag_embed_footer(footer: Dict[str, str]) -> None:              validator(value) -def validate_tag_embed_author(author: Any) -> None: +def validate_embed_author(author: Any) -> None:      """Raises a ValidationError if the given author is invalid."""      field_validators = {          'name': ( @@ -93,7 +89,7 @@ def validate_tag_embed_author(author: Any) -> None:              validator(value) -def validate_tag_embed(embed: Any) -> None: +def validate_embed(embed: Any) -> None:      """      Validate a JSON document containing an embed as possible to send on Discord. @@ -109,11 +105,11 @@ def validate_tag_embed(embed: Any) -> None:          >>> from django.contrib.postgres import fields as pgfields          >>> from django.db import models -        >>> from pydis_site.apps.api.models.bot.tag import validate_tag_embed +        >>> from pydis_site.apps.api.models.utils import validate_embed          >>> class MyMessage(models.Model):          ...     embed = pgfields.JSONField(          ...         validators=( -        ...             validate_tag_embed, +        ...             validate_embed,          ...         )          ...     )          ...     # ... @@ -149,10 +145,10 @@ def validate_tag_embed(embed: Any) -> None:          'description': (MaxLengthValidator(limit_value=2048),),          'fields': (              MaxLengthValidator(limit_value=25), -            validate_tag_embed_fields +            validate_embed_fields          ), -        'footer': (validate_tag_embed_footer,), -        'author': (validate_tag_embed_author,) +        'footer': (validate_embed_footer,), +        'author': (validate_embed_author,)      }      if not embed: @@ -175,24 +171,3 @@ def validate_tag_embed(embed: Any) -> None:          if field_name in field_validators:              for validator in field_validators[field_name]:                  validator(value) - - -class Tag(ModelReprMixin, models.Model): -    """A tag providing (hopefully) useful information.""" - -    title = models.CharField( -        max_length=100, -        help_text=( -            "The title of this tag, shown in searches and providing " -            "a quick overview over what this embed contains." -        ), -        primary_key=True -    ) -    embed = pgfields.JSONField( -        help_text="The actual embed shown by this tag.", -        validators=(validate_tag_embed,) -    ) - -    def __str__(self): -        """Returns the title of this tag, for display purposes.""" -        return self.title diff --git a/pydis_site/apps/api/serializers.py b/pydis_site/apps/api/serializers.py index 0915658a..ae57b307 100644 --- a/pydis_site/apps/api/serializers.py +++ b/pydis_site/apps/api/serializers.py @@ -22,7 +22,6 @@ from .models import (      OffensiveMessage,      Reminder,      Role, -    Tag,      User  ) @@ -256,16 +255,6 @@ class RoleSerializer(ModelSerializer):          fields = ('id', 'name', 'colour', 'permissions', 'position') -class TagSerializer(ModelSerializer): -    """A class providing (de-)serialization of `Tag` instances.""" - -    class Meta: -        """Metadata defined for the Django REST Framework.""" - -        model = Tag -        fields = ('title', 'embed') - -  class UserListSerializer(ListSerializer):      """List serializer for User model to handle bulk updates.""" diff --git a/pydis_site/apps/api/tests/test_models.py b/pydis_site/apps/api/tests/test_models.py index e0e347bb..853e6621 100644 --- a/pydis_site/apps/api/tests/test_models.py +++ b/pydis_site/apps/api/tests/test_models.py @@ -14,7 +14,6 @@ from pydis_site.apps.api.models import (      OffensiveMessage,      Reminder,      Role, -    Tag,      User  )  from pydis_site.apps.api.models.mixins import ModelReprMixin @@ -104,10 +103,6 @@ class StringDunderMethodTests(SimpleTestCase):                  ),                  creation=dt.utcnow()              ), -            Tag( -                title='bob', -                embed={'content': "the builder"} -            ),              User(                  id=5,                  name='bob', diff --git a/pydis_site/apps/api/tests/test_off_topic_channel_names.py b/pydis_site/apps/api/tests/test_off_topic_channel_names.py index bd42cd81..3ab8b22d 100644 --- a/pydis_site/apps/api/tests/test_off_topic_channel_names.py +++ b/pydis_site/apps/api/tests/test_off_topic_channel_names.py @@ -10,12 +10,14 @@ class UnauthenticatedTests(APISubdomainTestCase):          self.client.force_authenticate(user=None)      def test_cannot_read_off_topic_channel_name_list(self): +        """Return a 401 response when not authenticated."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(url)          self.assertEqual(response.status_code, 401)      def test_cannot_read_off_topic_channel_name_list_with_random_item_param(self): +        """Return a 401 response when `random_items` provided and not authenticated."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(f'{url}?random_items=no') @@ -24,6 +26,7 @@ class UnauthenticatedTests(APISubdomainTestCase):  class EmptyDatabaseTests(APISubdomainTestCase):      def test_returns_empty_object(self): +        """Return empty list when no names in database."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(url) @@ -31,6 +34,7 @@ class EmptyDatabaseTests(APISubdomainTestCase):          self.assertEqual(response.json(), [])      def test_returns_empty_list_with_get_all_param(self): +        """Return empty list when no names and `random_items` param provided."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(f'{url}?random_items=5') @@ -38,6 +42,7 @@ class EmptyDatabaseTests(APISubdomainTestCase):          self.assertEqual(response.json(), [])      def test_returns_400_for_bad_random_items_param(self): +        """Return error message when passing not integer as `random_items`."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(f'{url}?random_items=totally-a-valid-integer') @@ -47,6 +52,7 @@ class EmptyDatabaseTests(APISubdomainTestCase):          })      def test_returns_400_for_negative_random_items_param(self): +        """Return error message when passing negative int as `random_items`."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(f'{url}?random_items=-5') @@ -59,10 +65,11 @@ class EmptyDatabaseTests(APISubdomainTestCase):  class ListTests(APISubdomainTestCase):      @classmethod      def setUpTestData(cls): -        cls.test_name = OffTopicChannelName.objects.create(name='lemons-lemonade-stand') -        cls.test_name_2 = OffTopicChannelName.objects.create(name='bbq-with-bisk') +        cls.test_name = OffTopicChannelName.objects.create(name='lemons-lemonade-stand', used=False) +        cls.test_name_2 = OffTopicChannelName.objects.create(name='bbq-with-bisk', used=True)      def test_returns_name_in_list(self): +        """Return all off-topic channel names."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(url) @@ -76,11 +83,21 @@ class ListTests(APISubdomainTestCase):          )      def test_returns_single_item_with_random_items_param_set_to_1(self): +        """Return not-used name instead used."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.get(f'{url}?random_items=1')          self.assertEqual(response.status_code, 200)          self.assertEqual(len(response.json()), 1) +        self.assertEqual(response.json(), [self.test_name.name]) + +    def test_running_out_of_names_with_random_parameter(self): +        """Reset names `used` parameter to `False` when running out of names.""" +        url = reverse('bot:offtopicchannelname-list', host='api') +        response = self.client.get(f'{url}?random_items=2') + +        self.assertEqual(response.status_code, 200) +        self.assertEqual(response.json(), [self.test_name.name, self.test_name_2.name])  class CreationTests(APISubdomainTestCase): @@ -93,6 +110,7 @@ class CreationTests(APISubdomainTestCase):          self.assertEqual(response.status_code, 201)      def test_returns_201_for_unicode_chars(self): +        """Accept all valid characters."""          url = reverse('bot:offtopicchannelname-list', host='api')          names = (              '𝖠𝖡𝖢𝖣𝖤𝖥𝖦𝖧𝖨𝖩𝖪𝖫𝖬𝖭𝖮𝖯𝖰𝖱𝖲𝖳𝖴𝖵𝖶𝖷𝖸𝖹', @@ -104,6 +122,7 @@ class CreationTests(APISubdomainTestCase):              self.assertEqual(response.status_code, 201)      def test_returns_400_for_missing_name_param(self): +        """Return error message when name not provided."""          url = reverse('bot:offtopicchannelname-list', host='api')          response = self.client.post(url)          self.assertEqual(response.status_code, 400) @@ -112,6 +131,7 @@ class CreationTests(APISubdomainTestCase):          })      def test_returns_400_for_bad_name_param(self): +        """Return error message when invalid characters provided."""          url = reverse('bot:offtopicchannelname-list', host='api')          invalid_names = (              'space between words', @@ -134,18 +154,21 @@ class DeletionTests(APISubdomainTestCase):          cls.test_name_2 = OffTopicChannelName.objects.create(name='bbq-with-bisk')      def test_deleting_unknown_name_returns_404(self): +        """Return 404 reponse when trying to delete unknown name."""          url = reverse('bot:offtopicchannelname-detail', args=('unknown-name',), host='api')          response = self.client.delete(url)          self.assertEqual(response.status_code, 404)      def test_deleting_known_name_returns_204(self): +        """Return 204 response when deleting was successful."""          url = reverse('bot:offtopicchannelname-detail', args=(self.test_name.name,), host='api')          response = self.client.delete(url)          self.assertEqual(response.status_code, 204)      def test_name_gets_deleted(self): +        """Name gets actually deleted."""          url = reverse('bot:offtopicchannelname-detail', args=(self.test_name_2.name,), host='api')          response = self.client.delete(url) diff --git a/pydis_site/apps/api/tests/test_validators.py b/pydis_site/apps/api/tests/test_validators.py index 241af08c..8bb7b917 100644 --- a/pydis_site/apps/api/tests/test_validators.py +++ b/pydis_site/apps/api/tests/test_validators.py @@ -5,7 +5,7 @@ from django.test import TestCase  from ..models.bot.bot_setting import validate_bot_setting_name  from ..models.bot.offensive_message import future_date_validator -from ..models.bot.tag import validate_tag_embed +from ..models.utils import validate_embed  REQUIRED_KEYS = ( @@ -25,77 +25,77 @@ class BotSettingValidatorTests(TestCase):  class TagEmbedValidatorTests(TestCase):      def test_rejects_non_mapping(self):          with self.assertRaises(ValidationError): -            validate_tag_embed('non-empty non-mapping') +            validate_embed('non-empty non-mapping')      def test_rejects_missing_required_keys(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'unknown': "key"              })      def test_rejects_one_correct_one_incorrect(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'provider': "??",                  'title': ""              })      def test_rejects_empty_required_key(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': ''              })      def test_rejects_list_as_embed(self):          with self.assertRaises(ValidationError): -            validate_tag_embed([]) +            validate_embed([])      def test_rejects_required_keys_and_unknown_keys(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "the duck walked up to the lemonade stand",                  'and': "he said to the man running the stand"              })      def test_rejects_too_long_title(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': 'a' * 257              })      def test_rejects_too_many_fields(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': [{} for _ in range(26)]              })      def test_rejects_too_long_description(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'description': 'd' * 2049              })      def test_allows_valid_embed(self): -        validate_tag_embed({ +        validate_embed({              'title': "My embed",              'description': "look at my embed, my embed is amazing"          })      def test_allows_unvalidated_fields(self): -        validate_tag_embed({ +        validate_embed({              'title': "My embed",              'provider': "what am I??"          })      def test_rejects_fields_as_list_of_non_mappings(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': ['abc']              })      def test_rejects_fields_with_unknown_fields(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': [                      {                          'what': "is this field" @@ -105,7 +105,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_fields_with_too_long_name(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': [                      {                          'name': "a" * 257 @@ -115,7 +115,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_one_correct_one_incorrect_field(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': [                      {                          'name': "Totally valid", @@ -131,7 +131,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_missing_required_field_field(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': [                      {                          'name': "Totally valid", @@ -142,7 +142,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_invalid_inline_field_field(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'fields': [                      {                          'name': "Totally valid", @@ -153,7 +153,7 @@ class TagEmbedValidatorTests(TestCase):              })      def test_allows_valid_fields(self): -        validate_tag_embed({ +        validate_embed({              'fields': [                  {                      'name': "valid", @@ -174,14 +174,14 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_footer_as_non_mapping(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'footer': []              })      def test_rejects_footer_with_unknown_fields(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'footer': {                      'duck': "quack" @@ -190,7 +190,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_footer_with_empty_text(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'footer': {                      'text': "" @@ -198,7 +198,7 @@ class TagEmbedValidatorTests(TestCase):              })      def test_allows_footer_with_proper_values(self): -        validate_tag_embed({ +        validate_embed({              'title': "whatever",              'footer': {                  'text': "django good" @@ -207,14 +207,14 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_author_as_non_mapping(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'author': []              })      def test_rejects_author_with_unknown_field(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'author': {                      'field': "that is unknown" @@ -223,7 +223,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_author_with_empty_name(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'author': {                      'name': "" @@ -232,7 +232,7 @@ class TagEmbedValidatorTests(TestCase):      def test_rejects_author_with_one_correct_one_incorrect(self):          with self.assertRaises(ValidationError): -            validate_tag_embed({ +            validate_embed({                  'title': "whatever",                  'author': {                      # Relies on "dictionary insertion order remembering" (D.I.O.R.) behaviour @@ -242,7 +242,7 @@ class TagEmbedValidatorTests(TestCase):              })      def test_allows_author_with_proper_values(self): -        validate_tag_embed({ +        validate_embed({              'title': "whatever",              'author': {                  'name': "Bob" diff --git a/pydis_site/apps/api/urls.py b/pydis_site/apps/api/urls.py index a4fd5b2e..4dbf93db 100644 --- a/pydis_site/apps/api/urls.py +++ b/pydis_site/apps/api/urls.py @@ -14,7 +14,6 @@ from .viewsets import (      OffensiveMessageViewSet,      ReminderViewSet,      RoleViewSet, -    TagViewSet,      UserViewSet  ) @@ -62,10 +61,6 @@ bot_router.register(      RoleViewSet  )  bot_router.register( -    'tags', -    TagViewSet -) -bot_router.register(      'users',      UserViewSet  ) diff --git a/pydis_site/apps/api/viewsets/__init__.py b/pydis_site/apps/api/viewsets/__init__.py index 8699517e..dfbb880d 100644 --- a/pydis_site/apps/api/viewsets/__init__.py +++ b/pydis_site/apps/api/viewsets/__init__.py @@ -10,7 +10,6 @@ from .bot import (      OffTopicChannelNameViewSet,      ReminderViewSet,      RoleViewSet, -    TagViewSet,      UserViewSet  )  from .log_entry import LogEntryViewSet diff --git a/pydis_site/apps/api/viewsets/bot/__init__.py b/pydis_site/apps/api/viewsets/bot/__init__.py index e64e3988..84b87eab 100644 --- a/pydis_site/apps/api/viewsets/bot/__init__.py +++ b/pydis_site/apps/api/viewsets/bot/__init__.py @@ -9,5 +9,4 @@ from .off_topic_channel_name import OffTopicChannelNameViewSet  from .offensive_message import OffensiveMessageViewSet  from .reminder import ReminderViewSet  from .role import RoleViewSet -from .tag import TagViewSet  from .user import UserViewSet diff --git a/pydis_site/apps/api/viewsets/bot/off_topic_channel_name.py b/pydis_site/apps/api/viewsets/bot/off_topic_channel_name.py index d6da2399..826ad25e 100644 --- a/pydis_site/apps/api/viewsets/bot/off_topic_channel_name.py +++ b/pydis_site/apps/api/viewsets/bot/off_topic_channel_name.py @@ -1,3 +1,4 @@ +from django.db.models import Case, Value, When  from django.db.models.query import QuerySet  from django.http.request import HttpRequest  from django.shortcuts import get_object_or_404 @@ -20,7 +21,9 @@ class OffTopicChannelNameViewSet(DestroyModelMixin, ViewSet):      Return all known off-topic channel names from the database.      If the `random_items` query parameter is given, for example using...          $ curl api.pythondiscord.local:8000/bot/off-topic-channel-names?random_items=5 -    ... then the API will return `5` random items from the database. +    ... then the API will return `5` random items from the database +    that is not used in current rotation. +    When running out of names, API will mark all names to not used and start new rotation.      #### Response format      Return a list of off-topic-channel names: @@ -106,7 +109,27 @@ class OffTopicChannelNameViewSet(DestroyModelMixin, ViewSet):                      'random_items': ["Must be a positive integer."]                  }) -            queryset = self.get_queryset().order_by('?')[:random_count] +            queryset = self.get_queryset().order_by('used', '?')[:random_count] + +            # When any name is used in our listing then this means we reached end of round +            # and we need to reset all other names `used` to False +            if any(offtopic_name.used for offtopic_name in queryset): +                # These names that we just got have to be excluded from updating used to False +                self.get_queryset().update( +                    used=Case( +                        When( +                            name__in=(offtopic_name.name for offtopic_name in queryset), +                            then=Value(True) +                        ), +                        default=Value(False) +                    ) +                ) +            else: +                # Otherwise mark selected names `used` to True +                self.get_queryset().filter( +                    name__in=(offtopic_name.name for offtopic_name in queryset) +                ).update(used=True) +              serialized = self.serializer_class(queryset, many=True)              return Response(serialized.data) diff --git a/pydis_site/apps/api/viewsets/bot/tag.py b/pydis_site/apps/api/viewsets/bot/tag.py deleted file mode 100644 index 7e9ba117..00000000 --- a/pydis_site/apps/api/viewsets/bot/tag.py +++ /dev/null @@ -1,105 +0,0 @@ -from rest_framework.viewsets import ModelViewSet - -from pydis_site.apps.api.models.bot.tag import Tag -from pydis_site.apps.api.serializers import TagSerializer - - -class TagViewSet(ModelViewSet): -    """ -    View providing CRUD operations on tags shown by our bot. - -    ## Routes -    ### GET /bot/tags -    Returns all tags in the database. - -    #### Response format -    >>> [ -    ...     { -    ...         'title': "resources", -    ...         'embed': { -    ...             'content': "Did you really think I'd put something useful here?" -    ...         } -    ...     } -    ... ] - -    #### Status codes -    - 200: returned on success - -    ### GET /bot/tags/<title:str> -    Gets a single tag by its title. - -    #### Response format -    >>> { -    ...     'title': "My awesome tag", -    ...     'embed': { -    ...         'content': "totally not filler words" -    ...     } -    ... } - -    #### Status codes -    - 200: returned on success -    - 404: if a tag with the given `title` could not be found - -    ### POST /bot/tags -    Adds a single tag to the database. - -    #### Request body -    >>> { -    ...     'title': str, -    ...     'embed': dict -    ... } - -    The embed structure is the same as the embed structure that the Discord API -    expects. You can view the documentation for it here: -        https://discordapp.com/developers/docs/resources/channel#embed-object - -    #### Status codes -    - 201: returned on success -    - 400: if one of the given fields is invalid - -    ### PUT /bot/tags/<title:str> -    Update the tag with the given `title`. - -    #### Request body -    >>> { -    ...     'title': str, -    ...     'embed': dict -    ... } - -    The embed structure is the same as the embed structure that the Discord API -    expects. You can view the documentation for it here: -        https://discordapp.com/developers/docs/resources/channel#embed-object - -    #### Status codes -    - 200: returned on success -    - 400: if the request body was invalid, see response body for details -    - 404: if the tag with the given `title` could not be found - -    ### PATCH /bot/tags/<title:str> -    Update the tag with the given `title`. - -    #### Request body -    >>> { -    ...     'title': str, -    ...     'embed': dict -    ... } - -    The embed structure is the same as the embed structure that the Discord API -    expects. You can view the documentation for it here: -        https://discordapp.com/developers/docs/resources/channel#embed-object - -    #### Status codes -    - 200: returned on success -    - 400: if the request body was invalid, see response body for details -    - 404: if the tag with the given `title` could not be found - -    ### DELETE /bot/tags/<title:str> -    Deletes the tag with the given `title`. - -    #### Status codes -    - 204: returned on success -    - 404: if a tag with the given `title` does not exist -    """ - -    serializer_class = TagSerializer -    queryset = Tag.objects.all() diff --git a/pydis_site/apps/home/tests/mock_github_api_response.json b/pydis_site/apps/home/tests/mock_github_api_response.json index 35604a85..10be4f99 100644 --- a/pydis_site/apps/home/tests/mock_github_api_response.json +++ b/pydis_site/apps/home/tests/mock_github_api_response.json @@ -28,7 +28,7 @@      "forks_count": 31    },    { -    "full_name": "python-discord/flake8-annotations", +    "full_name": "python-discord/metricity",      "description": "test",      "stargazers_count": 97,      "language": "Python", diff --git a/pydis_site/apps/home/views/home.py b/pydis_site/apps/home/views/home.py index 20e38ab0..3b5cd5ac 100644 --- a/pydis_site/apps/home/views/home.py +++ b/pydis_site/apps/home/views/home.py @@ -23,7 +23,7 @@ class HomeView(View):          "python-discord/bot",          "python-discord/snekbox",          "python-discord/seasonalbot", -        "python-discord/flake8-annotations", +        "python-discord/metricity",          "python-discord/django-simple-bulma",      ] diff --git a/pydis_site/constants.py b/pydis_site/constants.py new file mode 100644 index 00000000..0b76694a --- /dev/null +++ b/pydis_site/constants.py @@ -0,0 +1,5 @@ +import git + +# Git SHA +repo = git.Repo(search_parent_directories=True) +GIT_SHA = repo.head.object.hexsha diff --git a/pydis_site/context_processors.py b/pydis_site/context_processors.py new file mode 100644 index 00000000..6937a3db --- /dev/null +++ b/pydis_site/context_processors.py @@ -0,0 +1,8 @@ +from django.template import RequestContext + +from pydis_site.constants import GIT_SHA + + +def git_sha_processor(_: RequestContext) -> dict: +    """Expose the git SHA for this repo to all views.""" +    return {'git_sha': GIT_SHA} diff --git a/pydis_site/settings.py b/pydis_site/settings.py index 2c87007c..1f042c1b 100644 --- a/pydis_site/settings.py +++ b/pydis_site/settings.py @@ -20,6 +20,7 @@ import sentry_sdk  from django.contrib.messages import constants as messages  from sentry_sdk.integrations.django import DjangoIntegration +from pydis_site.constants import GIT_SHA  if typing.TYPE_CHECKING:      from django.contrib.auth.models import User @@ -33,7 +34,8 @@ env = environ.Env(  sentry_sdk.init(      dsn=env('SITE_SENTRY_DSN'),      integrations=[DjangoIntegration()], -    send_default_pii=True +    send_default_pii=True, +    release=f"pydis-site@{GIT_SHA}"  )  # Build paths inside the project like this: os.path.join(BASE_DIR, ...) @@ -157,8 +159,8 @@ TEMPLATES = [                  'django.template.context_processors.static',                  'django.contrib.auth.context_processors.auth',                  'django.contrib.messages.context_processors.messages', -                  "sekizai.context_processors.sekizai", +                "pydis_site.context_processors.git_sha_processor"              ],          },      }, diff --git a/pydis_site/templates/base/base.html b/pydis_site/templates/base/base.html index 4c70d778..70426dc1 100644 --- a/pydis_site/templates/base/base.html +++ b/pydis_site/templates/base/base.html @@ -37,6 +37,7 @@    {% render_block "css" %}  </head>  <body class="site"> +  <!-- Git hash for this release: {{ git_sha }} -->  <main class="site-content">    {% if messages %} | 
