diff options
| author | 2021-03-06 16:27:26 -0500 | |
|---|---|---|
| committer | 2021-03-06 16:27:26 -0500 | |
| commit | b713a6cdd6f5c5d4c9c9a6672d9bf27998595442 (patch) | |
| tree | 1c2900fd84797e1a53c0e04ff386d9deafaf167f | |
| parent | Made multiline concatenated string conform to a certain style. (diff) | |
| parent | Remove trailing whitespace (diff) | |
Merge branch 'master' into swfarnsworth/infraction_message
| -rw-r--r-- | Pipfile | 3 | ||||
| -rw-r--r-- | Pipfile.lock | 464 | ||||
| -rw-r--r-- | bot/__main__.py | 26 | ||||
| -rw-r--r-- | bot/bot.py | 34 | ||||
| -rw-r--r-- | bot/constants.py | 15 | ||||
| -rw-r--r-- | bot/converters.py | 42 | ||||
| -rw-r--r-- | bot/exts/backend/error_handler.py | 2 | ||||
| -rw-r--r-- | bot/exts/filters/filtering.py | 66 | ||||
| -rw-r--r-- | bot/exts/fun/off_topic_names.py | 18 | ||||
| -rw-r--r-- | bot/exts/info/information.py | 15 | ||||
| -rw-r--r-- | bot/exts/info/tags.py | 7 | ||||
| -rw-r--r-- | bot/exts/moderation/defcon.py | 315 | ||||
| -rw-r--r-- | bot/exts/moderation/infraction/infractions.py | 4 | ||||
| -rw-r--r-- | bot/exts/moderation/infraction/superstarify.py | 4 | ||||
| -rw-r--r-- | bot/exts/moderation/slowmode.py | 4 | ||||
| -rw-r--r-- | bot/exts/utils/internal.py | 4 | ||||
| -rw-r--r-- | bot/log.py | 49 | ||||
| -rw-r--r-- | bot/resources/tags/comparison.md | 12 | ||||
| -rw-r--r-- | bot/resources/tags/empty-json.md | 11 | ||||
| -rw-r--r-- | bot/resources/tags/inline.md | 15 | ||||
| -rw-r--r-- | bot/resources/tags/off-topic.md | 2 | ||||
| -rw-r--r-- | bot/resources/tags/pep8.md | 6 | ||||
| -rw-r--r-- | bot/utils/time.py | 42 | ||||
| -rw-r--r-- | config-default.yml | 15 | 
24 files changed, 723 insertions, 452 deletions
| @@ -6,7 +6,7 @@ name = "pypi"  [packages]  aio-pika = "~=6.1"  aiodns = "~=2.0" -aiohttp = "~=3.5" +aiohttp = "~=3.7"  aioping = "~=0.3.1"  aioredis = "~=1.3.1"  "async-rediscache[fakeredis]" = "~=0.1.2" @@ -28,6 +28,7 @@ sphinx = "~=2.2"  statsd = "~=3.3"  arrow = "~=0.17"  emoji = "~=0.6" +python-json-logger = "~=2.0"  [dev-packages]  coverage = "~=5.0" diff --git a/Pipfile.lock b/Pipfile.lock index 636d07b1a..dc7f6f21f 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -1,7 +1,7 @@  {      "_meta": {          "hash": { -            "sha256": "26c8089f17d6d6bac11dbed366b1b46818b4546f243af756a106a32af5d9d8f6" +            "sha256": "81ca9d1891e71de1c3f71958f082e1a8cad71e5b3ca425dc561d0ae74664fdb0"          },          "pipfile-spec": 6,          "requires": { @@ -18,11 +18,11 @@      "default": {          "aio-pika": {              "hashes": [ -                "sha256:9773440a89840941ac3099a7720bf9d51e8764a484066b82ede4d395660ff430", -                "sha256:a8065be3c722eb8f9fff8c0e7590729e7782202cdb9363d9830d7d5d47b45c7c" +                "sha256:1d4305a5f78af3857310b4fe48348cdcf6c097e0e275ea88c2cd08570531a369", +                "sha256:e69afef8695f47c5d107bbdba21bdb845d5c249acb3be53ef5c2d497b02657c0"              ],              "index": "pypi", -            "version": "==6.7.1" +            "version": "==6.8.0"          },          "aiodns": {              "hashes": [ @@ -34,46 +34,46 @@          },          "aiohttp": {              "hashes": [ -                "sha256:0b795072bb1bf87b8620120a6373a3c61bfcb8da7e5c2377f4bb23ff4f0b62c9", -                "sha256:0d438c8ca703b1b714e82ed5b7a4412c82577040dadff479c08405e2a715564f", -                "sha256:16a3cb5df5c56f696234ea9e65e227d1ebe9c18aa774d36ff42f532139066a5f", -                "sha256:1edfd82a98c5161497bbb111b2b70c0813102ad7e0aa81cbeb34e64c93863005", -                "sha256:2406dc1dda01c7f6060ab586e4601f18affb7a6b965c50a8c90ff07569cf782a", -                "sha256:2858b2504c8697beb9357be01dc47ef86438cc1cb36ecb6991796d19475faa3e", -                "sha256:2a7b7640167ab536c3cb90cfc3977c7094f1c5890d7eeede8b273c175c3910fd", -                "sha256:3228b7a51e3ed533f5472f54f70fd0b0a64c48dc1649a0f0e809bec312934d7a", -                "sha256:328b552513d4f95b0a2eea4c8573e112866107227661834652a8984766aa7656", -                "sha256:39f4b0a6ae22a1c567cb0630c30dd082481f95c13ca528dc501a7766b9c718c0", -                "sha256:3b0036c978cbcc4a4512278e98e3e6d9e6b834dc973206162eddf98b586ef1c6", -                "sha256:3ea8c252d8df5e9166bcf3d9edced2af132f4ead8ac422eac723c5781063709a", -                "sha256:41608c0acbe0899c852281978492f9ce2c6fbfaf60aff0cefc54a7c4516b822c", -                "sha256:59d11674964b74a81b149d4ceaff2b674b3b0e4d0f10f0be1533e49c4a28408b", -                "sha256:5e479df4b2d0f8f02133b7e4430098699450e1b2a826438af6bec9a400530957", -                "sha256:684850fb1e3e55c9220aad007f8386d8e3e477c4ec9211ae54d968ecdca8c6f9", -                "sha256:6ccc43d68b81c424e46192a778f97da94ee0630337c9bbe5b2ecc9b0c1c59001", -                "sha256:6d42debaf55450643146fabe4b6817bb2a55b23698b0434107e892a43117285e", -                "sha256:710376bf67d8ff4500a31d0c207b8941ff4fba5de6890a701d71680474fe2a60", -                "sha256:756ae7efddd68d4ea7d89c636b703e14a0c686688d42f588b90778a3c2fc0564", -                "sha256:77149002d9386fae303a4a162e6bce75cc2161347ad2ba06c2f0182561875d45", -                "sha256:78e2f18a82b88cbc37d22365cf8d2b879a492faedb3f2975adb4ed8dfe994d3a", -                "sha256:7d9b42127a6c0bdcc25c3dcf252bb3ddc70454fac593b1b6933ae091396deb13", -                "sha256:8389d6044ee4e2037dca83e3f6994738550f6ee8cfb746762283fad9b932868f", -                "sha256:9c1a81af067e72261c9cbe33ea792893e83bc6aa987bfbd6fdc1e5e7b22777c4", -                "sha256:c1e0920909d916d3375c7a1fdb0b1c78e46170e8bb42792312b6eb6676b2f87f", -                "sha256:c68fdf21c6f3573ae19c7ee65f9ff185649a060c9a06535e9c3a0ee0bbac9235", -                "sha256:c733ef3bdcfe52a1a75564389bad4064352274036e7e234730526d155f04d914", -                "sha256:c9c58b0b84055d8bc27b7df5a9d141df4ee6ff59821f922dd73155861282f6a3", -                "sha256:d03abec50df423b026a5aa09656bd9d37f1e6a49271f123f31f9b8aed5dc3ea3", -                "sha256:d2cfac21e31e841d60dc28c0ec7d4ec47a35c608cb8906435d47ef83ffb22150", -                "sha256:dcc119db14757b0c7bce64042158307b9b1c76471e655751a61b57f5a0e4d78e", -                "sha256:df3a7b258cc230a65245167a202dd07320a5af05f3d41da1488ba0fa05bc9347", -                "sha256:df48a623c58180874d7407b4d9ec06a19b84ed47f60a3884345b1a5099c1818b", -                "sha256:e1b95972a0ae3f248a899cdbac92ba2e01d731225f566569311043ce2226f5e7", -                "sha256:f326b3c1bbfda5b9308252ee0dcb30b612ee92b0e105d4abec70335fab5b1245", -                "sha256:f411cb22115cb15452d099fec0ee636b06cf81bfb40ed9c02d30c8dc2bc2e3d1" +                "sha256:119feb2bd551e58d83d1b38bfa4cb921af8ddedec9fad7183132db334c3133e0", +                "sha256:16d0683ef8a6d803207f02b899c928223eb219111bd52420ef3d7a8aa76227b6", +                "sha256:2eb3efe243e0f4ecbb654b08444ae6ffab37ac0ef8f69d3a2ffb958905379daf", +                "sha256:2ffea7904e70350da429568113ae422c88d2234ae776519549513c8f217f58a9", +                "sha256:40bd1b101b71a18a528ffce812cc14ff77d4a2a1272dfb8b11b200967489ef3e", +                "sha256:418597633b5cd9639e514b1d748f358832c08cd5d9ef0870026535bd5eaefdd0", +                "sha256:481d4b96969fbfdcc3ff35eea5305d8565a8300410d3d269ccac69e7256b1329", +                "sha256:4c1bdbfdd231a20eee3e56bd0ac1cd88c4ff41b64ab679ed65b75c9c74b6c5c2", +                "sha256:5563ad7fde451b1986d42b9bb9140e2599ecf4f8e42241f6da0d3d624b776f40", +                "sha256:58c62152c4c8731a3152e7e650b29ace18304d086cb5552d317a54ff2749d32a", +                "sha256:5b50e0b9460100fe05d7472264d1975f21ac007b35dcd6fd50279b72925a27f4", +                "sha256:5d84ecc73141d0a0d61ece0742bb7ff5751b0657dab8405f899d3ceb104cc7de", +                "sha256:5dde6d24bacac480be03f4f864e9a67faac5032e28841b00533cd168ab39cad9", +                "sha256:5e91e927003d1ed9283dee9abcb989334fc8e72cf89ebe94dc3e07e3ff0b11e9", +                "sha256:62bc216eafac3204877241569209d9ba6226185aa6d561c19159f2e1cbb6abfb", +                "sha256:6c8200abc9dc5f27203986100579fc19ccad7a832c07d2bc151ce4ff17190076", +                "sha256:6ca56bdfaf825f4439e9e3673775e1032d8b6ea63b8953d3812c71bd6a8b81de", +                "sha256:71680321a8a7176a58dfbc230789790639db78dad61a6e120b39f314f43f1907", +                "sha256:7c7820099e8b3171e54e7eedc33e9450afe7cd08172632d32128bd527f8cb77d", +                "sha256:7dbd087ff2f4046b9b37ba28ed73f15fd0bc9f4fdc8ef6781913da7f808d9536", +                "sha256:822bd4fd21abaa7b28d65fc9871ecabaddc42767884a626317ef5b75c20e8a2d", +                "sha256:8ec1a38074f68d66ccb467ed9a673a726bb397142c273f90d4ba954666e87d54", +                "sha256:950b7ef08b2afdab2488ee2edaff92a03ca500a48f1e1aaa5900e73d6cf992bc", +                "sha256:99c5a5bf7135607959441b7d720d96c8e5c46a1f96e9d6d4c9498be8d5f24212", +                "sha256:b84ad94868e1e6a5e30d30ec419956042815dfaea1b1df1cef623e4564c374d9", +                "sha256:bc3d14bf71a3fb94e5acf5bbf67331ab335467129af6416a437bd6024e4f743d", +                "sha256:c2a80fd9a8d7e41b4e38ea9fe149deed0d6aaede255c497e66b8213274d6d61b", +                "sha256:c44d3c82a933c6cbc21039326767e778eface44fca55c65719921c4b9661a3f7", +                "sha256:cc31e906be1cc121ee201adbdf844522ea3349600dd0a40366611ca18cd40e81", +                "sha256:d5d102e945ecca93bcd9801a7bb2fa703e37ad188a2f81b1e65e4abe4b51b00c", +                "sha256:dd7936f2a6daa861143e376b3a1fb56e9b802f4980923594edd9ca5670974895", +                "sha256:dee68ec462ff10c1d836c0ea2642116aba6151c6880b688e56b4c0246770f297", +                "sha256:e76e78863a4eaec3aee5722d85d04dcbd9844bc6cd3bfa6aa880ff46ad16bfcb", +                "sha256:eab51036cac2da8a50d7ff0ea30be47750547c9aa1aa2cf1a1b710a1827e7dbe", +                "sha256:f4496d8d04da2e98cc9133e238ccebf6a13ef39a93da2e87146c8c8ac9768242", +                "sha256:fbd3b5e18d34683decc00d9a360179ac1e7a320a5fee10ab8053ffd6deab76e0", +                "sha256:feb24ff1226beeb056e247cf2e24bba5232519efb5645121c4aea5b6ad74c1f2"              ],              "index": "pypi", -            "version": "==3.7.3" +            "version": "==3.7.4"          },          "aioping": {              "hashes": [ @@ -168,44 +168,45 @@          },          "cffi": {              "hashes": [ -                "sha256:00a1ba5e2e95684448de9b89888ccd02c98d512064b4cb987d48f4b40aa0421e", -                "sha256:00e28066507bfc3fe865a31f325c8391a1ac2916219340f87dfad602c3e48e5d", -                "sha256:045d792900a75e8b1e1b0ab6787dd733a8190ffcf80e8c8ceb2fb10a29ff238a", -                "sha256:0638c3ae1a0edfb77c6765d487fee624d2b1ee1bdfeffc1f0b58c64d149e7eec", -                "sha256:105abaf8a6075dc96c1fe5ae7aae073f4696f2905fde6aeada4c9d2926752362", -                "sha256:155136b51fd733fa94e1c2ea5211dcd4c8879869008fc811648f16541bf99668", -                "sha256:1a465cbe98a7fd391d47dce4b8f7e5b921e6cd805ef421d04f5f66ba8f06086c", -                "sha256:1d2c4994f515e5b485fd6d3a73d05526aa0fcf248eb135996b088d25dfa1865b", -                "sha256:2c24d61263f511551f740d1a065eb0212db1dbbbbd241db758f5244281590c06", -                "sha256:51a8b381b16ddd370178a65360ebe15fbc1c71cf6f584613a7ea08bfad946698", -                "sha256:594234691ac0e9b770aee9fcdb8fa02c22e43e5c619456efd0d6c2bf276f3eb2", -                "sha256:5cf4be6c304ad0b6602f5c4e90e2f59b47653ac1ed9c662ed379fe48a8f26b0c", -                "sha256:64081b3f8f6f3c3de6191ec89d7dc6c86a8a43911f7ecb422c60e90c70be41c7", -                "sha256:6bc25fc545a6b3d57b5f8618e59fc13d3a3a68431e8ca5fd4c13241cd70d0009", -                "sha256:798caa2a2384b1cbe8a2a139d80734c9db54f9cc155c99d7cc92441a23871c03", -                "sha256:7c6b1dece89874d9541fc974917b631406233ea0440d0bdfbb8e03bf39a49b3b", -                "sha256:840793c68105fe031f34d6a086eaea153a0cd5c491cde82a74b420edd0a2b909", -                "sha256:8d6603078baf4e11edc4168a514c5ce5b3ba6e3e9c374298cb88437957960a53", -                "sha256:9cc46bc107224ff5b6d04369e7c595acb700c3613ad7bcf2e2012f62ece80c35", -                "sha256:9f7a31251289b2ab6d4012f6e83e58bc3b96bd151f5b5262467f4bb6b34a7c26", -                "sha256:9ffb888f19d54a4d4dfd4b3f29bc2c16aa4972f1c2ab9c4ab09b8ab8685b9c2b", -                "sha256:a5ed8c05548b54b998b9498753fb9cadbfd92ee88e884641377d8a8b291bcc01", -                "sha256:a7711edca4dcef1a75257b50a2fbfe92a65187c47dab5a0f1b9b332c5919a3fb", -                "sha256:af5c59122a011049aad5dd87424b8e65a80e4a6477419c0c1015f73fb5ea0293", -                "sha256:b18e0a9ef57d2b41f5c68beefa32317d286c3d6ac0484efd10d6e07491bb95dd", -                "sha256:b4e248d1087abf9f4c10f3c398896c87ce82a9856494a7155823eb45a892395d", -                "sha256:ba4e9e0ae13fc41c6b23299545e5ef73055213e466bd107953e4a013a5ddd7e3", -                "sha256:c6332685306b6417a91b1ff9fae889b3ba65c2292d64bd9245c093b1b284809d", -                "sha256:d5ff0621c88ce83a28a10d2ce719b2ee85635e85c515f12bac99a95306da4b2e", -                "sha256:d9efd8b7a3ef378dd61a1e77367f1924375befc2eba06168b6ebfa903a5e59ca", -                "sha256:df5169c4396adc04f9b0a05f13c074df878b6052430e03f50e68adf3a57aa28d", -                "sha256:ebb253464a5d0482b191274f1c8bf00e33f7e0b9c66405fbffc61ed2c839c775", -                "sha256:ec80dc47f54e6e9a78181ce05feb71a0353854cc26999db963695f950b5fb375", -                "sha256:f032b34669220030f905152045dfa27741ce1a6db3324a5bc0b96b6c7420c87b", -                "sha256:f60567825f791c6f8a592f3c6e3bd93dd2934e3f9dac189308426bd76b00ef3b", -                "sha256:f803eaa94c2fcda012c047e62bc7a51b0bdabda1cad7a92a522694ea2d76e49f" -            ], -            "version": "==1.14.4" +                "sha256:005a36f41773e148deac64b08f233873a4d0c18b053d37da83f6af4d9087b813", +                "sha256:0857f0ae312d855239a55c81ef453ee8fd24136eaba8e87a2eceba644c0d4c06", +                "sha256:1071534bbbf8cbb31b498d5d9db0f274f2f7a865adca4ae429e147ba40f73dea", +                "sha256:158d0d15119b4b7ff6b926536763dc0714313aa59e320ddf787502c70c4d4bee", +                "sha256:1f436816fc868b098b0d63b8920de7d208c90a67212546d02f84fe78a9c26396", +                "sha256:2894f2df484ff56d717bead0a5c2abb6b9d2bf26d6960c4604d5c48bbc30ee73", +                "sha256:29314480e958fd8aab22e4a58b355b629c59bf5f2ac2492b61e3dc06d8c7a315", +                "sha256:34eff4b97f3d982fb93e2831e6750127d1355a923ebaeeb565407b3d2f8d41a1", +                "sha256:35f27e6eb43380fa080dccf676dece30bef72e4a67617ffda586641cd4508d49", +                "sha256:3d3dd4c9e559eb172ecf00a2a7517e97d1e96de2a5e610bd9b68cea3925b4892", +                "sha256:43e0b9d9e2c9e5d152946b9c5fe062c151614b262fda2e7b201204de0b99e482", +                "sha256:48e1c69bbacfc3d932221851b39d49e81567a4d4aac3b21258d9c24578280058", +                "sha256:51182f8927c5af975fece87b1b369f722c570fe169f9880764b1ee3bca8347b5", +                "sha256:58e3f59d583d413809d60779492342801d6e82fefb89c86a38e040c16883be53", +                "sha256:5de7970188bb46b7bf9858eb6890aad302577a5f6f75091fd7cdd3ef13ef3045", +                "sha256:65fa59693c62cf06e45ddbb822165394a288edce9e276647f0046e1ec26920f3", +                "sha256:69e395c24fc60aad6bb4fa7e583698ea6cc684648e1ffb7fe85e3c1ca131a7d5", +                "sha256:6c97d7350133666fbb5cf4abdc1178c812cb205dc6f41d174a7b0f18fb93337e", +                "sha256:6e4714cc64f474e4d6e37cfff31a814b509a35cb17de4fb1999907575684479c", +                "sha256:72d8d3ef52c208ee1c7b2e341f7d71c6fd3157138abf1a95166e6165dd5d4369", +                "sha256:8ae6299f6c68de06f136f1f9e69458eae58f1dacf10af5c17353eae03aa0d827", +                "sha256:8b198cec6c72df5289c05b05b8b0969819783f9418e0409865dac47288d2a053", +                "sha256:99cd03ae7988a93dd00bcd9d0b75e1f6c426063d6f03d2f90b89e29b25b82dfa", +                "sha256:9cf8022fb8d07a97c178b02327b284521c7708d7c71a9c9c355c178ac4bbd3d4", +                "sha256:9de2e279153a443c656f2defd67769e6d1e4163952b3c622dcea5b08a6405322", +                "sha256:9e93e79c2551ff263400e1e4be085a1210e12073a31c2011dbbda14bda0c6132", +                "sha256:9ff227395193126d82e60319a673a037d5de84633f11279e336f9c0f189ecc62", +                "sha256:a465da611f6fa124963b91bf432d960a555563efe4ed1cc403ba5077b15370aa", +                "sha256:ad17025d226ee5beec591b52800c11680fca3df50b8b29fe51d882576e039ee0", +                "sha256:afb29c1ba2e5a3736f1c301d9d0abe3ec8b86957d04ddfa9d7a6a42b9367e396", +                "sha256:b85eb46a81787c50650f2392b9b4ef23e1f126313b9e0e9013b35c15e4288e2e", +                "sha256:bb89f306e5da99f4d922728ddcd6f7fcebb3241fc40edebcb7284d7514741991", +                "sha256:cbde590d4faaa07c72bf979734738f328d239913ba3e043b1e98fe9a39f8b2b6", +                "sha256:cd2868886d547469123fadc46eac7ea5253ea7fcb139f12e1dfc2bbd406427d1", +                "sha256:d42b11d692e11b6634f7613ad8df5d6d5f8875f5d48939520d351007b3c13406", +                "sha256:f2d45f97ab6bb54753eab54fffe75aaf3de4ff2341c9daee1987ee1837636f1d", +                "sha256:fd78e5fee591709f32ef6edb9a015b4aa1a5022598e36227500c8f4e02328d9c" +            ], +            "version": "==1.14.5"          },          "chardet": {              "hashes": [ @@ -363,11 +364,11 @@          },          "jinja2": {              "hashes": [ -                "sha256:89aab215427ef59c34ad58735269eb58b1a5808103067f7bb9d5836c651b3bb0", -                "sha256:f0a4641d3cf955324a89c04f3d94663aa4d638abe8f733ecd3582848e1c37035" +                "sha256:03e47ad063331dd6a3f04a43eddca8a966a26ba0c5b7207a9a9e4e08f1b29419", +                "sha256:a6d58433de0ae800347cab1fa3043cebbabe8baa9d29e668f1c768cb87a333c6"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", -            "version": "==2.11.2" +            "version": "==2.11.3"          },          "lxml": {              "hashes": [ @@ -427,8 +428,12 @@                  "sha256:09c4b7f37d6c648cb13f9230d847adf22f8171b1ccc4d5682398e77f40309235",                  "sha256:1027c282dad077d0bae18be6794e6b6b8c91d58ed8a8d89a89d59693b9131db5",                  "sha256:13d3144e1e340870b25e7b10b98d779608c02016d5184cfb9927a9f10c689f42", +                "sha256:195d7d2c4fbb0ee8139a6cf67194f3973a6b3042d742ebe0a9ed36d8b6f0c07f", +                "sha256:22c178a091fc6630d0d045bdb5992d2dfe14e3259760e713c490da5323866c39",                  "sha256:24982cc2533820871eba85ba648cd53d8623687ff11cbb805be4ff7b4c971aff",                  "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b", +                "sha256:2beec1e0de6924ea551859edb9e7679da6e4870d32cb766240ce17e0a0ba2014", +                "sha256:3b8a6499709d29c2e2399569d96719a1b21dcd94410a586a18526b143ec8470f",                  "sha256:43a55c2930bbc139570ac2452adf3d70cdbb3cfe5912c71cdce1c2c6bbd9c5d1",                  "sha256:46c99d2de99945ec5cb54f23c8cd5689f6d7177305ebff350a58ce5f8de1669e",                  "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183", @@ -437,35 +442,50 @@                  "sha256:62fe6c95e3ec8a7fad637b7f3d372c15ec1caa01ab47926cfdf7a75b40e0eac1",                  "sha256:6788b695d50a51edb699cb55e35487e430fa21f1ed838122d722e0ff0ac5ba15",                  "sha256:6dd73240d2af64df90aa7c4e7481e23825ea70af4b4922f8ede5b9e35f78a3b1", +                "sha256:6f1e273a344928347c1290119b493a1f0303c52f5a5eae5f16d74f48c15d4a85", +                "sha256:6fffc775d90dcc9aed1b89219549b329a9250d918fd0b8fa8d93d154918422e1",                  "sha256:717ba8fe3ae9cc0006d7c451f0bb265ee07739daf76355d06366154ee68d221e",                  "sha256:79855e1c5b8da654cf486b830bd42c06e8780cea587384cf6545b7d9ac013a0b",                  "sha256:7c1699dfe0cf8ff607dbdcc1e9b9af1755371f92a68f706051cc8c37d447c905", +                "sha256:7fed13866cf14bba33e7176717346713881f56d9d2bcebab207f7a036f41b850", +                "sha256:84dee80c15f1b560d55bcfe6d47b27d070b4681c699c572af2e3c7cc90a3b8e0",                  "sha256:88e5fcfb52ee7b911e8bb6d6aa2fd21fbecc674eadd44118a9cc3863f938e735",                  "sha256:8defac2f2ccd6805ebf65f5eeb132adcf2ab57aa11fdf4c0dd5169a004710e7d", +                "sha256:98bae9582248d6cf62321dcb52aaf5d9adf0bad3b40582925ef7c7f0ed85fceb",                  "sha256:98c7086708b163d425c67c7a91bad6e466bb99d797aa64f965e9d25c12111a5e",                  "sha256:9add70b36c5666a2ed02b43b335fe19002ee5235efd4b8a89bfcf9005bebac0d",                  "sha256:9bf40443012702a1d2070043cb6291650a0841ece432556f784f004937f0f32c", +                "sha256:a6a744282b7718a2a62d2ed9d993cad6f5f585605ad352c11de459f4108df0a1", +                "sha256:acf08ac40292838b3cbbb06cfe9b2cb9ec78fce8baca31ddb87aaac2e2dc3bc2",                  "sha256:ade5e387d2ad0d7ebf59146cc00c8044acbd863725f887353a10df825fc8ae21",                  "sha256:b00c1de48212e4cc9603895652c5c410df699856a2853135b3967591e4beebc2",                  "sha256:b1282f8c00509d99fef04d8ba936b156d419be841854fe901d8ae224c59f0be5", +                "sha256:b1dba4527182c95a0db8b6060cc98ac49b9e2f5e64320e2b56e47cb2831978c7",                  "sha256:b2051432115498d3562c084a49bba65d97cf251f5a331c64a12ee7e04dacc51b", +                "sha256:b7d644ddb4dbd407d31ffb699f1d140bc35478da613b441c582aeb7c43838dd8",                  "sha256:ba59edeaa2fc6114428f1637ffff42da1e311e29382d81b339c1817d37ec93c6", +                "sha256:bf5aa3cbcfdf57fa2ee9cd1822c862ef23037f5c832ad09cfea57fa846dec193",                  "sha256:c8716a48d94b06bb3b2524c2b77e055fb313aeb4ea620c8dd03a105574ba704f", +                "sha256:caabedc8323f1e93231b52fc32bdcde6db817623d33e100708d9a68e1f53b26b",                  "sha256:cd5df75523866410809ca100dc9681e301e3c27567cf498077e8551b6d20e42f",                  "sha256:cdb132fc825c38e1aeec2c8aa9338310d29d337bebbd7baa06889d09a60a1fa2", +                "sha256:d53bc011414228441014aa71dbec320c66468c1030aae3a6e29778a3382d96e5", +                "sha256:d73a845f227b0bfe8a7455ee623525ee656a9e2e749e4742706d80a6065d5e2c", +                "sha256:d9be0ba6c527163cbed5e0857c451fcd092ce83947944d6c14bc95441203f032",                  "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7", -                "sha256:e8313f01ba26fbbe36c7be1966a7b7424942f670f38e666995b88d012765b9be" +                "sha256:e8313f01ba26fbbe36c7be1966a7b7424942f670f38e666995b88d012765b9be", +                "sha256:feb7b34d6325451ef96bc0e36e1a6c0c1c64bc1fbec4b854f4529e51887b1621"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",              "version": "==1.1.1"          },          "more-itertools": {              "hashes": [ -                "sha256:8e1a2a43b2f2727425f2b5839587ae37093f19153dc26c0927d1048ff6557330", -                "sha256:b3a9005928e5bed54076e6e549c792b306fddfe72b2d1d22dd63d42d5d3899cf" +                "sha256:5652a9ac72209ed7df8d9c15daf4e1aa0e3d2ccd3c87f8265a0673cd9cbc9ced", +                "sha256:c5d6da9ca3ff65220c3bfd2a8db06d698f05d4d2b9be57e1deb2be5a45019713"              ],              "index": "pypi", -            "version": "==8.6.0" +            "version": "==8.7.0"          },          "multidict": {              "hashes": [ @@ -519,11 +539,11 @@          },          "packaging": {              "hashes": [ -                "sha256:24e0da08660a87484d1602c30bb4902d74816b6985b93de36926f5bc95741858", -                "sha256:78598185a7008a470d64526a8059de9aaa449238f280fc9eb6b13ba6c4109093" +                "sha256:5b327ac1320dc863dca72f4514ecc086f31186744b84a230374cc1fd776feae5", +                "sha256:67714da7f7bc052e064859c05c595155bd1ee9f69f76557e21f051443c20947a"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", -            "version": "==20.8" +            "version": "==20.9"          },          "pamqp": {              "hashes": [ @@ -576,18 +596,18 @@          },          "pygments": {              "hashes": [ -                "sha256:bc9591213a8f0e0ca1a5e68a479b4887fdc3e75d0774e5c71c31920c427de435", -                "sha256:df49d09b498e83c1a73128295860250b0b7edd4c723a32e9bc0d295c7c2ec337" +                "sha256:37a13ba168a02ac54cc5891a42b1caec333e59b66addb7fa633ea8a6d73445c0", +                "sha256:b21b072d0ccdf29297a82a2363359d99623597b8a265b8081760e4d0f7153c88"              ],              "markers": "python_version >= '3.5'", -            "version": "==2.7.4" +            "version": "==2.8.0"          },          "pyparsing": {              "hashes": [                  "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1",                  "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"              ], -            "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2'", +            "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2, 3.3'",              "version": "==2.4.7"          },          "python-dateutil": { @@ -598,31 +618,46 @@              "index": "pypi",              "version": "==2.8.1"          }, +        "python-json-logger": { +            "hashes": [ +                "sha256:f26eea7898db40609563bed0a7ca11af12e2a79858632706d835a0f961b7d398" +            ], +            "index": "pypi", +            "version": "==2.0.1" +        },          "pytz": {              "hashes": [ -                "sha256:16962c5fb8db4a8f63a26646d8886e9d769b6c511543557bc84e9569fb9a9cb4", -                "sha256:180befebb1927b16f6b57101720075a984c019ac16b1b7575673bea42c6c3da5" +                "sha256:83a4a90894bf38e243cf052c8b58f381bfe9a7a483f6a9cab140bc7f702ac4da", +                "sha256:eb10ce3e7736052ed3623d49975ce333bcd712c7bb19a58b9e2089d4057d0798"              ], -            "version": "==2020.5" +            "version": "==2021.1"          },          "pyyaml": {              "hashes": [ -                "sha256:06a0d7ba600ce0b2d2fe2e78453a470b5a6e000a985dd4a4e54e436cc36b0e97", -                "sha256:240097ff019d7c70a4922b6869d8a86407758333f02203e0fc6ff79c5dcede76", -                "sha256:4f4b913ca1a7319b33cfb1369e91e50354d6f07a135f3b901aca02aa95940bd2", -                "sha256:6034f55dab5fea9e53f436aa68fa3ace2634918e8b5994d82f3621c04ff5ed2e", -                "sha256:69f00dca373f240f842b2931fb2c7e14ddbacd1397d57157a9b005a6a9942648", -                "sha256:73f099454b799e05e5ab51423c7bcf361c58d3206fa7b0d555426b1f4d9a3eaf", -                "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f", -                "sha256:7739fc0fa8205b3ee8808aea45e968bc90082c10aef6ea95e855e10abf4a37b2", -                "sha256:95f71d2af0ff4227885f7a6605c37fd53d3a106fcab511b8860ecca9fcf400ee", -                "sha256:ad9c67312c84def58f3c04504727ca879cb0013b2517c85a9a253f0cb6380c0a", -                "sha256:b8eac752c5e14d3eca0e6dd9199cd627518cb5ec06add0de9d32baeee6fe645d", -                "sha256:cc8955cfbfc7a115fa81d85284ee61147059a753344bc51098f3ccd69b0d7e0c", -                "sha256:d13155f591e6fcc1ec3b30685d50bf0711574e2c0dfffd7644babf8b5102ca1a" +                "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf", +                "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696", +                "sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393", +                "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77", +                "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922", +                "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5", +                "sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8", +                "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10", +                "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc", +                "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018", +                "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e", +                "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253", +                "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183", +                "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb", +                "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185", +                "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db", +                "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46", +                "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b", +                "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63", +                "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df", +                "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc"              ],              "index": "pypi", -            "version": "==5.3.1" +            "version": "==5.4.1"          },          "redis": {              "hashes": [ @@ -642,26 +677,26 @@          },          "sentry-sdk": {              "hashes": [ -                "sha256:0a711ec952441c2ec89b8f5d226c33bc697914f46e876b44a4edd3e7864cf4d0", -                "sha256:737a094e49a529dd0fdcaafa9e97cf7c3d5eb964bd229821d640bc77f3502b3f" +                "sha256:4ae8d1ced6c67f1c8ea51d82a16721c166c489b76876c9f2c202b8a50334b237", +                "sha256:e75c8c58932bda8cd293ea8e4b242527129e1caaec91433d21b8b2f20fee030b"              ],              "index": "pypi", -            "version": "==0.19.5" +            "version": "==0.20.3"          },          "six": {              "hashes": [                  "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259",                  "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"              ], -            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2'", +            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",              "version": "==1.15.0"          },          "snowballstemmer": {              "hashes": [ -                "sha256:209f257d7533fdb3cb73bdbd24f436239ca3b2fa67d56f6ff88e86be08cc5ef0", -                "sha256:df3bac3df4c2c01363f3dd2cfa78cce2840a79b9f1c2d2de9ce8d31683992f52" +                "sha256:b51b447bea85f9968c13b650126a888aabd4cb4463fca868ec596826325dedc2", +                "sha256:e997baa4f2e9139951b6f4c631bad912dfd3c792467e2f03d7239464af90e914"              ], -            "version": "==2.0.0" +            "version": "==2.1.0"          },          "sortedcontainers": {              "hashes": [ @@ -672,11 +707,11 @@          },          "soupsieve": {              "hashes": [ -                "sha256:4bb21a6ee4707bf43b61230e80740e71bfe56e55d1f1f50924b087bb2975c851", -                "sha256:6dc52924dc0bc710a5d16794e6b3480b2c7c08b07729505feab2b2c16661ff6e" +                "sha256:407fa1e8eb3458d1b5614df51d9651a1180ea5fedf07feb46e45d7e25e6d6cdd", +                "sha256:d3a5ea5b350423f47d07639f74475afedad48cf41c0ad7a82ca13a3928af34f6"              ],              "markers": "python_version >= '3.0'", -            "version": "==2.1" +            "version": "==2.2"          },          "sphinx": {              "hashes": [ @@ -752,11 +787,11 @@          },          "urllib3": {              "hashes": [ -                "sha256:19188f96923873c92ccb987120ec4acaa12f0461fa9ce5d3d0772bc965a39e08", -                "sha256:d8ff90d979214d7b4f8ce956e80f4028fc6860e4431f731ea4a8c08f23f99473" +                "sha256:1b465e494e3e0d8939b50680403e3aedaa2bc434b7d5af64dfd3c958d7f5ae80", +                "sha256:de3eedaad74a2683334e282005cd8d7f22f4d55fa690a2a1020a416cb0a47e73"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", -            "version": "==1.26.2" +            "version": "==1.26.3"          },          "yarl": {              "hashes": [ @@ -842,58 +877,61 @@          },          "coverage": {              "hashes": [ -                "sha256:08b3ba72bd981531fd557f67beee376d6700fba183b167857038997ba30dd297", -                "sha256:2757fa64e11ec12220968f65d086b7a29b6583d16e9a544c889b22ba98555ef1", -                "sha256:3102bb2c206700a7d28181dbe04d66b30780cde1d1c02c5f3c165cf3d2489497", -                "sha256:3498b27d8236057def41de3585f317abae235dd3a11d33e01736ffedb2ef8606", -                "sha256:378ac77af41350a8c6b8801a66021b52da8a05fd77e578b7380e876c0ce4f528", -                "sha256:38f16b1317b8dd82df67ed5daa5f5e7c959e46579840d77a67a4ceb9cef0a50b", -                "sha256:3911c2ef96e5ddc748a3c8b4702c61986628bb719b8378bf1e4a6184bbd48fe4", -                "sha256:3a3c3f8863255f3c31db3889f8055989527173ef6192a283eb6f4db3c579d830", -                "sha256:3b14b1da110ea50c8bcbadc3b82c3933974dbeea1832e814aab93ca1163cd4c1", -                "sha256:535dc1e6e68fad5355f9984d5637c33badbdc987b0c0d303ee95a6c979c9516f", -                "sha256:6f61319e33222591f885c598e3e24f6a4be3533c1d70c19e0dc59e83a71ce27d", -                "sha256:723d22d324e7997a651478e9c5a3120a0ecbc9a7e94071f7e1954562a8806cf3", -                "sha256:76b2775dda7e78680d688daabcb485dc87cf5e3184a0b3e012e1d40e38527cc8", -                "sha256:782a5c7df9f91979a7a21792e09b34a658058896628217ae6362088b123c8500", -                "sha256:7e4d159021c2029b958b2363abec4a11db0ce8cd43abb0d9ce44284cb97217e7", -                "sha256:8dacc4073c359f40fcf73aede8428c35f84639baad7e1b46fce5ab7a8a7be4bb", -                "sha256:8f33d1156241c43755137288dea619105477961cfa7e47f48dbf96bc2c30720b", -                "sha256:8ffd4b204d7de77b5dd558cdff986a8274796a1e57813ed005b33fd97e29f059", -                "sha256:93a280c9eb736a0dcca19296f3c30c720cb41a71b1f9e617f341f0a8e791a69b", -                "sha256:9a4f66259bdd6964d8cf26142733c81fb562252db74ea367d9beb4f815478e72", -                "sha256:9a9d4ff06804920388aab69c5ea8a77525cf165356db70131616acd269e19b36", -                "sha256:a2070c5affdb3a5e751f24208c5c4f3d5f008fa04d28731416e023c93b275277", -                "sha256:a4857f7e2bc6921dbd487c5c88b84f5633de3e7d416c4dc0bb70256775551a6c", -                "sha256:a607ae05b6c96057ba86c811d9c43423f35e03874ffb03fbdcd45e0637e8b631", -                "sha256:a66ca3bdf21c653e47f726ca57f46ba7fc1f260ad99ba783acc3e58e3ebdb9ff", -                "sha256:ab110c48bc3d97b4d19af41865e14531f300b482da21783fdaacd159251890e8", -                "sha256:b239711e774c8eb910e9b1ac719f02f5ae4bf35fa0420f438cdc3a7e4e7dd6ec", -                "sha256:be0416074d7f253865bb67630cf7210cbc14eb05f4099cc0f82430135aaa7a3b", -                "sha256:c46643970dff9f5c976c6512fd35768c4a3819f01f61169d8cdac3f9290903b7", -                "sha256:c5ec71fd4a43b6d84ddb88c1df94572479d9a26ef3f150cef3dacefecf888105", -                "sha256:c6e5174f8ca585755988bc278c8bb5d02d9dc2e971591ef4a1baabdf2d99589b", -                "sha256:c89b558f8a9a5a6f2cfc923c304d49f0ce629c3bd85cb442ca258ec20366394c", -                "sha256:cc44e3545d908ecf3e5773266c487ad1877be718d9dc65fc7eb6e7d14960985b", -                "sha256:cc6f8246e74dd210d7e2b56c76ceaba1cc52b025cd75dbe96eb48791e0250e98", -                "sha256:cd556c79ad665faeae28020a0ab3bda6cd47d94bec48e36970719b0b86e4dcf4", -                "sha256:ce6f3a147b4b1a8b09aae48517ae91139b1b010c5f36423fa2b866a8b23df879", -                "sha256:ceb499d2b3d1d7b7ba23abe8bf26df5f06ba8c71127f188333dddcf356b4b63f", -                "sha256:cef06fb382557f66d81d804230c11ab292d94b840b3cb7bf4450778377b592f4", -                "sha256:e448f56cfeae7b1b3b5bcd99bb377cde7c4eb1970a525c770720a352bc4c8044", -                "sha256:e52d3d95df81c8f6b2a1685aabffadf2d2d9ad97203a40f8d61e51b70f191e4e", -                "sha256:ee2f1d1c223c3d2c24e3afbb2dd38be3f03b1a8d6a83ee3d9eb8c36a52bee899", -                "sha256:f2c6888eada180814b8583c3e793f3f343a692fc802546eed45f40a001b1169f", -                "sha256:f51dbba78d68a44e99d484ca8c8f604f17e957c1ca09c3ebc2c7e3bbd9ba0448", -                "sha256:f54de00baf200b4539a5a092a759f000b5f45fd226d6d25a76b0dff71177a714", -                "sha256:fa10fee7e32213f5c7b0d6428ea92e3a3fdd6d725590238a3f92c0de1c78b9d2", -                "sha256:fabeeb121735d47d8eab8671b6b031ce08514c86b7ad8f7d5490a7b6dcd6267d", -                "sha256:fac3c432851038b3e6afe086f777732bcf7f6ebbfd90951fa04ee53db6d0bcdd", -                "sha256:fda29412a66099af6d6de0baa6bd7c52674de177ec2ad2630ca264142d69c6c7", -                "sha256:ff1330e8bc996570221b450e2d539134baa9465f5cb98aff0e0f73f34172e0ae" +                "sha256:004d1880bed2d97151facef49f08e255a20ceb6f9432df75f4eef018fdd5a78c", +                "sha256:01d84219b5cdbfc8122223b39a954820929497a1cb1422824bb86b07b74594b6", +                "sha256:040af6c32813fa3eae5305d53f18875bedd079960822ef8ec067a66dd8afcd45", +                "sha256:06191eb60f8d8a5bc046f3799f8a07a2d7aefb9504b0209aff0b47298333302a", +                "sha256:13034c4409db851670bc9acd836243aeee299949bd5673e11844befcb0149f03", +                "sha256:13c4ee887eca0f4c5a247b75398d4114c37882658300e153113dafb1d76de529", +                "sha256:184a47bbe0aa6400ed2d41d8e9ed868b8205046518c52464fde713ea06e3a74a", +                "sha256:18ba8bbede96a2c3dde7b868de9dcbd55670690af0988713f0603f037848418a", +                "sha256:1aa846f56c3d49205c952d8318e76ccc2ae23303351d9270ab220004c580cfe2", +                "sha256:217658ec7187497e3f3ebd901afdca1af062b42cfe3e0dafea4cced3983739f6", +                "sha256:24d4a7de75446be83244eabbff746d66b9240ae020ced65d060815fac3423759", +                "sha256:2910f4d36a6a9b4214bb7038d537f015346f413a975d57ca6b43bf23d6563b53", +                "sha256:2949cad1c5208b8298d5686d5a85b66aae46d73eec2c3e08c817dd3513e5848a", +                "sha256:2a3859cb82dcbda1cfd3e6f71c27081d18aa251d20a17d87d26d4cd216fb0af4", +                "sha256:2cafbbb3af0733db200c9b5f798d18953b1a304d3f86a938367de1567f4b5bff", +                "sha256:2e0d881ad471768bf6e6c2bf905d183543f10098e3b3640fc029509530091502", +                "sha256:30c77c1dc9f253283e34c27935fded5015f7d1abe83bc7821680ac444eaf7793", +                "sha256:3487286bc29a5aa4b93a072e9592f22254291ce96a9fbc5251f566b6b7343cdb", +                "sha256:372da284cfd642d8e08ef606917846fa2ee350f64994bebfbd3afb0040436905", +                "sha256:41179b8a845742d1eb60449bdb2992196e211341818565abded11cfa90efb821", +                "sha256:44d654437b8ddd9eee7d1eaee28b7219bec228520ff809af170488fd2fed3e2b", +                "sha256:4a7697d8cb0f27399b0e393c0b90f0f1e40c82023ea4d45d22bce7032a5d7b81", +                "sha256:51cb9476a3987c8967ebab3f0fe144819781fca264f57f89760037a2ea191cb0", +                "sha256:52596d3d0e8bdf3af43db3e9ba8dcdaac724ba7b5ca3f6358529d56f7a166f8b", +                "sha256:53194af30d5bad77fcba80e23a1441c71abfb3e01192034f8246e0d8f99528f3", +                "sha256:5fec2d43a2cc6965edc0bb9e83e1e4b557f76f843a77a2496cbe719583ce8184", +                "sha256:6c90e11318f0d3c436a42409f2749ee1a115cd8b067d7f14c148f1ce5574d701", +                "sha256:74d881fc777ebb11c63736622b60cb9e4aee5cace591ce274fb69e582a12a61a", +                "sha256:7501140f755b725495941b43347ba8a2777407fc7f250d4f5a7d2a1050ba8e82", +                "sha256:796c9c3c79747146ebd278dbe1e5c5c05dd6b10cc3bcb8389dfdf844f3ead638", +                "sha256:869a64f53488f40fa5b5b9dcb9e9b2962a66a87dab37790f3fcfb5144b996ef5", +                "sha256:8963a499849a1fc54b35b1c9f162f4108017b2e6db2c46c1bed93a72262ed083", +                "sha256:8d0a0725ad7c1a0bcd8d1b437e191107d457e2ec1084b9f190630a4fb1af78e6", +                "sha256:900fbf7759501bc7807fd6638c947d7a831fc9fdf742dc10f02956ff7220fa90", +                "sha256:92b017ce34b68a7d67bd6d117e6d443a9bf63a2ecf8567bb3d8c6c7bc5014465", +                "sha256:970284a88b99673ccb2e4e334cfb38a10aab7cd44f7457564d11898a74b62d0a", +                "sha256:972c85d205b51e30e59525694670de6a8a89691186012535f9d7dbaa230e42c3", +                "sha256:9a1ef3b66e38ef8618ce5fdc7bea3d9f45f3624e2a66295eea5e57966c85909e", +                "sha256:af0e781009aaf59e25c5a678122391cb0f345ac0ec272c7961dc5455e1c40066", +                "sha256:b6d534e4b2ab35c9f93f46229363e17f63c53ad01330df9f2d6bd1187e5eaacf", +                "sha256:b7895207b4c843c76a25ab8c1e866261bcfe27bfaa20c192de5190121770672b", +                "sha256:c0891a6a97b09c1f3e073a890514d5012eb256845c451bd48f7968ef939bf4ae", +                "sha256:c2723d347ab06e7ddad1a58b2a821218239249a9e4365eaff6649d31180c1669", +                "sha256:d1f8bf7b90ba55699b3a5e44930e93ff0189aa27186e96071fac7dd0d06a1873", +                "sha256:d1f9ce122f83b2305592c11d64f181b87153fc2c2bbd3bb4a3dde8303cfb1a6b", +                "sha256:d314ed732c25d29775e84a960c3c60808b682c08d86602ec2c3008e1202e3bb6", +                "sha256:d636598c8305e1f90b439dbf4f66437de4a5e3c31fdf47ad29542478c8508bbb", +                "sha256:deee1077aae10d8fa88cb02c845cfba9b62c55e1183f52f6ae6a2df6a2187160", +                "sha256:ebe78fe9a0e874362175b02371bdfbee64d8edc42a044253ddf4ee7d3c15212c", +                "sha256:f030f8873312a16414c0d8e1a1ddff2d3235655a2174e3648b4fa66b3f2f1079", +                "sha256:f0b278ce10936db1a37e6954e15a3730bea96a0997c26d7fee88e6c396c2086d", +                "sha256:f11642dddbb0253cc8853254301b51390ba0081750a8ac03f20ea8103f0c56b6"              ],              "index": "pypi", -            "version": "==5.3.1" +            "version": "==5.5"          },          "coveralls": {              "hashes": [ @@ -933,11 +971,11 @@          },          "flake8-annotations": {              "hashes": [ -                "sha256:3a377140556aecf11fa9f3bb18c10db01f5ea56dc79a730e2ec9b4f1f49e2055", -                "sha256:e17947a48a5b9f632fe0c72682fc797c385e451048e7dfb20139f448a074cb3e" +                "sha256:8968ff12f296433028ad561c680ccc03a7cd62576d100c3f1475e058b3c11b43", +                "sha256:bd0505616c0d85ebb45c6052d339c69f320d3f87fa079ab4e91a4f234a863d05"              ],              "index": "pypi", -            "version": "==2.5.0" +            "version": "==2.6.0"          },          "flake8-bugbear": {              "hashes": [ @@ -995,11 +1033,11 @@          },          "identify": {              "hashes": [ -                "sha256:18994e850ba50c37bcaed4832be8b354d6a06c8fb31f54e0e7ece76d32f69bc8", -                "sha256:892473bf12e655884132a3a32aca737a3cbefaa34a850ff52d501773a45837bc" +                "sha256:2179e7359471ab55729f201b3fdf7dc2778e221f868410fedcb0987b791ba552", +                "sha256:2a5fdf2f5319cc357eda2550bea713a404392495961022cf2462624ce62f0f46"              ], -            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", -            "version": "==1.5.12" +            "markers": "python_full_version >= '3.6.1'", +            "version": "==2.1.0"          },          "idna": {              "hashes": [ @@ -1033,11 +1071,11 @@          },          "pre-commit": {              "hashes": [ -                "sha256:6c86d977d00ddc8a60d68eec19f51ef212d9462937acf3ea37c7adec32284ac0", -                "sha256:ee784c11953e6d8badb97d19bc46b997a3a9eded849881ec587accd8608d74a4" +                "sha256:16212d1fde2bed88159287da88ff03796863854b04dc9f838a55979325a3d20e", +                "sha256:399baf78f13f4de82a29b649afd74bef2c4e28eb4f021661fc7f29246e8c7a3a"              ],              "index": "pypi", -            "version": "==2.9.3" +            "version": "==2.10.1"          },          "pycodestyle": {              "hashes": [ @@ -1065,22 +1103,30 @@          },          "pyyaml": {              "hashes": [ -                "sha256:06a0d7ba600ce0b2d2fe2e78453a470b5a6e000a985dd4a4e54e436cc36b0e97", -                "sha256:240097ff019d7c70a4922b6869d8a86407758333f02203e0fc6ff79c5dcede76", -                "sha256:4f4b913ca1a7319b33cfb1369e91e50354d6f07a135f3b901aca02aa95940bd2", -                "sha256:6034f55dab5fea9e53f436aa68fa3ace2634918e8b5994d82f3621c04ff5ed2e", -                "sha256:69f00dca373f240f842b2931fb2c7e14ddbacd1397d57157a9b005a6a9942648", -                "sha256:73f099454b799e05e5ab51423c7bcf361c58d3206fa7b0d555426b1f4d9a3eaf", -                "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f", -                "sha256:7739fc0fa8205b3ee8808aea45e968bc90082c10aef6ea95e855e10abf4a37b2", -                "sha256:95f71d2af0ff4227885f7a6605c37fd53d3a106fcab511b8860ecca9fcf400ee", -                "sha256:ad9c67312c84def58f3c04504727ca879cb0013b2517c85a9a253f0cb6380c0a", -                "sha256:b8eac752c5e14d3eca0e6dd9199cd627518cb5ec06add0de9d32baeee6fe645d", -                "sha256:cc8955cfbfc7a115fa81d85284ee61147059a753344bc51098f3ccd69b0d7e0c", -                "sha256:d13155f591e6fcc1ec3b30685d50bf0711574e2c0dfffd7644babf8b5102ca1a" +                "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf", +                "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696", +                "sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393", +                "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77", +                "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922", +                "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5", +                "sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8", +                "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10", +                "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc", +                "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018", +                "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e", +                "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253", +                "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183", +                "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb", +                "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185", +                "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db", +                "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46", +                "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b", +                "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63", +                "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df", +                "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc"              ],              "index": "pypi", -            "version": "==5.3.1" +            "version": "==5.4.1"          },          "requests": {              "hashes": [ @@ -1095,39 +1141,39 @@                  "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259",                  "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"              ], -            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2'", +            "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",              "version": "==1.15.0"          },          "snowballstemmer": {              "hashes": [ -                "sha256:209f257d7533fdb3cb73bdbd24f436239ca3b2fa67d56f6ff88e86be08cc5ef0", -                "sha256:df3bac3df4c2c01363f3dd2cfa78cce2840a79b9f1c2d2de9ce8d31683992f52" +                "sha256:b51b447bea85f9968c13b650126a888aabd4cb4463fca868ec596826325dedc2", +                "sha256:e997baa4f2e9139951b6f4c631bad912dfd3c792467e2f03d7239464af90e914"              ], -            "version": "==2.0.0" +            "version": "==2.1.0"          },          "toml": {              "hashes": [                  "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b",                  "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"              ], -            "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2'", +            "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2, 3.3'",              "version": "==0.10.2"          },          "urllib3": {              "hashes": [ -                "sha256:19188f96923873c92ccb987120ec4acaa12f0461fa9ce5d3d0772bc965a39e08", -                "sha256:d8ff90d979214d7b4f8ce956e80f4028fc6860e4431f731ea4a8c08f23f99473" +                "sha256:1b465e494e3e0d8939b50680403e3aedaa2bc434b7d5af64dfd3c958d7f5ae80", +                "sha256:de3eedaad74a2683334e282005cd8d7f22f4d55fa690a2a1020a416cb0a47e73"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", -            "version": "==1.26.2" +            "version": "==1.26.3"          },          "virtualenv": {              "hashes": [ -                "sha256:0c111a2236b191422b37fe8c28b8c828ced39aab4bf5627fa5c331aeffb570d9", -                "sha256:14b34341e742bdca219e10708198e704e8a7064dd32f474fc16aca68ac53a306" +                "sha256:147b43894e51dd6bba882cf9c282447f780e2251cd35172403745fc381a0a80d", +                "sha256:2be72df684b74df0ea47679a7df93fd0e04e72520022c57b479d8f881485dbe3"              ],              "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", -            "version": "==20.3.1" +            "version": "==20.4.2"          }      }  } diff --git a/bot/__main__.py b/bot/__main__.py index 257216fa7..9317563c8 100644 --- a/bot/__main__.py +++ b/bot/__main__.py @@ -1,10 +1,28 @@ +import logging + +import aiohttp +  import bot  from bot import constants -from bot.bot import Bot +from bot.bot import Bot, StartupError  from bot.log import setup_sentry  setup_sentry() -bot.instance = Bot.create() -bot.instance.load_extensions() -bot.instance.run(constants.Bot.token) +try: +    bot.instance = Bot.create() +    bot.instance.load_extensions() +    bot.instance.run(constants.Bot.token) +except StartupError as e: +    message = "Unknown Startup Error Occurred." +    if isinstance(e.exception, (aiohttp.ClientConnectorError, aiohttp.ServerDisconnectedError)): +        message = "Could not connect to site API. Is it running?" +    elif isinstance(e.exception, OSError): +        message = "Could not connect to Redis. Is it running?" + +    # The exception is logged with an empty message so the actual message is visible at the bottom +    log = logging.getLogger("bot") +    log.fatal("", exc_info=e.exception) +    log.fatal(message) + +    exit(69) diff --git a/bot/bot.py b/bot/bot.py index d5f108575..3a2af472d 100644 --- a/bot/bot.py +++ b/bot/bot.py @@ -19,6 +19,14 @@ log = logging.getLogger('bot')  LOCALHOST = "127.0.0.1" +class StartupError(Exception): +    """Exception class for startup errors.""" + +    def __init__(self, base: Exception): +        super().__init__() +        self.exception = base + +  class Bot(commands.Bot):      """A subclass of `discord.ext.commands.Bot` with an aiohttp session and an API client.""" @@ -81,6 +89,22 @@ class Bot(commands.Bot):          for item in full_cache:              self.insert_item_into_filter_list_cache(item) +    async def ping_services(self) -> None: +        """A helper to make sure all the services the bot relies on are available on startup.""" +        # Connect Site/API +        attempts = 0 +        while True: +            try: +                log.info(f"Attempting site connection: {attempts + 1}/{constants.URLs.connect_max_retries}") +                await self.api_client.get("healthcheck") +                break + +            except (aiohttp.ClientConnectorError, aiohttp.ServerDisconnectedError): +                attempts += 1 +                if attempts == constants.URLs.connect_max_retries: +                    raise +                await asyncio.sleep(constants.URLs.connect_cooldown) +      @classmethod      def create(cls) -> "Bot":          """Create and return an instance of a Bot.""" @@ -223,6 +247,11 @@ class Bot(commands.Bot):              # here. Normally, this shouldn't happen.              await self.redis_session.connect() +        try: +            await self.ping_services() +        except Exception as e: +            raise StartupError(e) +          # Build the FilterList cache          await self.cache_filter_list_data() @@ -318,5 +347,8 @@ def _create_redis_session(loop: asyncio.AbstractEventLoop) -> RedisSession:          use_fakeredis=constants.Redis.use_fakeredis,          global_namespace="bot",      ) -    loop.run_until_complete(redis_session.connect()) +    try: +        loop.run_until_complete(redis_session.connect()) +    except OSError as e: +        raise StartupError(e)      return redis_session diff --git a/bot/constants.py b/bot/constants.py index 69bc82b89..394d59a73 100644 --- a/bot/constants.py +++ b/bot/constants.py @@ -280,9 +280,9 @@ class Emojis(metaclass=YAMLGetter):      badge_staff: str      badge_verified_bot_developer: str -    defcon_disabled: str  # noqa: E704 -    defcon_enabled: str  # noqa: E704 -    defcon_updated: str  # noqa: E704 +    defcon_shutdown: str  # noqa: E704 +    defcon_unshutdown: str  # noqa: E704 +    defcon_update: str  # noqa: E704      failmail: str @@ -319,9 +319,9 @@ class Icons(metaclass=YAMLGetter):      crown_red: str      defcon_denied: str    # noqa: E704 -    defcon_disabled: str  # noqa: E704 -    defcon_enabled: str   # noqa: E704 -    defcon_updated: str   # noqa: E704 +    defcon_shutdown: str  # noqa: E704 +    defcon_unshutdown: str   # noqa: E704 +    defcon_update: str   # noqa: E704      filtering: str @@ -487,6 +487,7 @@ class Roles(metaclass=YAMLGetter):      admins: int      core_developers: int +    devops: int      helpers: int      moderators: int      owners: int @@ -531,6 +532,8 @@ class URLs(metaclass=YAMLGetter):      github_bot_repo: str      # Base site vars +    connect_max_retries: int +    connect_cooldown: int      site: str      site_api: str      site_schema: str diff --git a/bot/converters.py b/bot/converters.py index 0d9a519df..67525cd4d 100644 --- a/bot/converters.py +++ b/bot/converters.py @@ -16,6 +16,7 @@ from discord.utils import DISCORD_EPOCH, snowflake_time  from bot.api import ResponseCodeError  from bot.constants import URLs  from bot.utils.regex import INVITE_RE +from bot.utils.time import parse_duration_string  log = logging.getLogger(__name__) @@ -301,16 +302,6 @@ class TagContentConverter(Converter):  class DurationDelta(Converter):      """Convert duration strings into dateutil.relativedelta.relativedelta objects.""" -    duration_parser = re.compile( -        r"((?P<years>\d+?) ?(years|year|Y|y) ?)?" -        r"((?P<months>\d+?) ?(months|month|m) ?)?" -        r"((?P<weeks>\d+?) ?(weeks|week|W|w) ?)?" -        r"((?P<days>\d+?) ?(days|day|D|d) ?)?" -        r"((?P<hours>\d+?) ?(hours|hour|H|h) ?)?" -        r"((?P<minutes>\d+?) ?(minutes|minute|M) ?)?" -        r"((?P<seconds>\d+?) ?(seconds|second|S|s))?" -    ) -      async def convert(self, ctx: Context, duration: str) -> relativedelta:          """          Converts a `duration` string to a relativedelta object. @@ -326,13 +317,9 @@ class DurationDelta(Converter):          The units need to be provided in descending order of magnitude.          """ -        match = self.duration_parser.fullmatch(duration) -        if not match: +        if not (delta := parse_duration_string(duration)):              raise BadArgument(f"`{duration}` is not a valid duration string.") -        duration_dict = {unit: int(amount) for unit, amount in match.groupdict(default=0).items()} -        delta = relativedelta(**duration_dict) -          return delta @@ -357,27 +344,38 @@ class Duration(DurationDelta):  class OffTopicName(Converter):      """A converter that ensures an added off-topic name is valid.""" +    ALLOWED_CHARACTERS = "ABCDEFGHIJKLMNOPQRSTUVWXYZ!?'`-" + +    @classmethod +    def translate_name(cls, name: str, *, from_unicode: bool = True) -> str: +        """ +        Translates `name` into a format that is allowed in discord channel names. + +        If `from_unicode` is True, the name is translated from a discord-safe format, back to normalized text. +        """ +        if from_unicode: +            table = str.maketrans(cls.ALLOWED_CHARACTERS, '𝖠𝖡𝖢𝖣𝖤𝖥𝖦𝖧𝖨𝖩𝖪𝖫𝖬𝖭𝖮𝖯𝖰𝖱𝖲𝖳𝖴𝖵𝖶𝖷𝖸𝖹ǃ?’’-') +        else: +            table = str.maketrans('𝖠𝖡𝖢𝖣𝖤𝖥𝖦𝖧𝖨𝖩𝖪𝖫𝖬𝖭𝖮𝖯𝖰𝖱𝖲𝖳𝖴𝖵𝖶𝖷𝖸𝖹ǃ?’’-', cls.ALLOWED_CHARACTERS) + +        return name.translate(table) +      async def convert(self, ctx: Context, argument: str) -> str:          """Attempt to replace any invalid characters with their approximate Unicode equivalent.""" -        allowed_characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ!?'`-" -          # Chain multiple words to a single one          argument = "-".join(argument.split())          if not (2 <= len(argument) <= 96):              raise BadArgument("Channel name must be between 2 and 96 chars long") -        elif not all(c.isalnum() or c in allowed_characters for c in argument): +        elif not all(c.isalnum() or c in self.ALLOWED_CHARACTERS for c in argument):              raise BadArgument(                  "Channel name must only consist of "                  "alphanumeric characters, minus signs or apostrophes."              )          # Replace invalid characters with unicode alternatives. -        table = str.maketrans( -            allowed_characters, '𝖠𝖡𝖢𝖣𝖤𝖥𝖦𝖧𝖨𝖩𝖪𝖫𝖬𝖭𝖮𝖯𝖰𝖱𝖲𝖳𝖴𝖵𝖶𝖷𝖸𝖹ǃ?’’-' -        ) -        return argument.translate(table) +        return self.translate_name(argument)  class ISODateTime(Converter): diff --git a/bot/exts/backend/error_handler.py b/bot/exts/backend/error_handler.py index d2cce5558..9cb54cdab 100644 --- a/bot/exts/backend/error_handler.py +++ b/bot/exts/backend/error_handler.py @@ -239,10 +239,12 @@ class ErrorHandler(Cog):          elif isinstance(e, errors.BadUnionArgument):              embed = self._get_error_embed("Bad argument", f"{e}\n{e.errors[-1]}")              await ctx.send(embed=embed) +            await prepared_help_command              self.bot.stats.incr("errors.bad_union_argument")          elif isinstance(e, errors.ArgumentParsingError):              embed = self._get_error_embed("Argument parsing error", str(e))              await ctx.send(embed=embed) +            prepared_help_command.close()              self.bot.stats.incr("errors.argument_parsing_error")          else:              embed = self._get_error_embed( diff --git a/bot/exts/filters/filtering.py b/bot/exts/filters/filtering.py index 3527bf8bb..c90b18dcb 100644 --- a/bot/exts/filters/filtering.py +++ b/bot/exts/filters/filtering.py @@ -2,7 +2,7 @@ import asyncio  import logging  import re  from datetime import datetime, timedelta -from typing import Any, Dict, List, Mapping, NamedTuple, Optional, Union +from typing import Any, Dict, List, Mapping, NamedTuple, Optional, Tuple, Union  import dateutil  import discord.errors @@ -137,6 +137,10 @@ class Filtering(Cog):          """Fetch items from the filter_list_cache."""          return self.bot.filter_list_cache[f"{list_type.upper()}.{allowed}"].keys() +    def _get_filterlist_value(self, list_type: str, value: Any, *, allowed: bool) -> dict: +        """Fetch one specific value from filter_list_cache.""" +        return self.bot.filter_list_cache[f"{list_type.upper()}.{allowed}"][value] +      @staticmethod      def _expand_spoilers(text: str) -> str:          """Return a string containing all interpretations of a spoilered message.""" @@ -236,7 +240,13 @@ class Filtering(Cog):                  # We also do not need to worry about filters that take the full message,                  # since all we have is an arbitrary string.                  if _filter["enabled"] and _filter["content_only"]: -                    match = await _filter["function"](result) +                    filter_result = await _filter["function"](result) +                    reason = None + +                    if isinstance(filter_result, tuple): +                        match, reason = filter_result +                    else: +                        match = filter_result                      if match:                          # If this is a filter (not a watchlist), we set the variable so we know @@ -245,7 +255,7 @@ class Filtering(Cog):                              filter_triggered = True                          stats = self._add_stats(filter_name, match, result) -                        await self._send_log(filter_name, _filter, msg, stats, is_eval=True) +                        await self._send_log(filter_name, _filter, msg, stats, reason, is_eval=True)                          break  # We don't want multiple filters to trigger @@ -267,9 +277,17 @@ class Filtering(Cog):                      # Does the filter only need the message content or the full message?                      if _filter["content_only"]: -                        match = await _filter["function"](msg.content) +                        payload = msg.content +                    else: +                        payload = msg + +                    result = await _filter["function"](payload) +                    reason = None + +                    if isinstance(result, tuple): +                        match, reason = result                      else: -                        match = await _filter["function"](msg) +                        match = result                      if match:                          is_private = msg.channel.type is discord.ChannelType.private @@ -316,7 +334,7 @@ class Filtering(Cog):                                  log.trace(f"Offensive message {msg.id} will be deleted on {delete_date}")                          stats = self._add_stats(filter_name, match, msg.content) -                        await self._send_log(filter_name, _filter, msg, stats) +                        await self._send_log(filter_name, _filter, msg, stats, reason)                          break  # We don't want multiple filters to trigger @@ -326,6 +344,7 @@ class Filtering(Cog):          _filter: Dict[str, Any],          msg: discord.Message,          stats: Stats, +        reason: Optional[str] = None,          *,          is_eval: bool = False,      ) -> None: @@ -339,6 +358,7 @@ class Filtering(Cog):              ping_everyone = Filter.ping_everyone and _filter.get("ping_everyone", True)          eval_msg = "using !eval " if is_eval else "" +        footer = f"Reason: {reason}" if reason else None          message = (              f"The {filter_name} {_filter['type']} was triggered by {format_user(msg.author)} "              f"{channel_str} {eval_msg}with [the following message]({msg.jump_url}):\n\n" @@ -357,6 +377,7 @@ class Filtering(Cog):              channel_id=Channels.mod_alerts,              ping_everyone=ping_everyone,              additional_embeds=stats.additional_embeds, +            footer=footer,          )      def _add_stats(self, name: str, match: FilterMatch, content: str) -> Stats: @@ -381,13 +402,14 @@ class Filtering(Cog):          if name == "filter_invites" and match is not True:              additional_embeds = []              for _, data in match.items(): +                reason = f"Reason: {data['reason']} | " if data.get('reason') else ""                  embed = discord.Embed(description=(                      f"**Members:**\n{data['members']}\n"                      f"**Active:**\n{data['active']}"                  ))                  embed.set_author(name=data["name"])                  embed.set_thumbnail(url=data["icon"]) -                embed.set_footer(text=f"Guild ID: {data['id']}") +                embed.set_footer(text=f"{reason}Guild ID: {data['id']}")                  additional_embeds.append(embed)          elif name == "watch_rich_embeds": @@ -411,39 +433,46 @@ class Filtering(Cog):              and not msg.author.bot                          # Author not a bot          ) -    async def _has_watch_regex_match(self, text: str) -> Union[bool, re.Match]: +    async def _has_watch_regex_match(self, text: str) -> Tuple[Union[bool, re.Match], Optional[str]]:          """          Return True if `text` matches any regex from `word_watchlist` or `token_watchlist` configs.          `word_watchlist`'s patterns are placed between word boundaries while `token_watchlist` is          matched as-is. Spoilers are expanded, if any, and URLs are ignored. +        Second return value is a reason written to database about blacklist entry (can be None).          """          if SPOILER_RE.search(text):              text = self._expand_spoilers(text)          # Make sure it's not a URL          if URL_RE.search(text): -            return False +            return False, None          watchlist_patterns = self._get_filterlist_items('filter_token', allowed=False)          for pattern in watchlist_patterns:              match = re.search(pattern, text, flags=re.IGNORECASE)              if match: -                return match +                return match, self._get_filterlist_value('filter_token', pattern, allowed=False)['comment'] + +        return False, None -    async def _has_urls(self, text: str) -> bool: -        """Returns True if the text contains one of the blacklisted URLs from the config file.""" +    async def _has_urls(self, text: str) -> Tuple[bool, Optional[str]]: +        """ +        Returns True if the text contains one of the blacklisted URLs from the config file. + +        Second return value is a reason of URL blacklisting (can be None). +        """          if not URL_RE.search(text): -            return False +            return False, None          text = text.lower()          domain_blacklist = self._get_filterlist_items("domain_name", allowed=False)          for url in domain_blacklist:              if url.lower() in text: -                return True +                return True, self._get_filterlist_value("domain_name", url, allowed=False)["comment"] -        return False +        return False, None      @staticmethod      async def _has_zalgo(text: str) -> bool: @@ -500,6 +529,10 @@ class Filtering(Cog):              )              if invite_not_allowed: +                reason = None +                if guild_id in guild_invite_blacklist: +                    reason = self._get_filterlist_value("guild_invite", guild_id, allowed=False)["comment"] +                  guild_icon_hash = guild["icon"]                  guild_icon = (                      "https://cdn.discordapp.com/icons/" @@ -511,7 +544,8 @@ class Filtering(Cog):                      "id": guild['id'],                      "icon": guild_icon,                      "members": response["approximate_member_count"], -                    "active": response["approximate_presence_count"] +                    "active": response["approximate_presence_count"], +                    "reason": reason                  }          return invite_data if invite_data else False diff --git a/bot/exts/fun/off_topic_names.py b/bot/exts/fun/off_topic_names.py index 7fc93b88c..845b8175c 100644 --- a/bot/exts/fun/off_topic_names.py +++ b/bot/exts/fun/off_topic_names.py @@ -139,10 +139,20 @@ class OffTopicNames(Cog):      @has_any_role(*MODERATION_ROLES)      async def search_command(self, ctx: Context, *, query: OffTopicName) -> None:          """Search for an off-topic name.""" -        result = await self.bot.api_client.get('bot/off-topic-channel-names') -        in_matches = {name for name in result if query in name} -        close_matches = difflib.get_close_matches(query, result, n=10, cutoff=0.70) -        lines = sorted(f"• {name}" for name in in_matches.union(close_matches)) +        query = OffTopicName.translate_name(query, from_unicode=False).lower() + +        # Map normalized names to returned names for search purposes +        result = { +            OffTopicName.translate_name(name, from_unicode=False).lower(): name +            for name in await self.bot.api_client.get('bot/off-topic-channel-names') +        } + +        # Search normalized keys +        in_matches = {name for name in result.keys() if query in name} +        close_matches = difflib.get_close_matches(query, result.keys(), n=10, cutoff=0.70) + +        # Send Results +        lines = sorted(f"• {result[name]}" for name in in_matches.union(close_matches))          embed = Embed(              title="Query results",              colour=Colour.blue() diff --git a/bot/exts/info/information.py b/bot/exts/info/information.py index 4499e4c25..92ddf0fbd 100644 --- a/bot/exts/info/information.py +++ b/bot/exts/info/information.py @@ -17,7 +17,7 @@ from bot.decorators import in_whitelist  from bot.pagination import LinePaginator  from bot.utils.channel import is_mod_channel, is_staff_channel  from bot.utils.checks import cooldown_with_role_bypass, has_no_roles_check, in_whitelist_check -from bot.utils.time import time_since +from bot.utils.time import humanize_delta, time_since  log = logging.getLogger(__name__) @@ -52,7 +52,7 @@ class Information(Cog):          )          return {role.name.title(): len(role.members) for role in roles} -    def get_extended_server_info(self) -> str: +    def get_extended_server_info(self, ctx: Context) -> str:          """Return additional server info only visible in moderation channels."""          talentpool_info = ""          if cog := self.bot.get_cog("Talentpool"): @@ -64,9 +64,9 @@ class Information(Cog):          defcon_info = ""          if cog := self.bot.get_cog("Defcon"): -            defcon_status = "Enabled" if cog.enabled else "Disabled" -            defcon_days = cog.days.days if cog.enabled else "-" -            defcon_info = f"Defcon status: {defcon_status}\nDefcon days: {defcon_days}\n" +            defcon_info = f"Defcon threshold: {humanize_delta(cog.threshold)}\n" + +        verification = f"Verification level: {ctx.guild.verification_level.name}\n"          python_general = self.bot.get_channel(constants.Channels.python_general) @@ -74,6 +74,7 @@ class Information(Cog):              {talentpool_info}\              {bb_info}\              {defcon_info}\ +            {verification}\              {python_general.mention} cooldown: {python_general.slowmode_delay}s          """) @@ -198,11 +199,11 @@ class Information(Cog):          # Additional info if ran in moderation channels          if is_mod_channel(ctx.channel): -            embed.add_field(name="Moderation:", value=self.get_extended_server_info()) +            embed.add_field(name="Moderation:", value=self.get_extended_server_info(ctx))          await ctx.send(embed=embed) -    @command(name="user", aliases=["user_info", "member", "member_info"]) +    @command(name="user", aliases=["user_info", "member", "member_info", "u"])      async def user_info(self, ctx: Context, user: FetchedMember = None) -> None:          """Returns info about a user."""          if user is None: diff --git a/bot/exts/info/tags.py b/bot/exts/info/tags.py index 00b4d1a78..bb91a8563 100644 --- a/bot/exts/info/tags.py +++ b/bot/exts/info/tags.py @@ -189,7 +189,7 @@ class Tags(Cog):          If a tag is not specified, display a paginated embed of all tags.          Tags are on cooldowns on a per-tag, per-channel basis. If a tag is on cooldown, display -        nothing and return False. +        nothing and return True.          """          def _command_on_cooldown(tag_name: str) -> bool:              """ @@ -217,7 +217,7 @@ class Tags(Cog):                  f"{ctx.author} tried to get the '{tag_name}' tag, but the tag is on cooldown. "                  f"Cooldown ends in {time_left:.1f} seconds."              ) -            return False +            return True          if tag_name is not None:              temp_founds = self._get_tag(tag_name) @@ -285,7 +285,8 @@ class Tags(Cog):          """          Get a specified tag, or a list of all tags if no tag is specified. -        Returns False if a tag is on cooldown, or if no matches are found. +        Returns True if something can be sent, or if the tag is on cooldown. +        Returns False if no matches are found.          """          return await self.display_tag(ctx, tag_name) diff --git a/bot/exts/moderation/defcon.py b/bot/exts/moderation/defcon.py index caa6fb917..bd16289b9 100644 --- a/bot/exts/moderation/defcon.py +++ b/bot/exts/moderation/defcon.py @@ -1,17 +1,25 @@ -from __future__ import annotations - +import asyncio  import logging +import traceback  from collections import namedtuple -from datetime import datetime, timedelta +from datetime import datetime  from enum import Enum +from typing import Optional, Union -from discord import Colour, Embed, Member +from aioredis import RedisError +from async_rediscache import RedisCache +from dateutil.relativedelta import relativedelta +from discord import Colour, Embed, Member, User +from discord.ext import tasks  from discord.ext.commands import Cog, Context, group, has_any_role  from bot.bot import Bot  from bot.constants import Channels, Colours, Emojis, Event, Icons, MODERATION_ROLES, Roles +from bot.converters import DurationDelta, Expiry  from bot.exts.moderation.modlog import ModLog  from bot.utils.messages import format_user +from bot.utils.scheduling import Scheduler +from bot.utils.time import humanize_delta, parse_duration_string, relativedelta_to_timedelta  log = logging.getLogger(__name__) @@ -28,71 +36,81 @@ will be resolved soon. In the meantime, please feel free to peruse the resources  BASE_CHANNEL_TOPIC = "Python Discord Defense Mechanism" +SECONDS_IN_DAY = 86400 +  class Action(Enum):      """Defcon Action.""" -    ActionInfo = namedtuple('LogInfoDetails', ['icon', 'color', 'template']) +    ActionInfo = namedtuple('LogInfoDetails', ['icon', 'emoji', 'color', 'template']) -    ENABLED = ActionInfo(Icons.defcon_enabled, Colours.soft_green, "**Days:** {days}\n\n") -    DISABLED = ActionInfo(Icons.defcon_disabled, Colours.soft_red, "") -    UPDATED = ActionInfo(Icons.defcon_updated, Colour.blurple(), "**Days:** {days}\n\n") +    SERVER_OPEN = ActionInfo(Icons.defcon_unshutdown, Emojis.defcon_unshutdown, Colours.soft_green, "") +    SERVER_SHUTDOWN = ActionInfo(Icons.defcon_shutdown, Emojis.defcon_shutdown, Colours.soft_red, "") +    DURATION_UPDATE = ActionInfo( +        Icons.defcon_update, Emojis.defcon_update, Colour.blurple(), "**Threshold:** {threshold}\n\n" +    )  class Defcon(Cog):      """Time-sensitive server defense mechanisms.""" -    days = None  # type: timedelta -    enabled = False  # type: bool +    # RedisCache[str, str] +    # The cache's keys are "threshold" and "expiry". +    # The caches' values are strings formatted as valid input to the DurationDelta converter, or empty when off. +    defcon_settings = RedisCache()      def __init__(self, bot: Bot):          self.bot = bot          self.channel = None -        self.days = timedelta(days=0) +        self.threshold = relativedelta(days=0) +        self.expiry = None + +        self.scheduler = Scheduler(self.__class__.__name__) -        self.bot.loop.create_task(self.sync_settings()) +        self.bot.loop.create_task(self._sync_settings())      @property      def mod_log(self) -> ModLog:          """Get currently loaded ModLog cog instance."""          return self.bot.get_cog("ModLog") -    async def sync_settings(self) -> None: +    @defcon_settings.atomic_transaction +    async def _sync_settings(self) -> None:          """On cog load, try to synchronize DEFCON settings to the API.""" +        log.trace("Waiting for the guild to become available before syncing.")          await self.bot.wait_until_guild_available()          self.channel = await self.bot.fetch_channel(Channels.defcon) -        try: -            response = await self.bot.api_client.get('bot/bot-settings/defcon') -            data = response['data'] +        log.trace("Syncing settings.") -        except Exception:  # Yikes! +        try: +            settings = await self.defcon_settings.to_dict() +            self.threshold = parse_duration_string(settings["threshold"]) if settings.get("threshold") else None +            self.expiry = datetime.fromisoformat(settings["expiry"]) if settings.get("expiry") else None +        except RedisError:              log.exception("Unable to get DEFCON settings!") -            await self.bot.get_channel(Channels.dev_log).send( -                f"<@&{Roles.admins}> **WARNING**: Unable to get DEFCON settings!" +            await self.channel.send( +                f"<@&{Roles.moderators}> <@&{Roles.devops}> **WARNING**: Unable to get DEFCON settings!" +                f"\n\n```{traceback.format_exc()}```"              )          else: -            if data["enabled"]: -                self.enabled = True -                self.days = timedelta(days=data["days"]) -                log.info(f"DEFCON enabled: {self.days.days} days") +            if self.expiry: +                self.scheduler.schedule_at(self.expiry, 0, self._remove_threshold()) -            else: -                self.enabled = False -                self.days = timedelta(days=0) -                log.info("DEFCON disabled") +            self._update_notifier() +            log.info(f"DEFCON synchronized: {humanize_delta(self.threshold) if self.threshold else '-'}") -            await self.update_channel_topic() +        self._update_channel_topic()      @Cog.listener()      async def on_member_join(self, member: Member) -> None: -        """If DEFCON is enabled, check newly joining users to see if they meet the account age threshold.""" -        if self.enabled and self.days.days > 0: +        """Check newly joining users to see if they meet the account age threshold.""" +        if self.threshold:              now = datetime.utcnow() -            if now - member.created_at < self.days: -                log.info(f"Rejecting user {member}: Account is too new and DEFCON is enabled") +            if now - member.created_at < relativedelta_to_timedelta(self.threshold): +                log.info(f"Rejecting user {member}: Account is too new")                  message_sent = False @@ -124,134 +142,163 @@ class Defcon(Cog):          """Check the DEFCON status or run a subcommand."""          await ctx.send_help(ctx.command) -    async def _defcon_action(self, ctx: Context, days: int, action: Action) -> None: -        """Providing a structured way to do an defcon action.""" -        try: -            response = await self.bot.api_client.get('bot/bot-settings/defcon') -            data = response['data'] - -            if "enable_date" in data and action is Action.DISABLED: -                enabled = datetime.fromisoformat(data["enable_date"]) - -                delta = datetime.now() - enabled - -                self.bot.stats.timing("defcon.enabled", delta) -        except Exception: -            pass - -        error = None -        try: -            await self.bot.api_client.put( -                'bot/bot-settings/defcon', -                json={ -                    'name': 'defcon', -                    'data': { -                        # TODO: retrieve old days count -                        'days': days, -                        'enabled': action is not Action.DISABLED, -                        'enable_date': datetime.now().isoformat() -                    } -                } -            ) -        except Exception as err: -            log.exception("Unable to update DEFCON settings.") -            error = err -        finally: -            await ctx.send(self.build_defcon_msg(action, error)) -            await self.send_defcon_log(action, ctx.author, error) - -            self.bot.stats.gauge("defcon.threshold", days) - -    @defcon_group.command(name='enable', aliases=('on', 'e'), root_aliases=("defon",)) -    @has_any_role(*MODERATION_ROLES) -    async def enable_command(self, ctx: Context) -> None: -        """ -        Enable DEFCON mode. Useful in a pinch, but be sure you know what you're doing! - -        Currently, this just adds an account age requirement. Use !defcon days <int> to set how old an account must be, -        in days. -        """ -        self.enabled = True -        await self._defcon_action(ctx, days=0, action=Action.ENABLED) -        await self.update_channel_topic() - -    @defcon_group.command(name='disable', aliases=('off', 'd'), root_aliases=("defoff",)) -    @has_any_role(*MODERATION_ROLES) -    async def disable_command(self, ctx: Context) -> None: -        """Disable DEFCON mode. Useful in a pinch, but be sure you know what you're doing!""" -        self.enabled = False -        await self._defcon_action(ctx, days=0, action=Action.DISABLED) -        await self.update_channel_topic() - -    @defcon_group.command(name='status', aliases=('s',)) +    @defcon_group.command(aliases=('s',))      @has_any_role(*MODERATION_ROLES) -    async def status_command(self, ctx: Context) -> None: +    async def status(self, ctx: Context) -> None:          """Check the current status of DEFCON mode."""          embed = Embed(              colour=Colour.blurple(), title="DEFCON Status", -            description=f"**Enabled:** {self.enabled}\n" -                        f"**Days:** {self.days.days}" +            description=f""" +                **Threshold:** {humanize_delta(self.threshold) if self.threshold else "-"} +                **Expires in:** {humanize_delta(relativedelta(self.expiry, datetime.utcnow())) if self.expiry else "-"} +                **Verification level:** {ctx.guild.verification_level.name} +                """          )          await ctx.send(embed=embed) -    @defcon_group.command(name='days') +    @defcon_group.command(aliases=('t', 'd'))      @has_any_role(*MODERATION_ROLES) -    async def days_command(self, ctx: Context, days: int) -> None: -        """Set how old an account must be to join the server, in days, with DEFCON mode enabled.""" -        self.days = timedelta(days=days) -        self.enabled = True -        await self._defcon_action(ctx, days=days, action=Action.UPDATED) -        await self.update_channel_topic() - -    async def update_channel_topic(self) -> None: +    async def threshold( +        self, ctx: Context, threshold: Union[DurationDelta, int], expiry: Optional[Expiry] = None +    ) -> None: +        """ +        Set how old an account must be to join the server. + +        The threshold is the minimum required account age. Can accept either a duration string or a number of days. +        Set it to 0 to have no threshold. +        The expiry allows to automatically remove the threshold after a designated time. If no expiry is specified, +        the cog will remind to remove the threshold hourly. +        """ +        if isinstance(threshold, int): +            threshold = relativedelta(days=threshold) +        await self._update_threshold(ctx.author, threshold=threshold, expiry=expiry) + +    @defcon_group.command() +    @has_any_role(Roles.admins) +    async def shutdown(self, ctx: Context) -> None: +        """Shut down the server by setting send permissions of everyone to False.""" +        role = ctx.guild.default_role +        permissions = role.permissions + +        permissions.update(send_messages=False, add_reactions=False) +        await role.edit(reason="DEFCON shutdown", permissions=permissions) +        await ctx.send(f"{Action.SERVER_SHUTDOWN.value.emoji} Server shut down.") + +    @defcon_group.command() +    @has_any_role(Roles.admins) +    async def unshutdown(self, ctx: Context) -> None: +        """Open up the server again by setting send permissions of everyone to None.""" +        role = ctx.guild.default_role +        permissions = role.permissions + +        permissions.update(send_messages=True, add_reactions=True) +        await role.edit(reason="DEFCON unshutdown", permissions=permissions) +        await ctx.send(f"{Action.SERVER_OPEN.value.emoji} Server reopened.") + +    def _update_channel_topic(self) -> None:          """Update the #defcon channel topic with the current DEFCON status.""" -        if self.enabled: -            day_str = "days" if self.days.days > 1 else "day" -            new_topic = f"{BASE_CHANNEL_TOPIC}\n(Status: Enabled, Threshold: {self.days.days} {day_str})" -        else: -            new_topic = f"{BASE_CHANNEL_TOPIC}\n(Status: Disabled)" +        new_topic = f"{BASE_CHANNEL_TOPIC}\n(Threshold: {humanize_delta(self.threshold) if self.threshold else '-'})"          self.mod_log.ignore(Event.guild_channel_update, Channels.defcon) -        await self.channel.edit(topic=new_topic) - -    def build_defcon_msg(self, action: Action, e: Exception = None) -> str: -        """Build in-channel response string for DEFCON action.""" -        if action is Action.ENABLED: -            msg = f"{Emojis.defcon_enabled} DEFCON enabled.\n\n" -        elif action is Action.DISABLED: -            msg = f"{Emojis.defcon_disabled} DEFCON disabled.\n\n" -        elif action is Action.UPDATED: -            msg = ( -                f"{Emojis.defcon_updated} DEFCON days updated; accounts must be {self.days.days} " -                f"day{'s' if self.days.days > 1 else ''} old to join the server.\n\n" +        asyncio.create_task(self.channel.edit(topic=new_topic)) + +    @defcon_settings.atomic_transaction +    async def _update_threshold(self, author: User, threshold: relativedelta, expiry: Optional[Expiry] = None) -> None: +        """Update the new threshold in the cog, cache, defcon channel, and logs, and additionally schedule expiry.""" +        self.threshold = threshold +        if threshold == relativedelta(days=0):  # If the threshold is 0, we don't need to schedule anything +            expiry = None +        self.expiry = expiry + +        # Either way, we cancel the old task. +        self.scheduler.cancel_all() +        if self.expiry is not None: +            self.scheduler.schedule_at(expiry, 0, self._remove_threshold()) + +        self._update_notifier() + +        # Make sure to handle the critical part of the update before writing to Redis. +        error = "" +        try: +            await self.defcon_settings.update( +                { +                    'threshold': Defcon._stringify_relativedelta(self.threshold) if self.threshold else "", +                    'expiry': expiry.isoformat() if expiry else 0 +                }              ) +        except RedisError: +            error = ", but failed to write to cache" + +        action = Action.DURATION_UPDATE -        if e: -            msg += ( -                "**There was a problem updating the site** - This setting may be reverted when the bot restarts.\n\n" -                f"```py\n{e}\n```" +        expiry_message = "" +        if expiry: +            expiry_message = f" for the next {humanize_delta(relativedelta(expiry, datetime.utcnow()), max_units=2)}" + +        if self.threshold: +            channel_message = ( +                f"updated; accounts must be {humanize_delta(self.threshold)} " +                f"old to join the server{expiry_message}"              ) +        else: +            channel_message = "removed" + +        await self.channel.send( +            f"{action.value.emoji} DEFCON threshold {channel_message}{error}." +        ) +        await self._send_defcon_log(action, author) +        self._update_channel_topic() + +        self._log_threshold_stat(threshold) -        return msg +    async def _remove_threshold(self) -> None: +        """Resets the threshold back to 0.""" +        await self._update_threshold(self.bot.user, relativedelta(days=0)) -    async def send_defcon_log(self, action: Action, actor: Member, e: Exception = None) -> None: +    @staticmethod +    def _stringify_relativedelta(delta: relativedelta) -> str: +        """Convert a relativedelta object to a duration string.""" +        units = [("years", "y"), ("months", "m"), ("days", "d"), ("hours", "h"), ("minutes", "m"), ("seconds", "s")] +        return "".join(f"{getattr(delta, unit)}{symbol}" for unit, symbol in units if getattr(delta, unit)) or "0s" + +    def _log_threshold_stat(self, threshold: relativedelta) -> None: +        """Adds the threshold to the bot stats in days.""" +        threshold_days = relativedelta_to_timedelta(threshold).total_seconds() / SECONDS_IN_DAY +        self.bot.stats.gauge("defcon.threshold", threshold_days) + +    async def _send_defcon_log(self, action: Action, actor: User) -> None:          """Send log message for DEFCON action."""          info = action.value          log_msg: str = (              f"**Staffer:** {actor.mention} {actor} (`{actor.id}`)\n" -            f"{info.template.format(days=self.days.days)}" +            f"{info.template.format(threshold=(humanize_delta(self.threshold) if self.threshold else '-'))}"          )          status_msg = f"DEFCON {action.name.lower()}" -        if e: -            log_msg += ( -                "**There was a problem updating the site** - This setting may be reverted when the bot restarts.\n\n" -                f"```py\n{e}\n```" -            ) -          await self.mod_log.send_log_message(info.icon, info.color, status_msg, log_msg) +    def _update_notifier(self) -> None: +        """Start or stop the notifier according to the DEFCON status.""" +        if self.threshold and self.expiry is None and not self.defcon_notifier.is_running(): +            log.info("DEFCON notifier started.") +            self.defcon_notifier.start() + +        elif (not self.threshold or self.expiry is not None) and self.defcon_notifier.is_running(): +            log.info("DEFCON notifier stopped.") +            self.defcon_notifier.cancel() + +    @tasks.loop(hours=1) +    async def defcon_notifier(self) -> None: +        """Routinely notify moderators that DEFCON is active.""" +        await self.channel.send(f"Defcon is on and is set to {humanize_delta(self.threshold)}.") + +    def cog_unload(self) -> None: +        """Cancel the notifer and threshold removal tasks when the cog unloads.""" +        log.trace("Cog unload: canceling defcon notifier task.") +        self.defcon_notifier.cancel() +        self.scheduler.cancel_all() +  def setup(bot: Bot) -> None:      """Load the Defcon cog.""" diff --git a/bot/exts/moderation/infraction/infractions.py b/bot/exts/moderation/infraction/infractions.py index 7349d65f2..3b5b1df45 100644 --- a/bot/exts/moderation/infraction/infractions.py +++ b/bot/exts/moderation/infraction/infractions.py @@ -126,7 +126,7 @@ class Infractions(InfractionScheduler, commands.Cog):              duration = await Duration().convert(ctx, "1h")          await self.apply_mute(ctx, user, reason, expires_at=duration) -    @command() +    @command(aliases=("tban",))      async def tempban(          self,          ctx: Context, @@ -198,7 +198,7 @@ class Infractions(InfractionScheduler, commands.Cog):      # endregion      # region: Temporary shadow infractions -    @command(hidden=True, aliases=["shadowtempban", "stempban"]) +    @command(hidden=True, aliases=["shadowtempban", "stempban", "stban"])      async def shadow_tempban(          self,          ctx: Context, diff --git a/bot/exts/moderation/infraction/superstarify.py b/bot/exts/moderation/infraction/superstarify.py index ffc470c54..704dddf9c 100644 --- a/bot/exts/moderation/infraction/superstarify.py +++ b/bot/exts/moderation/infraction/superstarify.py @@ -104,7 +104,7 @@ class Superstarify(InfractionScheduler, Cog):              await self.reapply_infraction(infraction, action) -    @command(name="superstarify", aliases=("force_nick", "star", "starify")) +    @command(name="superstarify", aliases=("force_nick", "star", "starify", "superstar"))      async def superstarify(          self,          ctx: Context, @@ -183,7 +183,7 @@ class Superstarify(InfractionScheduler, Cog):              )              await ctx.send(embed=embed) -    @command(name="unsuperstarify", aliases=("release_nick", "unstar", "unstarify")) +    @command(name="unsuperstarify", aliases=("release_nick", "unstar", "unstarify", "unsuperstar"))      async def unsuperstarify(self, ctx: Context, member: Member) -> None:          """Remove the superstarify infraction and allow the user to change their nickname."""          await self.pardon_infraction(ctx, "superstar", member) diff --git a/bot/exts/moderation/slowmode.py b/bot/exts/moderation/slowmode.py index c449752e1..d8baff76a 100644 --- a/bot/exts/moderation/slowmode.py +++ b/bot/exts/moderation/slowmode.py @@ -1,5 +1,4 @@  import logging -from datetime import datetime  from typing import Optional  from dateutil.relativedelta import relativedelta @@ -54,8 +53,7 @@ class Slowmode(Cog):          # Convert `dateutil.relativedelta.relativedelta` to `datetime.timedelta`          # Must do this to get the delta in a particular unit of time -        utcnow = datetime.utcnow() -        slowmode_delay = (utcnow + delay - utcnow).total_seconds() +        slowmode_delay = time.relativedelta_to_timedelta(delay).total_seconds()          humanized_delay = time.humanize_delta(delay) diff --git a/bot/exts/utils/internal.py b/bot/exts/utils/internal.py index a7ab43f37..6f2da3131 100644 --- a/bot/exts/utils/internal.py +++ b/bot/exts/utils/internal.py @@ -240,12 +240,12 @@ async def func():  # (None,) -> Any          stats_embed = discord.Embed(              title="WebSocket statistics", -            description=f"Receiving {per_s:0.2f} event per second.", +            description=f"Receiving {per_s:0.2f} events per second.",              color=discord.Color.blurple()          )          for event_type, count in self.socket_events.most_common(25): -            stats_embed.add_field(name=event_type, value=count, inline=False) +            stats_embed.add_field(name=event_type, value=f"{count:,}", inline=True)          await ctx.send(embed=stats_embed) diff --git a/bot/log.py b/bot/log.py index e92233a33..bc3bba0af 100644 --- a/bot/log.py +++ b/bot/log.py @@ -1,11 +1,12 @@  import logging  import os  import sys -from logging import Logger, handlers +from logging import Logger, StreamHandler, handlers  from pathlib import Path  import coloredlogs  import sentry_sdk +from pythonjsonlogger import jsonlogger  from sentry_sdk.integrations.logging import LoggingIntegration  from sentry_sdk.integrations.redis import RedisIntegration @@ -13,6 +14,15 @@ from bot import constants  TRACE_LEVEL = 5 +PROD_FIELDS = [ +    "asctime", +    "name", +    "levelname", +    "message", +    "funcName", +    "filename" +] +  def setup() -> None:      """Set up loggers.""" @@ -33,21 +43,28 @@ def setup() -> None:      root_log.setLevel(log_level)      root_log.addHandler(file_handler) -    if "COLOREDLOGS_LEVEL_STYLES" not in os.environ: -        coloredlogs.DEFAULT_LEVEL_STYLES = { -            **coloredlogs.DEFAULT_LEVEL_STYLES, -            "trace": {"color": 246}, -            "critical": {"background": "red"}, -            "debug": coloredlogs.DEFAULT_LEVEL_STYLES["info"] -        } - -    if "COLOREDLOGS_LOG_FORMAT" not in os.environ: -        coloredlogs.DEFAULT_LOG_FORMAT = format_string - -    if "COLOREDLOGS_LOG_LEVEL" not in os.environ: -        coloredlogs.DEFAULT_LOG_LEVEL = log_level - -    coloredlogs.install(logger=root_log, stream=sys.stdout) +    if constants.DEBUG_MODE: +        if "COLOREDLOGS_LEVEL_STYLES" not in os.environ: +            coloredlogs.DEFAULT_LEVEL_STYLES = { +                **coloredlogs.DEFAULT_LEVEL_STYLES, +                "trace": {"color": 246}, +                "critical": {"background": "red"}, +                "debug": coloredlogs.DEFAULT_LEVEL_STYLES["info"] +            } + +        if "COLOREDLOGS_LOG_FORMAT" not in os.environ: +            coloredlogs.DEFAULT_LOG_FORMAT = format_string + +        if "COLOREDLOGS_LOG_LEVEL" not in os.environ: +            coloredlogs.DEFAULT_LOG_LEVEL = log_level + +        coloredlogs.install(logger=root_log, stream=sys.stdout) +    else: +        json_format = " ".join([f"%({field})s" for field in PROD_FIELDS]) +        stream_handler = StreamHandler() +        formatter = jsonlogger.JsonFormatter(json_format) +        stream_handler.setFormatter(formatter) +        root_log.addHandler(stream_handler)      logging.getLogger("discord").setLevel(logging.WARNING)      logging.getLogger("websockets").setLevel(logging.WARNING) diff --git a/bot/resources/tags/comparison.md b/bot/resources/tags/comparison.md new file mode 100644 index 000000000..12844bd2f --- /dev/null +++ b/bot/resources/tags/comparison.md @@ -0,0 +1,12 @@ +**Assignment vs. Comparison** + +The assignment operator (`=`) is used to assign variables. +```python +x = 5 +print(x)  # Prints 5 +``` +The equality operator (`==`) is used to compare values. +```python +if x == 5: +    print("The value of x is 5") +``` diff --git a/bot/resources/tags/empty-json.md b/bot/resources/tags/empty-json.md new file mode 100644 index 000000000..935544bb7 --- /dev/null +++ b/bot/resources/tags/empty-json.md @@ -0,0 +1,11 @@ +When using JSON, you might run into the following error: +``` +JSONDecodeError: Expecting value: line 1 column 1 (char 0) +``` +This error could have appeared because you just created the JSON file and there is nothing in it at the moment. + +Whilst having empty data is no problem, the file itself may never be completely empty. + +You most likely wanted to structure your JSON as a dictionary. To do this, edit your empty JSON file so that it instead contains `{}`. + +Different data types are also supported. If you wish to read more on these, please refer to [this article](https://www.tutorialspoint.com/json/json_data_types.htm). diff --git a/bot/resources/tags/inline.md b/bot/resources/tags/inline.md index a6a7c35d6..4ece74ef7 100644 --- a/bot/resources/tags/inline.md +++ b/bot/resources/tags/inline.md @@ -1,16 +1,7 @@  **Inline codeblocks** -In addition to multi-line codeblocks, discord has support for inline codeblocks as well. These are small codeblocks that are usually a single line, that can fit between non-codeblocks on the same line. +Inline codeblocks look `like this`. To create them you surround text with single backticks, so \`hello\` would become `hello`. -The following is an example of how it's done: +Note that backticks are not quotes, see [this](https://superuser.com/questions/254076/how-do-i-type-the-tick-and-backtick-characters-on-windows/254077#254077) if you are struggling to find the backtick key. -The \`\_\_init\_\_\` method customizes the newly created instance. - -And results in the following: - -The `__init__` method customizes the newly created instance. - -**Note:**   -• These are **backticks** not quotes   -• Avoid using them for multiple lines   -• Useful for negating formatting you don't want +For how to make multiline codeblocks see the `!codeblock` tag. diff --git a/bot/resources/tags/off-topic.md b/bot/resources/tags/off-topic.md index c7f98a813..6a864a1d5 100644 --- a/bot/resources/tags/off-topic.md +++ b/bot/resources/tags/off-topic.md @@ -6,3 +6,5 @@ There are three off-topic channels:  • <#463035268514185226>    Their names change randomly every 24 hours, but you can always find them under the `OFF-TOPIC/GENERAL` category in the channel list. + +Please read our [off-topic etiquette](https://pythondiscord.com/pages/resources/guides/off-topic-etiquette/) before participating in conversations. diff --git a/bot/resources/tags/pep8.md b/bot/resources/tags/pep8.md index cab4c4db8..57b176122 100644 --- a/bot/resources/tags/pep8.md +++ b/bot/resources/tags/pep8.md @@ -1,3 +1,5 @@ -**PEP 8** is the official style guide for Python. It includes comprehensive guidelines for code formatting, variable naming, and making your code easy to read. Professional Python developers are usually required to follow the guidelines, and will often use code-linters like `flake8` to verify that the code they\'re writing complies with the style guide. +**PEP 8** is the official style guide for Python. It includes comprehensive guidelines for code formatting, variable naming, and making your code easy to read. Professional Python developers are usually required to follow the guidelines, and will often use code-linters like flake8 to verify that the code they're writing complies with the style guide. -You can find the PEP 8 document [here](https://www.python.org/dev/peps/pep-0008). +More information: +• [PEP 8 document](https://www.python.org/dev/peps/pep-0008) +• [Our PEP 8 song!](https://www.youtube.com/watch?v=hgI0p1zf31k) :notes: diff --git a/bot/utils/time.py b/bot/utils/time.py index 47e49904b..f862e40f7 100644 --- a/bot/utils/time.py +++ b/bot/utils/time.py @@ -1,5 +1,6 @@  import asyncio  import datetime +import re  from typing import Optional  import dateutil.parser @@ -8,6 +9,16 @@ from dateutil.relativedelta import relativedelta  RFC1123_FORMAT = "%a, %d %b %Y %H:%M:%S GMT"  INFRACTION_FORMAT = "%Y-%m-%d %H:%M" +_DURATION_REGEX = re.compile( +    r"((?P<years>\d+?) ?(years|year|Y|y) ?)?" +    r"((?P<months>\d+?) ?(months|month|m) ?)?" +    r"((?P<weeks>\d+?) ?(weeks|week|W|w) ?)?" +    r"((?P<days>\d+?) ?(days|day|D|d) ?)?" +    r"((?P<hours>\d+?) ?(hours|hour|H|h) ?)?" +    r"((?P<minutes>\d+?) ?(minutes|minute|M) ?)?" +    r"((?P<seconds>\d+?) ?(seconds|second|S|s))?" +) +  def _stringify_time_unit(value: int, unit: str) -> str:      """ @@ -74,6 +85,37 @@ def humanize_delta(delta: relativedelta, precision: str = "seconds", max_units:      return humanized +def parse_duration_string(duration: str) -> Optional[relativedelta]: +    """ +    Converts a `duration` string to a relativedelta object. + +    The function supports the following symbols for each unit of time: +    - years: `Y`, `y`, `year`, `years` +    - months: `m`, `month`, `months` +    - weeks: `w`, `W`, `week`, `weeks` +    - days: `d`, `D`, `day`, `days` +    - hours: `H`, `h`, `hour`, `hours` +    - minutes: `M`, `minute`, `minutes` +    - seconds: `S`, `s`, `second`, `seconds` +    The units need to be provided in descending order of magnitude. +    If the string does represent a durationdelta object, it will return None. +    """ +    match = _DURATION_REGEX.fullmatch(duration) +    if not match: +        return None + +    duration_dict = {unit: int(amount) for unit, amount in match.groupdict(default=0).items()} +    delta = relativedelta(**duration_dict) + +    return delta + + +def relativedelta_to_timedelta(delta: relativedelta) -> datetime.timedelta: +    """Converts a relativedelta object to a timedelta object.""" +    utcnow = datetime.datetime.utcnow() +    return utcnow + delta - utcnow + +  def time_since(past_datetime: datetime.datetime, precision: str = "seconds", max_units: int = 6) -> str:      """      Takes a datetime and returns a human-readable string that describes how long ago that datetime was. diff --git a/config-default.yml b/config-default.yml index 7d9afaa0e..18d9cd370 100644 --- a/config-default.yml +++ b/config-default.yml @@ -47,9 +47,9 @@ style:          badge_staff: "<:discord_staff:743882896498098226>"          badge_verified_bot_developer: "<:verified_bot_dev:743882897299210310>" -        defcon_disabled: "<:defcondisabled:470326273952972810>" -        defcon_enabled:  "<:defconenabled:470326274213150730>" -        defcon_updated:  "<:defconsettingsupdated:470326274082996224>" +        defcon_shutdown:    "<:defcondisabled:470326273952972810>" +        defcon_unshutdown:  "<:defconenabled:470326274213150730>" +        defcon_update:      "<:defconsettingsupdated:470326274082996224>"          failmail: "<:failmail:633660039931887616>" @@ -83,9 +83,9 @@ style:          crown_red:     "https://cdn.discordapp.com/emojis/469964154879344640.png"          defcon_denied:   "https://cdn.discordapp.com/emojis/472475292078964738.png" -        defcon_disabled: "https://cdn.discordapp.com/emojis/470326273952972810.png" -        defcon_enabled:  "https://cdn.discordapp.com/emojis/470326274213150730.png" -        defcon_updated:  "https://cdn.discordapp.com/emojis/472472638342561793.png" +        defcon_shutdown: "https://cdn.discordapp.com/emojis/470326273952972810.png" +        defcon_unshutdown:  "https://cdn.discordapp.com/emojis/470326274213150730.png" +        defcon_update:  "https://cdn.discordapp.com/emojis/472472638342561793.png"          filtering: "https://cdn.discordapp.com/emojis/472472638594482195.png" @@ -261,6 +261,7 @@ guild:          # Staff          admins:             &ADMINS_ROLE    267628507062992896          core_developers:                    587606783669829632 +        devops:                             409416496733880320          helpers:            &HELPERS_ROLE   267630620367257601          moderators:         &MODS_ROLE      267629731250176001          owners:             &OWNERS_ROLE    267627879762755584 @@ -338,6 +339,8 @@ keys:  urls:      # PyDis site vars +    connect_max_retries:       3 +    connect_cooldown:          5      site:        &DOMAIN       "pythondiscord.com"      site_api:    &API          "pydis-api.default.svc.cluster.local"      site_api_schema:           "http://" | 
