Compare commits

..

14 Commits

Author SHA1 Message Date
f4bd33b4d5 Fix linting on Game
Also default guess value to 0
2022-10-24 20:53:51 -05:00
545e0442eb Fix linting on database models (except todo) 2022-10-24 20:31:47 -05:00
e0dc0e6b78 Fix linting issues with client 2022-10-24 20:22:04 -05:00
b154efb6cc Fix linting issues with main 2022-10-24 20:06:50 -05:00
57a808f19d Create lint log 2022-10-24 20:05:21 -05:00
b32d76bf16 The decorator should be outside of the class 2022-10-24 20:02:26 -05:00
07ecd2248c Format with Black 2022-10-24 20:00:39 -05:00
7d684d8b9d Install pylint and black
for formatting and ensuring the code conforms to PEP standards
2022-10-24 19:59:18 -05:00
1845e42488 Use the with statement on the game class
This is because the class is a context manager / state machine
2022-10-24 19:55:50 -05:00
7101db5394 We'll be fine with one database connection, and don't have to repeat ourselves 2022-10-24 19:13:34 -05:00
49d3672936 Add extra time tracking 2022-10-24 19:12:48 -05:00
8ab3cd611f These need to be named properly 2022-10-24 18:49:26 -05:00
e771ea8764 Ignore databases 2022-10-24 18:47:12 -05:00
321fa9ae7c It's an Integer 2022-10-24 18:43:38 -05:00
8 changed files with 412 additions and 78 deletions

2
.gitignore vendored
View File

@ -1,2 +1,4 @@
.idea/* .idea/*
__pycache__/ __pycache__/
*.db
*.log

View File

@ -7,6 +7,8 @@ name = "pypi"
discord = "*" discord = "*"
peewee = "*" peewee = "*"
dateparser = "*" dateparser = "*"
pylint = "*"
black = "*"
[dev-packages] [dev-packages]

View File

@ -1,7 +1,7 @@
{ {
"_meta": { "_meta": {
"hash": { "hash": {
"sha256": "051e0e9b85a0d5118c0feb3c2c44aaa4b2a2178966685b0bdef0b5f5816db308" "sha256": "0f1a335d8076579fbcc256d8fdcfedb582793218af78b4ccbdbc5572e32c25dd"
}, },
"pipfile-spec": 6, "pipfile-spec": 6,
"requires": { "requires": {
@ -117,6 +117,14 @@
"markers": "python_version >= '3.6'", "markers": "python_version >= '3.6'",
"version": "==1.2.0" "version": "==1.2.0"
}, },
"astroid": {
"hashes": [
"sha256:1c00a14f5a3ed0339d38d2e2e5b74ea2591df5861c0936bb292b84ccf3a78d83",
"sha256:72702205200b2a638358369d90c222d74ebc376787af8fb2f7f2a86f7b5cc85f"
],
"markers": "python_full_version >= '3.7.2'",
"version": "==2.12.12"
},
"async-timeout": { "async-timeout": {
"hashes": [ "hashes": [
"sha256:2163e1640ddb52b7a8c80d0a67a08587e5d245cc9c553a74a847056bc2976b15", "sha256:2163e1640ddb52b7a8c80d0a67a08587e5d245cc9c553a74a847056bc2976b15",
@ -133,6 +141,33 @@
"markers": "python_version >= '3.5'", "markers": "python_version >= '3.5'",
"version": "==22.1.0" "version": "==22.1.0"
}, },
"black": {
"hashes": [
"sha256:14ff67aec0a47c424bc99b71005202045dc09270da44a27848d534600ac64fc7",
"sha256:197df8509263b0b8614e1df1756b1dd41be6738eed2ba9e9769f3880c2b9d7b6",
"sha256:1e464456d24e23d11fced2bc8c47ef66d471f845c7b7a42f3bd77bf3d1789650",
"sha256:2039230db3c6c639bd84efe3292ec7b06e9214a2992cd9beb293d639c6402edb",
"sha256:21199526696b8f09c3997e2b4db8d0b108d801a348414264d2eb8eb2532e540d",
"sha256:2644b5d63633702bc2c5f3754b1b475378fbbfb481f62319388235d0cd104c2d",
"sha256:432247333090c8c5366e69627ccb363bc58514ae3e63f7fc75c54b1ea80fa7de",
"sha256:444ebfb4e441254e87bad00c661fe32df9969b2bf224373a448d8aca2132b395",
"sha256:5b9b29da4f564ba8787c119f37d174f2b69cdfdf9015b7d8c5c16121ddc054ae",
"sha256:5cc42ca67989e9c3cf859e84c2bf014f6633db63d1cbdf8fdb666dcd9e77e3fa",
"sha256:5d8f74030e67087b219b032aa33a919fae8806d49c867846bfacde57f43972ef",
"sha256:72ef3925f30e12a184889aac03d77d031056860ccae8a1e519f6cbb742736383",
"sha256:819dc789f4498ecc91438a7de64427c73b45035e2e3680c92e18795a839ebb66",
"sha256:915ace4ff03fdfff953962fa672d44be269deb2eaf88499a0f8805221bc68c87",
"sha256:9311e99228ae10023300ecac05be5a296f60d2fd10fff31cf5c1fa4ca4b1988d",
"sha256:974308c58d057a651d182208a484ce80a26dac0caef2895836a92dd6ebd725e0",
"sha256:b8b49776299fece66bffaafe357d929ca9451450f5466e997a7285ab0fe28e3b",
"sha256:c957b2b4ea88587b46cf49d1dc17681c1e672864fd7af32fc1e9664d572b3458",
"sha256:e41a86c6c650bcecc6633ee3180d80a025db041a8e2398dcc059b3afa8382cd4",
"sha256:f513588da599943e0cde4e32cc9879e825d58720d6557062d1098c5ad80080e1",
"sha256:fba8a281e570adafb79f7755ac8721b6cf1bbf691186a287e990c7929c7692ff"
],
"index": "pypi",
"version": "==22.10.0"
},
"charset-normalizer": { "charset-normalizer": {
"hashes": [ "hashes": [
"sha256:5a3d016c7c547f69d6f81fb0db9449ce888b418b5b9952cc5e6e66843e9dd845", "sha256:5a3d016c7c547f69d6f81fb0db9449ce888b418b5b9952cc5e6e66843e9dd845",
@ -141,13 +176,29 @@
"markers": "python_version >= '3.6'", "markers": "python_version >= '3.6'",
"version": "==2.1.1" "version": "==2.1.1"
}, },
"click": {
"hashes": [
"sha256:7682dc8afb30297001674575ea00d1814d808d6a36af415a82bd481d37ba7b8e",
"sha256:bb4d8133cb15a609f44e8213d9b391b0809795062913b383c62be0ee95b1db48"
],
"markers": "python_version >= '3.7'",
"version": "==8.1.3"
},
"dateparser": { "dateparser": {
"hashes": [ "hashes": [
"sha256:038196b1f12c7397e38aad3d61588833257f6f552baa63a1499e6987fa8d42d9", "sha256:3821bf191f95b2658c4abd91571c09821ce7a2bc179bf6cefd8b4515c3ccf9ef",
"sha256:9600874312ff28a41f96ec7ccdc73be1d1c44435719da47fea3339d55ff5a628" "sha256:d31659dc806a7d88e2b510b2c74f68b525ae531f145c62a57a99bd616b7f90cf"
], ],
"index": "pypi", "index": "pypi",
"version": "==1.1.1" "version": "==1.1.2"
},
"dill": {
"hashes": [
"sha256:a07ffd2351b8c678dfc4a856a3005f8067aea51d6ba6c700796a4d9e280f39f0",
"sha256:e5db55f3687856d8fbdab002ed78544e1c4559a130302693d839dfe8f93f2373"
],
"markers": "python_version >= '3.7'",
"version": "==0.3.6"
}, },
"discord": { "discord": {
"hashes": [ "hashes": [
@ -238,6 +289,65 @@
"markers": "python_version >= '3.5'", "markers": "python_version >= '3.5'",
"version": "==3.4" "version": "==3.4"
}, },
"isort": {
"hashes": [
"sha256:6f62d78e2f89b4500b080fe3a81690850cd254227f27f75c3a0c491a1f351ba7",
"sha256:e8443a5e7a020e9d7f97f1d7d9cd17c88bcb3bc7e218bf9cf5095fe550be2951"
],
"markers": "python_version < '4.0' and python_full_version >= '3.6.1'",
"version": "==5.10.1"
},
"lazy-object-proxy": {
"hashes": [
"sha256:043651b6cb706eee4f91854da4a089816a6606c1428fd391573ef8cb642ae4f7",
"sha256:07fa44286cda977bd4803b656ffc1c9b7e3bc7dff7d34263446aec8f8c96f88a",
"sha256:12f3bb77efe1367b2515f8cb4790a11cffae889148ad33adad07b9b55e0ab22c",
"sha256:2052837718516a94940867e16b1bb10edb069ab475c3ad84fd1e1a6dd2c0fcfc",
"sha256:2130db8ed69a48a3440103d4a520b89d8a9405f1b06e2cc81640509e8bf6548f",
"sha256:39b0e26725c5023757fc1ab2a89ef9d7ab23b84f9251e28f9cc114d5b59c1b09",
"sha256:46ff647e76f106bb444b4533bb4153c7370cdf52efc62ccfc1a28bdb3cc95442",
"sha256:4dca6244e4121c74cc20542c2ca39e5c4a5027c81d112bfb893cf0790f96f57e",
"sha256:553b0f0d8dbf21890dd66edd771f9b1b5f51bd912fa5f26de4449bfc5af5e029",
"sha256:677ea950bef409b47e51e733283544ac3d660b709cfce7b187f5ace137960d61",
"sha256:6a24357267aa976abab660b1d47a34aaf07259a0c3859a34e536f1ee6e76b5bb",
"sha256:6a6e94c7b02641d1311228a102607ecd576f70734dc3d5e22610111aeacba8a0",
"sha256:6aff3fe5de0831867092e017cf67e2750c6a1c7d88d84d2481bd84a2e019ec35",
"sha256:6ecbb350991d6434e1388bee761ece3260e5228952b1f0c46ffc800eb313ff42",
"sha256:7096a5e0c1115ec82641afbdd70451a144558ea5cf564a896294e346eb611be1",
"sha256:70ed0c2b380eb6248abdef3cd425fc52f0abd92d2b07ce26359fcbc399f636ad",
"sha256:8561da8b3dd22d696244d6d0d5330618c993a215070f473b699e00cf1f3f6443",
"sha256:85b232e791f2229a4f55840ed54706110c80c0a210d076eee093f2b2e33e1bfd",
"sha256:898322f8d078f2654d275124a8dd19b079080ae977033b713f677afcfc88e2b9",
"sha256:8f3953eb575b45480db6568306893f0bd9d8dfeeebd46812aa09ca9579595148",
"sha256:91ba172fc5b03978764d1df5144b4ba4ab13290d7bab7a50f12d8117f8630c38",
"sha256:9d166602b525bf54ac994cf833c385bfcc341b364e3ee71e3bf5a1336e677b55",
"sha256:a57d51ed2997e97f3b8e3500c984db50a554bb5db56c50b5dab1b41339b37e36",
"sha256:b9e89b87c707dd769c4ea91f7a31538888aad05c116a59820f28d59b3ebfe25a",
"sha256:bb8c5fd1684d60a9902c60ebe276da1f2281a318ca16c1d0a96db28f62e9166b",
"sha256:c19814163728941bb871240d45c4c30d33b8a2e85972c44d4e63dd7107faba44",
"sha256:c4ce15276a1a14549d7e81c243b887293904ad2d94ad767f42df91e75fd7b5b6",
"sha256:c7a683c37a8a24f6428c28c561c80d5f4fd316ddcf0c7cab999b15ab3f5c5c69",
"sha256:d609c75b986def706743cdebe5e47553f4a5a1da9c5ff66d76013ef396b5a8a4",
"sha256:d66906d5785da8e0be7360912e99c9188b70f52c422f9fc18223347235691a84",
"sha256:dd7ed7429dbb6c494aa9bc4e09d94b778a3579be699f9d67da7e6804c422d3de",
"sha256:df2631f9d67259dc9620d831384ed7732a198eb434eadf69aea95ad18c587a28",
"sha256:e368b7f7eac182a59ff1f81d5f3802161932a41dc1b1cc45c1f757dc876b5d2c",
"sha256:e40f2013d96d30217a51eeb1db28c9ac41e9d0ee915ef9d00da639c5b63f01a1",
"sha256:f769457a639403073968d118bc70110e7dce294688009f5c24ab78800ae56dc8",
"sha256:fccdf7c2c5821a8cbd0a9440a456f5050492f2270bd54e94360cac663398739b",
"sha256:fd45683c3caddf83abbb1249b653a266e7069a09f486daa8863fb0e7496a9fdb"
],
"markers": "python_version >= '3.6'",
"version": "==1.7.1"
},
"mccabe": {
"hashes": [
"sha256:348e0240c33b60bbdf4e523192ef919f28cb2c3d7d5c7794f74009290f236325",
"sha256:6c2d30ab6be0e4a46919781807b4f0d834ebdd6c6e3dca0bda5a15f863427b6e"
],
"markers": "python_version >= '3.6'",
"version": "==0.7.0"
},
"multidict": { "multidict": {
"hashes": [ "hashes": [
"sha256:0327292e745a880459ef71be14e709aaea2f783f3537588fb4ed09b6c01bca60", "sha256:0327292e745a880459ef71be14e709aaea2f783f3537588fb4ed09b6c01bca60",
@ -303,6 +413,21 @@
"markers": "python_version >= '3.7'", "markers": "python_version >= '3.7'",
"version": "==6.0.2" "version": "==6.0.2"
}, },
"mypy-extensions": {
"hashes": [
"sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d",
"sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"
],
"version": "==0.4.3"
},
"pathspec": {
"hashes": [
"sha256:46846318467efc4556ccfd27816e004270a9eeeeb4d062ce5e6fc7a87c573f93",
"sha256:7ace6161b621d31e7902eb6b5ae148d12cfd23f4a249b9ffb6b9fee12084323d"
],
"markers": "python_version >= '3.7'",
"version": "==0.10.1"
},
"peewee": { "peewee": {
"hashes": [ "hashes": [
"sha256:cc934286d0c0842203abe66a3c6583d1463371e633b03d6da054d0f74e70706f" "sha256:cc934286d0c0842203abe66a3c6583d1463371e633b03d6da054d0f74e70706f"
@ -310,6 +435,22 @@
"index": "pypi", "index": "pypi",
"version": "==3.15.3" "version": "==3.15.3"
}, },
"platformdirs": {
"hashes": [
"sha256:027d8e83a2d7de06bbac4e5ef7e023c02b863d7ea5d079477e722bb41ab25788",
"sha256:58c8abb07dcb441e6ee4b11d8df0ac856038f944ab98b7be6b27b2a3c7feef19"
],
"markers": "python_version >= '3.7'",
"version": "==2.5.2"
},
"pylint": {
"hashes": [
"sha256:3b120505e5af1d06a5ad76b55d8660d44bf0f2fc3c59c2bdd94e39188ee3a4df",
"sha256:c2108037eb074334d9e874dc3c783752cc03d0796c88c9a9af282d0f161a1004"
],
"index": "pypi",
"version": "==2.15.5"
},
"python-dateutil": { "python-dateutil": {
"hashes": [ "hashes": [
"sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86", "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86",
@ -320,10 +461,10 @@
}, },
"pytz": { "pytz": {
"hashes": [ "hashes": [
"sha256:2c0784747071402c6e99f0bafdb7da0fa22645f06554c7ae06bf6358897e9c91", "sha256:335ab46900b1465e714b4fda4963d87363264eb662aab5e65da039c25f1f5b22",
"sha256:48ce799d83b6f8aab2020e369b627446696619e79645419610b9facd909b3174" "sha256:c4d88f472f54d615e9cd582a5004d1e5f624854a6a27a6211591c251f22a6914"
], ],
"version": "==2022.4" "version": "==2022.5"
}, },
"pytz-deprecation-shim": { "pytz-deprecation-shim": {
"hashes": [ "hashes": [
@ -421,13 +562,29 @@
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
"version": "==1.16.0" "version": "==1.16.0"
}, },
"tomli": {
"hashes": [
"sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc",
"sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"
],
"markers": "python_full_version < '3.11.0a7'",
"version": "==2.0.1"
},
"tomlkit": {
"hashes": [
"sha256:571854ebbb5eac89abcb4a2e47d7ea27b89bf29e09c35395da6f03dd4ae23d1c",
"sha256:f2ef9da9cef846ee027947dc99a45d6b68a63b0ebc21944649505bf2e8bc5fe7"
],
"markers": "python_version >= '3.6' and python_version < '4.0'",
"version": "==0.11.5"
},
"tzdata": { "tzdata": {
"hashes": [ "hashes": [
"sha256:74da81ecf2b3887c94e53fc1d466d4362aaf8b26fc87cda18f22004544694583", "sha256:323161b22b7802fdc78f20ca5f6073639c64f1a7227c40cd3e19fd1d0ce6650a",
"sha256:ada9133fbd561e6ec3d1674d3fba50251636e918aa97bd59d63735bef5a513bb" "sha256:e15b2b3005e2546108af42a0eb4ccab4d9e225e2dfbf4f77aad50c70a4b1f3ab"
], ],
"markers": "python_version >= '3.6'", "markers": "python_version >= '3.6'",
"version": "==2022.4" "version": "==2022.5"
}, },
"tzlocal": { "tzlocal": {
"hashes": [ "hashes": [
@ -437,6 +594,76 @@
"markers": "python_version >= '3.6'", "markers": "python_version >= '3.6'",
"version": "==4.2" "version": "==4.2"
}, },
"wrapt": {
"hashes": [
"sha256:00b6d4ea20a906c0ca56d84f93065b398ab74b927a7a3dbd470f6fc503f95dc3",
"sha256:01c205616a89d09827986bc4e859bcabd64f5a0662a7fe95e0d359424e0e071b",
"sha256:02b41b633c6261feff8ddd8d11c711df6842aba629fdd3da10249a53211a72c4",
"sha256:07f7a7d0f388028b2df1d916e94bbb40624c59b48ecc6cbc232546706fac74c2",
"sha256:11871514607b15cfeb87c547a49bca19fde402f32e2b1c24a632506c0a756656",
"sha256:1b376b3f4896e7930f1f772ac4b064ac12598d1c38d04907e696cc4d794b43d3",
"sha256:21ac0156c4b089b330b7666db40feee30a5d52634cc4560e1905d6529a3897ff",
"sha256:257fd78c513e0fb5cdbe058c27a0624c9884e735bbd131935fd49e9fe719d310",
"sha256:2b39d38039a1fdad98c87279b48bc5dce2c0ca0d73483b12cb72aa9609278e8a",
"sha256:2cf71233a0ed05ccdabe209c606fe0bac7379fdcf687f39b944420d2a09fdb57",
"sha256:2fe803deacd09a233e4762a1adcea5db5d31e6be577a43352936179d14d90069",
"sha256:3232822c7d98d23895ccc443bbdf57c7412c5a65996c30442ebe6ed3df335383",
"sha256:34aa51c45f28ba7f12accd624225e2b1e5a3a45206aa191f6f9aac931d9d56fe",
"sha256:36f582d0c6bc99d5f39cd3ac2a9062e57f3cf606ade29a0a0d6b323462f4dd87",
"sha256:380a85cf89e0e69b7cfbe2ea9f765f004ff419f34194018a6827ac0e3edfed4d",
"sha256:40e7bc81c9e2b2734ea4bc1aceb8a8f0ceaac7c5299bc5d69e37c44d9081d43b",
"sha256:43ca3bbbe97af00f49efb06e352eae40434ca9d915906f77def219b88e85d907",
"sha256:4fcc4649dc762cddacd193e6b55bc02edca674067f5f98166d7713b193932b7f",
"sha256:5a0f54ce2c092aaf439813735584b9537cad479575a09892b8352fea5e988dc0",
"sha256:5a9a0d155deafd9448baff28c08e150d9b24ff010e899311ddd63c45c2445e28",
"sha256:5b02d65b9ccf0ef6c34cba6cf5bf2aab1bb2f49c6090bafeecc9cd81ad4ea1c1",
"sha256:60db23fa423575eeb65ea430cee741acb7c26a1365d103f7b0f6ec412b893853",
"sha256:642c2e7a804fcf18c222e1060df25fc210b9c58db7c91416fb055897fc27e8cc",
"sha256:6a9a25751acb379b466ff6be78a315e2b439d4c94c1e99cb7266d40a537995d3",
"sha256:6b1a564e6cb69922c7fe3a678b9f9a3c54e72b469875aa8018f18b4d1dd1adf3",
"sha256:6d323e1554b3d22cfc03cd3243b5bb815a51f5249fdcbb86fda4bf62bab9e164",
"sha256:6e743de5e9c3d1b7185870f480587b75b1cb604832e380d64f9504a0535912d1",
"sha256:709fe01086a55cf79d20f741f39325018f4df051ef39fe921b1ebe780a66184c",
"sha256:7b7c050ae976e286906dd3f26009e117eb000fb2cf3533398c5ad9ccc86867b1",
"sha256:7d2872609603cb35ca513d7404a94d6d608fc13211563571117046c9d2bcc3d7",
"sha256:7ef58fb89674095bfc57c4069e95d7a31cfdc0939e2a579882ac7d55aadfd2a1",
"sha256:80bb5c256f1415f747011dc3604b59bc1f91c6e7150bd7db03b19170ee06b320",
"sha256:81b19725065dcb43df02b37e03278c011a09e49757287dca60c5aecdd5a0b8ed",
"sha256:833b58d5d0b7e5b9832869f039203389ac7cbf01765639c7309fd50ef619e0b1",
"sha256:88bd7b6bd70a5b6803c1abf6bca012f7ed963e58c68d76ee20b9d751c74a3248",
"sha256:8ad85f7f4e20964db4daadcab70b47ab05c7c1cf2a7c1e51087bfaa83831854c",
"sha256:8c0ce1e99116d5ab21355d8ebe53d9460366704ea38ae4d9f6933188f327b456",
"sha256:8d649d616e5c6a678b26d15ece345354f7c2286acd6db868e65fcc5ff7c24a77",
"sha256:903500616422a40a98a5a3c4ff4ed9d0066f3b4c951fa286018ecdf0750194ef",
"sha256:9736af4641846491aedb3c3f56b9bc5568d92b0692303b5a305301a95dfd38b1",
"sha256:988635d122aaf2bdcef9e795435662bcd65b02f4f4c1ae37fbee7401c440b3a7",
"sha256:9cca3c2cdadb362116235fdbd411735de4328c61425b0aa9f872fd76d02c4e86",
"sha256:9e0fd32e0148dd5dea6af5fee42beb949098564cc23211a88d799e434255a1f4",
"sha256:9f3e6f9e05148ff90002b884fbc2a86bd303ae847e472f44ecc06c2cd2fcdb2d",
"sha256:a85d2b46be66a71bedde836d9e41859879cc54a2a04fad1191eb50c2066f6e9d",
"sha256:a9a52172be0b5aae932bef82a79ec0a0ce87288c7d132946d645eba03f0ad8a8",
"sha256:aa31fdcc33fef9eb2552cbcbfee7773d5a6792c137b359e82879c101e98584c5",
"sha256:b014c23646a467558be7da3d6b9fa409b2c567d2110599b7cf9a0c5992b3b471",
"sha256:b21bb4c09ffabfa0e85e3a6b623e19b80e7acd709b9f91452b8297ace2a8ab00",
"sha256:b5901a312f4d14c59918c221323068fad0540e34324925c8475263841dbdfe68",
"sha256:b9b7a708dd92306328117d8c4b62e2194d00c365f18eff11a9b53c6f923b01e3",
"sha256:d1967f46ea8f2db647c786e78d8cc7e4313dbd1b0aca360592d8027b8508e24d",
"sha256:d52a25136894c63de15a35bc0bdc5adb4b0e173b9c0d07a2be9d3ca64a332735",
"sha256:d77c85fedff92cf788face9bfa3ebaa364448ebb1d765302e9af11bf449ca36d",
"sha256:d79d7d5dc8a32b7093e81e97dad755127ff77bcc899e845f41bf71747af0c569",
"sha256:dbcda74c67263139358f4d188ae5faae95c30929281bc6866d00573783c422b7",
"sha256:ddaea91abf8b0d13443f6dac52e89051a5063c7d014710dcb4d4abb2ff811a59",
"sha256:dee0ce50c6a2dd9056c20db781e9c1cfd33e77d2d569f5d1d9321c641bb903d5",
"sha256:dee60e1de1898bde3b238f18340eec6148986da0455d8ba7848d50470a7a32fb",
"sha256:e2f83e18fe2f4c9e7db597e988f72712c0c3676d337d8b101f6758107c42425b",
"sha256:e3fb1677c720409d5f671e39bac6c9e0e422584e5f518bfd50aa4cbbea02433f",
"sha256:ee2b1b1769f6707a8a445162ea16dddf74285c3964f605877a20e38545c3c462",
"sha256:ee6acae74a2b91865910eef5e7de37dc6895ad96fa23603d1d27ea69df545015",
"sha256:ef3f72c9666bba2bab70d2a8b79f2c6d2c1a42a7f7e2b0ec83bb2f9e383950af"
],
"markers": "python_version < '3.11'",
"version": "==1.14.1"
},
"yarl": { "yarl": {
"hashes": [ "hashes": [
"sha256:076eede537ab978b605f41db79a56cad2e7efeea2aa6e0fa8f05a26c24a034fb", "sha256:076eede537ab978b605f41db79a56cad2e7efeea2aa6e0fa8f05a26c24a034fb",

View File

@ -2,43 +2,68 @@
# Copyright 2022 - c0de <c0de@c0de.dev> # Copyright 2022 - c0de <c0de@c0de.dev>
# Licensed under the MIT License (https://opensource.org/licenses/MIT) # Licensed under the MIT License (https://opensource.org/licenses/MIT)
# pylint: disable=too-few-public-methods
"""
An implementation of a SQLite database
"""
import os import os
import datetime import datetime
from peewee import * from peewee import (
SqliteDatabase,
Model,
UUIDField,
IntegerField,
CharField,
DateTimeField,
ForeignKeyField,
)
# User can provide path to database, or it will be put next to models.py # User can provide path to database, or it will be put next to main.py
DATABASE = os.environ.get('database_path', os.getcwd() + '/ghostball.db') DATABASE = os.environ.get("database_path", os.getcwd() + "/ghostball.db")
database = SqliteDatabase(DATABASE) database = SqliteDatabase(DATABASE)
class BaseModel(Model): class BaseModel(Model):
"""All of our models will inherit this class """All of our models will inherit this class
and use the same database connection""" and use the same database connection"""
class Meta: class Meta:
"""meta"""
database = database database = database
class GameModel(BaseModel): class GameModel(BaseModel):
"""Games that are ran"""
game_id = UUIDField(primary_key=True) game_id = UUIDField(primary_key=True)
server_id = UUIDField() # Unsure if this is actually a uuid server_id = IntegerField()
pitch_value = IntegerField(null=True) pitch_value = IntegerField(null=True)
date_created = DateTimeField(default=datetime.datetime.now) date_created = DateTimeField(default=datetime.datetime.now)
date_ended = DateTimeField(null=True)
class GuessModel(BaseModel): class GuessModel(BaseModel):
"""Guesses for a particular game"""
player_id = IntegerField(primary_key=True) player_id = IntegerField(primary_key=True)
game_id = ForeignKeyField(GameModel, backref="guesses") game_id = ForeignKeyField(GameModel, backref="guesses")
player_name = CharField() player_name = CharField()
guess = IntegerField() guess = IntegerField(default=0)
difference = IntegerField(null=True) difference = IntegerField(null=True)
date_guessed = DateTimeField(null=True)
# TODO: Add unique constraint for player_id and game_id # TODO: Add unique constraint for player_id and game_id
# ie: one guess per player allowed per game # ie: one guess per player allowed per game
def create_models(): def create_models():
"""Create database tables"""
with database: with database:
database.create_tables([Play, Guess]) database.create_tables([GameModel, GuessModel])

View File

@ -2,26 +2,44 @@
# Copyright 2022 - c0de <c0de@c0de.dev> # Copyright 2022 - c0de <c0de@c0de.dev>
# Licensed under the MIT License (https://opensource.org/licenses/MIT) # Licensed under the MIT License (https://opensource.org/licenses/MIT)
# pylint: disable=wrong-import-position
"""
A discord bot that hosts Ghostball/Braveball.
A discord game where players guess the pitch speed
from a fantasy baseball pitcher, and whoever is
closer gets more points
"""
import sys import sys
import discord import discord
# Import game functions # Import game functions
sys.path.append('..') sys.path.append("..")
import game import game
class GhostBallClient(discord.Client): class GhostBallClient(discord.Client):
"""
Implementation of a Discord client that will monitor
a channel for messages, and if it recieves a message
defined in the Game object, and pass it along
"""
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super(GhostBallClient, self).__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.game = game.Game() with game.Game() as self.game:
self.game.discord = self self.game.discord = self
async def on_ready(self): async def on_ready(self):
"""Method called when connected to Discord"""
print("Logged on as", self.user) print("Logged on as", self.user)
async def on_message(self, message): async def on_message(self, message):
"""Method called when a message is recieved"""
# Don't respond to ourself # Don't respond to ourself
if message.author == self.user: if message.author == self.user:
return return
@ -31,7 +49,7 @@ class GhostBallClient(discord.Client):
await message.channel.send("pong") await message.channel.send("pong")
# Game commands # Game commands
if message.content.startswith('!'): if message.content.startswith("!"):
firstword = message.content[1:].split()[0] firstword = message.content[1:].split()[0]
# Determine if the first word is a command, and run it # Determine if the first word is a command, and run it

View File

@ -2,12 +2,39 @@
# Copyright 2022 - c0de <c0de@c0de.dev> # Copyright 2022 - c0de <c0de@c0de.dev>
# Licensed under the MIT License (https://opensource.org/licenses/MIT) # Licensed under the MIT License (https://opensource.org/licenses/MIT)
# pylint: disable=no-member
"""
A Context Manager / State Machine that keeps track of
a single game instance (there should only be one) in a
Discord channel
"""
import uuid import uuid
import dateparser # import dateparser
from database.models import database, GameModel, GuessModel from database.models import database, GameModel, GuessModel
async def check_is_running(method, start_new_game=True):
"""
Decorator that determines if the game is running or not
"""
async def wrapper(self):
if self.is_running and start_new_game:
return await self.message.channel.send("A game is already running")
if not self.is_running and not start_new_game:
return await self.message.channel.send("There is no game running")
await method(self)
return await wrapper
class Game: class Game:
""" """
The game state class The game state class
@ -20,11 +47,11 @@ class Game:
self.is_running = False self.is_running = False
self.commands = { self.commands = {
'ghostball': self.start, "ghostball": self.start,
'resolve': self.stop, "resolve": self.stop,
'guess': self.guess, "guess": self.guess,
'points': self.points, "points": self.points,
'help': self.help, "help": self.help,
} }
self.game = GameModel self.game = GameModel
@ -35,101 +62,129 @@ class Game:
# Discord client instance # Discord client instance
self.discord = None self.discord = None
async def start(self): def __enter__(self):
if self.is_running: """
return await self.message.channel.send("A game is already running") Allows use of `with Game() as game` for try/except statements
(https://peps.python.org/pep-0343/)
"""
database.connect() database.connect()
return self
def __exit__(self, exception_type, exception_value, exception_traceback):
"""
Automagically close the database
when this class has ended execution
"""
database.close()
@check_is_running
async def start(self):
"""
Start command - Starts a new game if there isn't already one running
"""
self.is_running = True self.is_running = True
# game.pitch_value is unknown at the start of the game # game.pitch_value is unknown at the start of the game
self.game = GameModel.create( self.game = GameModel.create(
game_id = uuid.uuid4(), game_id=uuid.uuid4(), server_id=self.message.guild.id
server_id = self.message.guild.id
) )
database.close() await self.message.send("Send me your guesses with !guess <number>")
await self.message.send("@flappy ball, pitch is in! Send me your guesses with !guess <number>") def __stop_args__(self):
def __stopArgs__(self):
pieces = self.message.content.split() pieces = self.message.content.split()
if len(pieces) == 2: if len(pieces) == 2:
return pieces[1], False, None, None return pieces[1], False, None, None
elif len(pieces) == 4:
if len(pieces) == 4:
return pieces[1], True, pieces[2], pieces[3] return pieces[1], True, pieces[2], pieces[3]
return None, False, None, None return None, False, None, None
@check_is_running(stop, start_new_game=False)
async def stop(self): async def stop(self):
if not self.is_running: """
return await self.message.channel.send("There is no game running to resolve") Stop command - Stops the game if it is currently running,
saves the pitch value, and displays differences
"""
# Determine arguments # Determine arguments
pitch_value, has_batter, batter_id, batter_guess = self.__stopArgs__() pitch_value, has_batter, batter_id, batter_guess = self.__stop_args__()
if not pitch_value: if not pitch_value:
return await self.message.channel.send(f"Invalid command <@{ str(self.message.author.id) }>!") return await self.message.channel.send(
f"Invalid command <@{ str(self.message.author.id) }>!"
database.connect() )
if has_batter: if has_batter:
player_id = batter_id[3:] player_id = batter_id[3:]
GuessModel.create( GuessModel.create(
game_id = self.game.game_id, game_id=self.game.game_id,
player_id = player_id, player_id=player_id,
player_name = self.discord.get_user(int(player_id).name), player_name=self.discord.get_user(int(player_id).name),
guess = int(batter_guess) guess=int(batter_guess),
) )
# Save the pitch value # Save the pitch value
self.game.update({'pitch_value': pitch_value}) self.game.update({"pitch_value": pitch_value})
# TODO: Determine differences # TODO: Determine differences
await self.message.channel.send("Difference calculation is not currently available") await self.message.channel.send(
"Difference calculation is not currently available"
)
# stop and discard game # stop and discard game
self.is_running = False self.is_running = False
self.game = None self.game = None
database.close()
@check_is_running(guess, start_new_game=False)
async def guess(self): async def guess(self):
if not self.is_running: """
return await self.message.channel.send("There is no game running to add guesses to") Guess command - Allows the player to add a guess to the current
running game
"""
value = int(self.message.content.split()[1]) value = int(self.message.content.split()[1])
if value < 1 or value > 1000: if value < 1 or value > 1000:
return await self.message.channel.send(f"Invalid value. It must be between 1 and 1000 inclusive") return await self.message.channel.send(
"Invalid value. It must be between 1 and 1000 inclusive"
database.connect()
GuessModel.create(
game_id = self.game.game_id,
player_id = self.message.author.id,
player_name = self.message.author.name,
guess = value
) )
database.close() # TODO: Check if the user tried to vote before, update their vote if so
player_guess, created = GuessModel.get_or_create(
game_id=self.game.game_id,
player_id=self.message.author.id,
player_name=self.message.author.name,
)
player_guess.update(guess=value)
if not created: # They updated their guess
await self.message.channel.send(
f"<@{ str(self.message.author.id) }> your guess has been updated"
)
return await self.message.add_reaction(emoji="\N{THUMBS UP SIGN}") return await self.message.add_reaction(emoji="\N{THUMBS UP SIGN}")
async def points(self): async def points(self):
# database.connect() """
Points command - returns a table ordered from highest to lowest
value = self.message.content.split() of the players and their points
try: """
if len(value) > 1: # TODO
timestamp = dateparser.parse(value[1]) # value = self.message.content.split()
except: # try:
return await self.message.channel.send("Invalid timestamp. Try again") # if len(value) > 1:
# timestamp = dateparser.parse(value[1])
# database.close() # except:
# return await self.message.channel.send("Invalid timestamp. Try again")
return await self.message.channel.send("Sorry, not implemented yet") return await self.message.channel.send("Sorry, not implemented yet")
async def help(self): async def help(self):
"""help command"""
# TODO: Add help message # TODO: Add help message
help_message = "help" help_message = "help"

4
GhostBallBot/lint.sh Executable file
View File

@ -0,0 +1,4 @@
#!/bin/bash
pylint main.py game.py discord_client database > lint.log
black main.py game.py discord_client database

View File

@ -2,15 +2,16 @@
# Copyright 2022 - c0de <c0de@c0de.dev> # Copyright 2022 - c0de <c0de@c0de.dev>
# Licensed under the MIT License (https://opensource.org/licenses/MIT) # Licensed under the MIT License (https://opensource.org/licenses/MIT)
# pylint: disable=missing-module-docstring
import os import os
from pathlib import Path
from discord_client.client import GhostBallClient from discord_client.client import GhostBallClient
from database.models import DATABASE, database, create_models from database.models import DATABASE, database, create_models
if __name__ == '__main__': if __name__ == "__main__":
client = GhostBallClient(intents=None) client = GhostBallClient(intents=None)
client.run(os.environ.get('discord_token')) client.run(os.environ.get("discord_token"))
# Set up the database if we haven't already # Set up the database if we haven't already
if not os.path.exists(DATABASE): if not os.path.exists(DATABASE):