From fee3fdda5249ee119172bf75934820dea9d9a0d4 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:33:05 +0000 Subject: [PATCH 01/18] chore: init uv project --- .python-version | 1 + pyproject.toml | 10 + uv.lock | 769 ++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 780 insertions(+) create mode 100644 .python-version create mode 100644 pyproject.toml create mode 100644 uv.lock diff --git a/.python-version b/.python-version new file mode 100644 index 0000000..24ee5b1 --- /dev/null +++ b/.python-version @@ -0,0 +1 @@ +3.13 diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..99d00a6 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,10 @@ +[project] +name = "sqlingo" +version = "0.1.0" +description = "Add your description here" +readme = "README.md" +requires-python = ">=3.13" +dependencies = [ + "duckdb>=1.5.2", + "streamlit>=1.56.0", +] diff --git a/uv.lock b/uv.lock new file mode 100644 index 0000000..57e9683 --- /dev/null +++ b/uv.lock @@ -0,0 +1,769 @@ +version = 1 +revision = 3 +requires-python = ">=3.13" +resolution-markers = [ + "python_full_version >= '3.14' and sys_platform == 'win32'", + "python_full_version >= '3.14' and sys_platform == 'emscripten'", + "python_full_version >= '3.14' and sys_platform != 'emscripten' and sys_platform != 'win32'", + "python_full_version < '3.14' and sys_platform == 'win32'", + "python_full_version < '3.14' and sys_platform == 'emscripten'", + "python_full_version < '3.14' and sys_platform != 'emscripten' and sys_platform != 'win32'", +] + +[[package]] +name = "altair" +version = "6.1.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "jinja2" }, + { name = "jsonschema" }, + { name = "narwhals" }, + { name = "packaging" }, + { name = "typing-extensions", marker = "python_full_version < '3.15'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/3a/1e/365a9144db3254f86f1b974660b9ede1e9a38c9dc0730e4a9b1192eec5d6/altair-6.1.0.tar.gz", hash = "sha256:dda699216cf85b040d968ae5a569ad45957616811e38760a85e5118269daca67", size = 765519, upload-time = "2026-04-21T13:08:46.44Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ce/63/5dacc8d8306c715088b897a479e551bc0779fd2f0f26c97fec5e36542b4e/altair-6.1.0-py3-none-any.whl", hash = "sha256:fdf5fd939512e5b2fc4441c82dfd2635e706defbd037db0ac429ef5ddce66c3b", size = 796996, upload-time = "2026-04-21T13:08:48.549Z" }, +] + +[[package]] +name = "attrs" +version = "26.1.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/9a/8e/82a0fe20a541c03148528be8cac2408564a6c9a0cc7e9171802bc1d26985/attrs-26.1.0.tar.gz", hash = "sha256:d03ceb89cb322a8fd706d4fb91940737b6642aa36998fe130a9bc96c985eff32", size = 952055, upload-time = "2026-03-19T14:22:25.026Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/64/b4/17d4b0b2a2dc85a6df63d1157e028ed19f90d4cd97c36717afef2bc2f395/attrs-26.1.0-py3-none-any.whl", hash = "sha256:c647aa4a12dfbad9333ca4e71fe62ddc36f4e63b2d260a37a8b83d2f043ac309", size = 67548, upload-time = "2026-03-19T14:22:23.645Z" }, +] + +[[package]] +name = "blinker" +version = "1.9.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/21/28/9b3f50ce0e048515135495f198351908d99540d69bfdc8c1d15b73dc55ce/blinker-1.9.0.tar.gz", hash = "sha256:b4ce2265a7abece45e7cc896e98dbebe6cead56bcf805a3d23136d145f5445bf", size = 22460, upload-time = "2024-11-08T17:25:47.436Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/10/cb/f2ad4230dc2eb1a74edf38f1a38b9b52277f75bef262d8908e60d957e13c/blinker-1.9.0-py3-none-any.whl", hash = "sha256:ba0efaa9080b619ff2f3459d1d500c57bddea4a6b424b60a91141db6fd2f08bc", size = 8458, upload-time = "2024-11-08T17:25:46.184Z" }, +] + +[[package]] +name = "cachetools" +version = "7.0.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/76/7b/1755ed2c6bfabd1d98b37ae73152f8dcf94aa40fee119d163c19ed484704/cachetools-7.0.6.tar.gz", hash = "sha256:e5d524d36d65703a87243a26ff08ad84f73352adbeafb1cde81e207b456aaf24", size = 37526, upload-time = "2026-04-20T19:02:23.289Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/fe/c4/cf76242a5da1410917107ff14551764aa405a5fd10cd10cf9a5ca8fa77f4/cachetools-7.0.6-py3-none-any.whl", hash = "sha256:4e94956cfdd3086f12042cdd29318f5ced3893014f7d0d059bf3ead3f85b7f8b", size = 13976, upload-time = "2026-04-20T19:02:21.187Z" }, +] + +[[package]] +name = "certifi" +version = "2026.4.22" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/25/ee/6caf7a40c36a1220410afe15a1cc64993a1f864871f698c0f93acb72842a/certifi-2026.4.22.tar.gz", hash = "sha256:8d455352a37b71bf76a79caa83a3d6c25afee4a385d632127b6afb3963f1c580", size = 137077, upload-time = "2026-04-22T11:26:11.191Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/22/30/7cd8fdcdfbc5b869528b079bfb76dcdf6056b1a2097a662e5e8c04f42965/certifi-2026.4.22-py3-none-any.whl", hash = "sha256:3cb2210c8f88ba2318d29b0388d1023c8492ff72ecdde4ebdaddbb13a31b1c4a", size = 135707, upload-time = "2026-04-22T11:26:09.372Z" }, +] + +[[package]] +name = "charset-normalizer" +version = "3.4.7" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/e7/a1/67fe25fac3c7642725500a3f6cfe5821ad557c3abb11c9d20d12c7008d3e/charset_normalizer-3.4.7.tar.gz", hash = "sha256:ae89db9e5f98a11a4bf50407d4363e7b09b31e55bc117b4f7d80aab97ba009e5", size = 144271, upload-time = "2026-04-02T09:28:39.342Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/c1/3b/66777e39d3ae1ddc77ee606be4ec6d8cbd4c801f65e5a1b6f2b11b8346dd/charset_normalizer-3.4.7-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:f496c9c3cc02230093d8330875c4c3cdfc3b73612a5fd921c65d39cbcef08063", size = 309627, upload-time = "2026-04-02T09:26:45.198Z" }, + { url = "https://files.pythonhosted.org/packages/2e/4e/b7f84e617b4854ade48a1b7915c8ccfadeba444d2a18c291f696e37f0d3b/charset_normalizer-3.4.7-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0ea948db76d31190bf08bd371623927ee1339d5f2a0b4b1b4a4439a65298703c", size = 207008, upload-time = "2026-04-02T09:26:46.824Z" }, + { url = "https://files.pythonhosted.org/packages/c4/bb/ec73c0257c9e11b268f018f068f5d00aa0ef8c8b09f7753ebd5f2880e248/charset_normalizer-3.4.7-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:a277ab8928b9f299723bc1a2dabb1265911b1a76341f90a510368ca44ad9ab66", size = 228303, upload-time = "2026-04-02T09:26:48.397Z" }, + { url = "https://files.pythonhosted.org/packages/85/fb/32d1f5033484494619f701e719429c69b766bfc4dbc61aa9e9c8c166528b/charset_normalizer-3.4.7-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:3bec022aec2c514d9cf199522a802bd007cd588ab17ab2525f20f9c34d067c18", size = 224282, upload-time = "2026-04-02T09:26:49.684Z" }, + { url = "https://files.pythonhosted.org/packages/fa/07/330e3a0dda4c404d6da83b327270906e9654a24f6c546dc886a0eb0ffb23/charset_normalizer-3.4.7-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:e044c39e41b92c845bc815e5ae4230804e8e7bc29e399b0437d64222d92809dd", size = 215595, upload-time = "2026-04-02T09:26:50.915Z" }, + { url = "https://files.pythonhosted.org/packages/e3/7c/fc890655786e423f02556e0216d4b8c6bcb6bdfa890160dc66bf52dee468/charset_normalizer-3.4.7-cp313-cp313-manylinux_2_31_armv7l.whl", hash = "sha256:f495a1652cf3fbab2eb0639776dad966c2fb874d79d87ca07f9d5f059b8bd215", size = 201986, upload-time = "2026-04-02T09:26:52.197Z" }, + { url = "https://files.pythonhosted.org/packages/d8/97/bfb18b3db2aed3b90cf54dc292ad79fdd5ad65c4eae454099475cbeadd0d/charset_normalizer-3.4.7-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e712b419df8ba5e42b226c510472b37bd57b38e897d3eca5e8cfd410a29fa859", size = 211711, upload-time = "2026-04-02T09:26:53.49Z" }, + { url = "https://files.pythonhosted.org/packages/6f/a5/a581c13798546a7fd557c82614a5c65a13df2157e9ad6373166d2a3e645d/charset_normalizer-3.4.7-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:7804338df6fcc08105c7745f1502ba68d900f45fd770d5bdd5288ddccb8a42d8", size = 210036, upload-time = "2026-04-02T09:26:54.975Z" }, + { url = "https://files.pythonhosted.org/packages/8c/bf/b3ab5bcb478e4193d517644b0fb2bf5497fbceeaa7a1bc0f4d5b50953861/charset_normalizer-3.4.7-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:481551899c856c704d58119b5025793fa6730adda3571971af568f66d2424bb5", size = 202998, upload-time = "2026-04-02T09:26:56.303Z" }, + { url = "https://files.pythonhosted.org/packages/e7/4e/23efd79b65d314fa320ec6017b4b5834d5c12a58ba4610aa353af2e2f577/charset_normalizer-3.4.7-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:f59099f9b66f0d7145115e6f80dd8b1d847176df89b234a5a6b3f00437aa0832", size = 230056, upload-time = "2026-04-02T09:26:57.554Z" }, + { url = "https://files.pythonhosted.org/packages/b9/9f/1e1941bc3f0e01df116e68dc37a55c4d249df5e6fa77f008841aef68264f/charset_normalizer-3.4.7-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:f59ad4c0e8f6bba240a9bb85504faa1ab438237199d4cce5f622761507b8f6a6", size = 211537, upload-time = "2026-04-02T09:26:58.843Z" }, + { url = "https://files.pythonhosted.org/packages/80/0f/088cbb3020d44428964a6c97fe1edfb1b9550396bf6d278330281e8b709c/charset_normalizer-3.4.7-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:3dedcc22d73ec993f42055eff4fcfed9318d1eeb9a6606c55892a26964964e48", size = 226176, upload-time = "2026-04-02T09:27:00.437Z" }, + { url = "https://files.pythonhosted.org/packages/6a/9f/130394f9bbe06f4f63e22641d32fc9b202b7e251c9aef4db044324dac493/charset_normalizer-3.4.7-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:64f02c6841d7d83f832cd97ccf8eb8a906d06eb95d5276069175c696b024b60a", size = 217723, upload-time = "2026-04-02T09:27:02.021Z" }, + { url = "https://files.pythonhosted.org/packages/73/55/c469897448a06e49f8fa03f6caae97074fde823f432a98f979cc42b90e69/charset_normalizer-3.4.7-cp313-cp313-win32.whl", hash = "sha256:4042d5c8f957e15221d423ba781e85d553722fc4113f523f2feb7b188cc34c5e", size = 148085, upload-time = "2026-04-02T09:27:03.192Z" }, + { url = "https://files.pythonhosted.org/packages/5d/78/1b74c5bbb3f99b77a1715c91b3e0b5bdb6fe302d95ace4f5b1bec37b0167/charset_normalizer-3.4.7-cp313-cp313-win_amd64.whl", hash = "sha256:3946fa46a0cf3e4c8cb1cc52f56bb536310d34f25f01ca9b6c16afa767dab110", size = 158819, upload-time = "2026-04-02T09:27:04.454Z" }, + { url = "https://files.pythonhosted.org/packages/68/86/46bd42279d323deb8687c4a5a811fd548cb7d1de10cf6535d099877a9a9f/charset_normalizer-3.4.7-cp313-cp313-win_arm64.whl", hash = "sha256:80d04837f55fc81da168b98de4f4b797ef007fc8a79ab71c6ec9bc4dd662b15b", size = 147915, upload-time = "2026-04-02T09:27:05.971Z" }, + { url = "https://files.pythonhosted.org/packages/97/c8/c67cb8c70e19ef1960b97b22ed2a1567711de46c4ddf19799923adc836c2/charset_normalizer-3.4.7-cp314-cp314-macosx_10_15_universal2.whl", hash = "sha256:c36c333c39be2dbca264d7803333c896ab8fa7d4d6f0ab7edb7dfd7aea6e98c0", size = 309234, upload-time = "2026-04-02T09:27:07.194Z" }, + { url = "https://files.pythonhosted.org/packages/99/85/c091fdee33f20de70d6c8b522743b6f831a2f1cd3ff86de4c6a827c48a76/charset_normalizer-3.4.7-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1c2aed2e5e41f24ea8ef1590b8e848a79b56f3a5564a65ceec43c9d692dc7d8a", size = 208042, upload-time = "2026-04-02T09:27:08.749Z" }, + { url = "https://files.pythonhosted.org/packages/87/1c/ab2ce611b984d2fd5d86a5a8a19c1ae26acac6bad967da4967562c75114d/charset_normalizer-3.4.7-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:54523e136b8948060c0fa0bc7b1b50c32c186f2fceee897a495406bb6e311d2b", size = 228706, upload-time = "2026-04-02T09:27:09.951Z" }, + { url = "https://files.pythonhosted.org/packages/a8/29/2b1d2cb00bf085f59d29eb773ce58ec2d325430f8c216804a0a5cd83cbca/charset_normalizer-3.4.7-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:715479b9a2802ecac752a3b0efa2b0b60285cf962ee38414211abdfccc233b41", size = 224727, upload-time = "2026-04-02T09:27:11.175Z" }, + { url = "https://files.pythonhosted.org/packages/47/5c/032c2d5a07fe4d4855fea851209cca2b6f03ebeb6d4e3afdb3358386a684/charset_normalizer-3.4.7-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bd6c2a1c7573c64738d716488d2cdd3c00e340e4835707d8fdb8dc1a66ef164e", size = 215882, upload-time = "2026-04-02T09:27:12.446Z" }, + { url = "https://files.pythonhosted.org/packages/2c/c2/356065d5a8b78ed04499cae5f339f091946a6a74f91e03476c33f0ab7100/charset_normalizer-3.4.7-cp314-cp314-manylinux_2_31_armv7l.whl", hash = "sha256:c45e9440fb78f8ddabcf714b68f936737a121355bf59f3907f4e17721b9d1aae", size = 200860, upload-time = "2026-04-02T09:27:13.721Z" }, + { url = "https://files.pythonhosted.org/packages/0c/cd/a32a84217ced5039f53b29f460962abb2d4420def55afabe45b1c3c7483d/charset_normalizer-3.4.7-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:3534e7dcbdcf757da6b85a0bbf5b6868786d5982dd959b065e65481644817a18", size = 211564, upload-time = "2026-04-02T09:27:15.272Z" }, + { url = "https://files.pythonhosted.org/packages/44/86/58e6f13ce26cc3b8f4a36b94a0f22ae2f00a72534520f4ae6857c4b81f89/charset_normalizer-3.4.7-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:e8ac484bf18ce6975760921bb6148041faa8fef0547200386ea0b52b5d27bf7b", size = 211276, upload-time = "2026-04-02T09:27:16.834Z" }, + { url = "https://files.pythonhosted.org/packages/8f/fe/d17c32dc72e17e155e06883efa84514ca375f8a528ba2546bee73fc4df81/charset_normalizer-3.4.7-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:a5fe03b42827c13cdccd08e6c0247b6a6d4b5e3cdc53fd1749f5896adcdc2356", size = 201238, upload-time = "2026-04-02T09:27:18.229Z" }, + { url = "https://files.pythonhosted.org/packages/6a/29/f33daa50b06525a237451cdb6c69da366c381a3dadcd833fa5676bc468b3/charset_normalizer-3.4.7-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:2d6eb928e13016cea4f1f21d1e10c1cebd5a421bc57ddf5b1142ae3f86824fab", size = 230189, upload-time = "2026-04-02T09:27:19.445Z" }, + { url = "https://files.pythonhosted.org/packages/b6/6e/52c84015394a6a0bdcd435210a7e944c5f94ea1055f5cc5d56c5fe368e7b/charset_normalizer-3.4.7-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:e74327fb75de8986940def6e8dee4f127cc9752bee7355bb323cc5b2659b6d46", size = 211352, upload-time = "2026-04-02T09:27:20.79Z" }, + { url = "https://files.pythonhosted.org/packages/8c/d7/4353be581b373033fb9198bf1da3cf8f09c1082561e8e922aa7b39bf9fe8/charset_normalizer-3.4.7-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:d6038d37043bced98a66e68d3aa2b6a35505dc01328cd65217cefe82f25def44", size = 227024, upload-time = "2026-04-02T09:27:22.063Z" }, + { url = "https://files.pythonhosted.org/packages/30/45/99d18aa925bd1740098ccd3060e238e21115fffbfdcb8f3ece837d0ace6c/charset_normalizer-3.4.7-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:7579e913a5339fb8fa133f6bbcfd8e6749696206cf05acdbdca71a1b436d8e72", size = 217869, upload-time = "2026-04-02T09:27:23.486Z" }, + { url = "https://files.pythonhosted.org/packages/5c/05/5ee478aa53f4bb7996482153d4bfe1b89e0f087f0ab6b294fcf92d595873/charset_normalizer-3.4.7-cp314-cp314-win32.whl", hash = "sha256:5b77459df20e08151cd6f8b9ef8ef1f961ef73d85c21a555c7eed5b79410ec10", size = 148541, upload-time = "2026-04-02T09:27:25.146Z" }, + { url = "https://files.pythonhosted.org/packages/48/77/72dcb0921b2ce86420b2d79d454c7022bf5be40202a2a07906b9f2a35c97/charset_normalizer-3.4.7-cp314-cp314-win_amd64.whl", hash = "sha256:92a0a01ead5e668468e952e4238cccd7c537364eb7d851ab144ab6627dbbe12f", size = 159634, upload-time = "2026-04-02T09:27:26.642Z" }, + { url = "https://files.pythonhosted.org/packages/c6/a3/c2369911cd72f02386e4e340770f6e158c7980267da16af8f668217abaa0/charset_normalizer-3.4.7-cp314-cp314-win_arm64.whl", hash = "sha256:67f6279d125ca0046a7fd386d01b311c6363844deac3e5b069b514ba3e63c246", size = 148384, upload-time = "2026-04-02T09:27:28.271Z" }, + { url = "https://files.pythonhosted.org/packages/94/09/7e8a7f73d24dba1f0035fbbf014d2c36828fc1bf9c88f84093e57d315935/charset_normalizer-3.4.7-cp314-cp314t-macosx_10_15_universal2.whl", hash = "sha256:effc3f449787117233702311a1b7d8f59cba9ced946ba727bdc329ec69028e24", size = 330133, upload-time = "2026-04-02T09:27:29.474Z" }, + { url = "https://files.pythonhosted.org/packages/8d/da/96975ddb11f8e977f706f45cddd8540fd8242f71ecdb5d18a80723dcf62c/charset_normalizer-3.4.7-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:fbccdc05410c9ee21bbf16a35f4c1d16123dcdeb8a1d38f33654fa21d0234f79", size = 216257, upload-time = "2026-04-02T09:27:30.793Z" }, + { url = "https://files.pythonhosted.org/packages/e5/e8/1d63bf8ef2d388e95c64b2098f45f84758f6d102a087552da1485912637b/charset_normalizer-3.4.7-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:733784b6d6def852c814bce5f318d25da2ee65dd4839a0718641c696e09a2960", size = 234851, upload-time = "2026-04-02T09:27:32.44Z" }, + { url = "https://files.pythonhosted.org/packages/9b/40/e5ff04233e70da2681fa43969ad6f66ca5611d7e669be0246c4c7aaf6dc8/charset_normalizer-3.4.7-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a89c23ef8d2c6b27fd200a42aa4ac72786e7c60d40efdc76e6011260b6e949c4", size = 233393, upload-time = "2026-04-02T09:27:34.03Z" }, + { url = "https://files.pythonhosted.org/packages/be/c1/06c6c49d5a5450f76899992f1ee40b41d076aee9279b49cf9974d2f313d5/charset_normalizer-3.4.7-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6c114670c45346afedc0d947faf3c7f701051d2518b943679c8ff88befe14f8e", size = 223251, upload-time = "2026-04-02T09:27:35.369Z" }, + { url = "https://files.pythonhosted.org/packages/2b/9f/f2ff16fb050946169e3e1f82134d107e5d4ae72647ec8a1b1446c148480f/charset_normalizer-3.4.7-cp314-cp314t-manylinux_2_31_armv7l.whl", hash = "sha256:a180c5e59792af262bf263b21a3c49353f25945d8d9f70628e73de370d55e1e1", size = 206609, upload-time = "2026-04-02T09:27:36.661Z" }, + { url = "https://files.pythonhosted.org/packages/69/d5/a527c0cd8d64d2eab7459784fb4169a0ac76e5a6fc5237337982fd61347e/charset_normalizer-3.4.7-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:3c9a494bc5ec77d43cea229c4f6db1e4d8fe7e1bbffa8b6f0f0032430ff8ab44", size = 220014, upload-time = "2026-04-02T09:27:38.019Z" }, + { url = "https://files.pythonhosted.org/packages/7e/80/8a7b8104a3e203074dc9aa2c613d4b726c0e136bad1cc734594b02867972/charset_normalizer-3.4.7-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:8d828b6667a32a728a1ad1d93957cdf37489c57b97ae6c4de2860fa749b8fc1e", size = 218979, upload-time = "2026-04-02T09:27:39.37Z" }, + { url = "https://files.pythonhosted.org/packages/02/9a/b759b503d507f375b2b5c153e4d2ee0a75aa215b7f2489cf314f4541f2c0/charset_normalizer-3.4.7-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:cf1493cd8607bec4d8a7b9b004e699fcf8f9103a9284cc94962cb73d20f9d4a3", size = 209238, upload-time = "2026-04-02T09:27:40.722Z" }, + { url = "https://files.pythonhosted.org/packages/c2/4e/0f3f5d47b86bdb79256e7290b26ac847a2832d9a4033f7eb2cd4bcf4bb5b/charset_normalizer-3.4.7-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:0c96c3b819b5c3e9e165495db84d41914d6894d55181d2d108cc1a69bfc9cce0", size = 236110, upload-time = "2026-04-02T09:27:42.33Z" }, + { url = "https://files.pythonhosted.org/packages/96/23/bce28734eb3ed2c91dcf93abeb8a5cf393a7b2749725030bb630e554fdd8/charset_normalizer-3.4.7-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:752a45dc4a6934060b3b0dab47e04edc3326575f82be64bc4fc293914566503e", size = 219824, upload-time = "2026-04-02T09:27:43.924Z" }, + { url = "https://files.pythonhosted.org/packages/2c/6f/6e897c6984cc4d41af319b077f2f600fc8214eb2fe2d6bcb79141b882400/charset_normalizer-3.4.7-cp314-cp314t-musllinux_1_2_s390x.whl", hash = "sha256:8778f0c7a52e56f75d12dae53ae320fae900a8b9b4164b981b9c5ce059cd1fcb", size = 233103, upload-time = "2026-04-02T09:27:45.348Z" }, + { url = "https://files.pythonhosted.org/packages/76/22/ef7bd0fe480a0ae9b656189ec00744b60933f68b4f42a7bb06589f6f576a/charset_normalizer-3.4.7-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:ce3412fbe1e31eb81ea42f4169ed94861c56e643189e1e75f0041f3fe7020abe", size = 225194, upload-time = "2026-04-02T09:27:46.706Z" }, + { url = "https://files.pythonhosted.org/packages/c5/a7/0e0ab3e0b5bc1219bd80a6a0d4d72ca74d9250cb2382b7c699c147e06017/charset_normalizer-3.4.7-cp314-cp314t-win32.whl", hash = "sha256:c03a41a8784091e67a39648f70c5f97b5b6a37f216896d44d2cdcb82615339a0", size = 159827, upload-time = "2026-04-02T09:27:48.053Z" }, + { url = "https://files.pythonhosted.org/packages/7a/1d/29d32e0fb40864b1f878c7f5a0b343ae676c6e2b271a2d55cc3a152391da/charset_normalizer-3.4.7-cp314-cp314t-win_amd64.whl", hash = "sha256:03853ed82eeebbce3c2abfdbc98c96dc205f32a79627688ac9a27370ea61a49c", size = 174168, upload-time = "2026-04-02T09:27:49.795Z" }, + { url = "https://files.pythonhosted.org/packages/de/32/d92444ad05c7a6e41fb2036749777c163baf7a0301a040cb672d6b2b1ae9/charset_normalizer-3.4.7-cp314-cp314t-win_arm64.whl", hash = "sha256:c35abb8bfff0185efac5878da64c45dafd2b37fb0383add1be155a763c1f083d", size = 153018, upload-time = "2026-04-02T09:27:51.116Z" }, + { url = "https://files.pythonhosted.org/packages/db/8f/61959034484a4a7c527811f4721e75d02d653a35afb0b6054474d8185d4c/charset_normalizer-3.4.7-py3-none-any.whl", hash = "sha256:3dce51d0f5e7951f8bb4900c257dad282f49190fdbebecd4ba99bcc41fef404d", size = 61958, upload-time = "2026-04-02T09:28:37.794Z" }, +] + +[[package]] +name = "click" +version = "8.3.3" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "colorama", marker = "sys_platform == 'win32'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/bb/63/f9e1ea081ce35720d8b92acde70daaedace594dc93b693c869e0d5910718/click-8.3.3.tar.gz", hash = "sha256:398329ad4837b2ff7cbe1dd166a4c0f8900c3ca3a218de04466f38f6497f18a2", size = 328061, upload-time = "2026-04-22T15:11:27.506Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ae/44/c1221527f6a71a01ec6fbad7fa78f1d50dfa02217385cf0fa3eec7087d59/click-8.3.3-py3-none-any.whl", hash = "sha256:a2bf429bb3033c89fa4936ffb35d5cb471e3719e1f3c8a7c3fff0b8314305613", size = 110502, upload-time = "2026-04-22T15:11:25.044Z" }, +] + +[[package]] +name = "colorama" +version = "0.4.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" }, +] + +[[package]] +name = "duckdb" +version = "1.5.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/0c/66/744b4931b799a42f8cb9bc7a6f169e7b8e51195b62b246db407fd90bf15f/duckdb-1.5.2.tar.gz", hash = "sha256:638da0d5102b6cb6f7d47f83d0600708ac1d3cb46c5e9aaabc845f9ba4d69246", size = 18017166, upload-time = "2026-04-13T11:30:09.065Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/98/f2/e3d742808f138d374be4bb516fade3d1f33749b813650810ab7885cdc363/duckdb-1.5.2-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:4420b3f47027a7849d0e1815532007f377fa95ee5810b47ea717d35525c12f79", size = 30064879, upload-time = "2026-04-13T11:29:30.763Z" }, + { url = "https://files.pythonhosted.org/packages/72/0d/f3dc1cf97e1267ca15e4307d456f96ce583961f0703fd75e62b2ad8d64fa/duckdb-1.5.2-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:bb42e6ed543902e14eae647850da24103a89f0bc2587dec5601b1c1f213bd2ed", size = 15969327, upload-time = "2026-04-13T11:29:33.481Z" }, + { url = "https://files.pythonhosted.org/packages/b1/e0/d5418def53ae4e05a63075705ff44ed5af5a1a5932627eb2b600c5df1c93/duckdb-1.5.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:98c0535cd6d901f61a5ea3c2e26a1fd28482953d794deb183daf568e3aa5dda6", size = 14225107, upload-time = "2026-04-13T11:29:35.882Z" }, + { url = "https://files.pythonhosted.org/packages/16/a7/15aaa59dbecc35e9711980fcdbf525b32a52470b32d18ef678193a146213/duckdb-1.5.2-cp313-cp313-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:486c862bf7f163c0110b6d85b3e5c031d224a671cca468f12ebb1d3a348f6b39", size = 19313433, upload-time = "2026-04-13T11:29:38.367Z" }, + { url = "https://files.pythonhosted.org/packages/bd/21/d903cc63a5140c822b7b62b373a87dc557e60c29b321dfb435061c5e67cf/duckdb-1.5.2-cp313-cp313-manylinux_2_26_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:70631c847ca918ee710ec874241b00cf9d2e5be90762cbb2a0389f17823c08f7", size = 21429837, upload-time = "2026-04-13T11:29:41.135Z" }, + { url = "https://files.pythonhosted.org/packages/e3/0a/b770d1f60c70597302130d6247f418549b7094251a02348fbaf1c7e147ae/duckdb-1.5.2-cp313-cp313-win_amd64.whl", hash = "sha256:52a21823f3fbb52f0f0e5425e20b07391ad882464b955879499b5ff0b45a376b", size = 13107699, upload-time = "2026-04-13T11:29:43.905Z" }, + { url = "https://files.pythonhosted.org/packages/d9/cf/e200fe431d700962d1a908d2ce89f53ccee1cc8db260174ae663ba09686b/duckdb-1.5.2-cp313-cp313-win_arm64.whl", hash = "sha256:411ad438bd4140f189a10e7f515781335962c5d18bd07837dc6d202e3985253d", size = 13927646, upload-time = "2026-04-13T11:29:46.598Z" }, + { url = "https://files.pythonhosted.org/packages/83/a1/f6286c67726cc1ea60a6e3c0d9fbc66527dde24ae089a51bbe298b13ca78/duckdb-1.5.2-cp314-cp314-macosx_10_15_universal2.whl", hash = "sha256:6b0fe75c148000f060aa1a27b293cacc0ea08cc1cad724fbf2143d56070a3785", size = 30078598, upload-time = "2026-04-13T11:29:49.828Z" }, + { url = "https://files.pythonhosted.org/packages/de/6a/59febb02f21a4a5c6b0b0099ef7c965fdd5e61e4904cf813809bb792e35f/duckdb-1.5.2-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:35579b8e3a064b5eaf15b0eafc558056a13f79a0a62e34cc4baf57119daecfec", size = 15975120, upload-time = "2026-04-13T11:29:52.631Z" }, + { url = "https://files.pythonhosted.org/packages/09/70/ce750854d37bb5a45cccbb2c3cb04df4af56aea8fc30a2499bb643b4a9c0/duckdb-1.5.2-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:ea58ff5b0880593a280cf5511734b17711b32ee1f58b47d726e8600848358160", size = 14227762, upload-time = "2026-04-13T11:29:55.564Z" }, + { url = "https://files.pythonhosted.org/packages/28/dc/ad45ac3c0b6c4687dc649e8f6cf01af1c8b0443932a39b2abb4ebcb3babd/duckdb-1.5.2-cp314-cp314-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ef461bca07313412dc09961c4a4757a851f56b95ac01c58fac6007632b7b94f2", size = 19315668, upload-time = "2026-04-13T11:29:58.427Z" }, + { url = "https://files.pythonhosted.org/packages/cc/b1/1464f468d2e5813f5808de95df9d3113a645a5bfa2ffcaecbc542ddae272/duckdb-1.5.2-cp314-cp314-manylinux_2_26_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:be37680ddb380015cb37318e378c53511c45c4f0d8fac5599d22b7d092b9217a", size = 21434056, upload-time = "2026-04-13T11:30:01.238Z" }, + { url = "https://files.pythonhosted.org/packages/ce/32/6673607e024722473fa7aafdd29c0e3dd231dd528f6cd8b5797fbeeb229d/duckdb-1.5.2-cp314-cp314-win_amd64.whl", hash = "sha256:0b291786014df1133f8f18b9df4d004484613146e858d71a21791e0fcca16cf4", size = 13633667, upload-time = "2026-04-13T11:30:04.05Z" }, + { url = "https://files.pythonhosted.org/packages/7a/e3/9d34173ec068631faea3ea6e73050700729363e7e33306a9a3218e5cdc61/duckdb-1.5.2-cp314-cp314-win_arm64.whl", hash = "sha256:c9f3e0b71b8a50fccfb42794899285d9d318ce2503782b9dd54868e5ecd0ad31", size = 14402513, upload-time = "2026-04-13T11:30:06.609Z" }, +] + +[[package]] +name = "gitdb" +version = "4.0.12" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "smmap" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/72/94/63b0fc47eb32792c7ba1fe1b694daec9a63620db1e313033d18140c2320a/gitdb-4.0.12.tar.gz", hash = "sha256:5ef71f855d191a3326fcfbc0d5da835f26b13fbcba60c32c21091c349ffdb571", size = 394684, upload-time = "2025-01-02T07:20:46.413Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a0/61/5c78b91c3143ed5c14207f463aecfc8f9dbb5092fb2869baf37c273b2705/gitdb-4.0.12-py3-none-any.whl", hash = "sha256:67073e15955400952c6565cc3e707c554a4eea2e428946f7a4c162fab9bd9bcf", size = 62794, upload-time = "2025-01-02T07:20:43.624Z" }, +] + +[[package]] +name = "gitpython" +version = "3.1.47" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "gitdb" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c1/bd/50db468e9b1310529a19fce651b3b0e753b5c07954d486cba31bbee9a5d5/gitpython-3.1.47.tar.gz", hash = "sha256:dba27f922bd2b42cb54c87a8ab3cb6beb6bf07f3d564e21ac848913a05a8a3cd", size = 216978, upload-time = "2026-04-22T02:44:44.059Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f2/c5/a1bc0996af85757903cf2bf444a7824e68e0035ce63fb41d6f76f9def68b/gitpython-3.1.47-py3-none-any.whl", hash = "sha256:489f590edfd6d20571b2c0e72c6a6ac6915ee8b8cd04572330e3842207a78905", size = 209547, upload-time = "2026-04-22T02:44:41.271Z" }, +] + +[[package]] +name = "idna" +version = "3.13" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ce/cc/762dfb036166873f0059f3b7de4565e1b5bc3d6f28a414c13da27e442f99/idna-3.13.tar.gz", hash = "sha256:585ea8fe5d69b9181ec1afba340451fba6ba764af97026f92a91d4eef164a242", size = 194210, upload-time = "2026-04-22T16:42:42.314Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/5d/13/ad7d7ca3808a898b4612b6fe93cde56b53f3034dcde235acb1f0e1df24c6/idna-3.13-py3-none-any.whl", hash = "sha256:892ea0cde124a99ce773decba204c5552b69c3c67ffd5f232eb7696135bc8bb3", size = 68629, upload-time = "2026-04-22T16:42:40.909Z" }, +] + +[[package]] +name = "jinja2" +version = "3.1.6" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "markupsafe" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115, upload-time = "2025-03-05T20:05:02.478Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899, upload-time = "2025-03-05T20:05:00.369Z" }, +] + +[[package]] +name = "jsonschema" +version = "4.26.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "attrs" }, + { name = "jsonschema-specifications" }, + { name = "referencing" }, + { name = "rpds-py" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/b3/fc/e067678238fa451312d4c62bf6e6cf5ec56375422aee02f9cb5f909b3047/jsonschema-4.26.0.tar.gz", hash = "sha256:0c26707e2efad8aa1bfc5b7ce170f3fccc2e4918ff85989ba9ffa9facb2be326", size = 366583, upload-time = "2026-01-07T13:41:07.246Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/69/90/f63fb5873511e014207a475e2bb4e8b2e570d655b00ac19a9a0ca0a385ee/jsonschema-4.26.0-py3-none-any.whl", hash = "sha256:d489f15263b8d200f8387e64b4c3a75f06629559fb73deb8fdfb525f2dab50ce", size = 90630, upload-time = "2026-01-07T13:41:05.306Z" }, +] + +[[package]] +name = "jsonschema-specifications" +version = "2025.9.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "referencing" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/19/74/a633ee74eb36c44aa6d1095e7cc5569bebf04342ee146178e2d36600708b/jsonschema_specifications-2025.9.1.tar.gz", hash = "sha256:b540987f239e745613c7a9176f3edb72b832a4ac465cf02712288397832b5e8d", size = 32855, upload-time = "2025-09-08T01:34:59.186Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/41/45/1a4ed80516f02155c51f51e8cedb3c1902296743db0bbc66608a0db2814f/jsonschema_specifications-2025.9.1-py3-none-any.whl", hash = "sha256:98802fee3a11ee76ecaca44429fda8a41bff98b00a0f2838151b113f210cc6fe", size = 18437, upload-time = "2025-09-08T01:34:57.871Z" }, +] + +[[package]] +name = "markupsafe" +version = "3.0.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/7e/99/7690b6d4034fffd95959cbe0c02de8deb3098cc577c67bb6a24fe5d7caa7/markupsafe-3.0.3.tar.gz", hash = "sha256:722695808f4b6457b320fdc131280796bdceb04ab50fe1795cd540799ebe1698", size = 80313, upload-time = "2025-09-27T18:37:40.426Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/38/2f/907b9c7bbba283e68f20259574b13d005c121a0fa4c175f9bed27c4597ff/markupsafe-3.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:e1cf1972137e83c5d4c136c43ced9ac51d0e124706ee1c8aa8532c1287fa8795", size = 11622, upload-time = "2025-09-27T18:36:41.777Z" }, + { url = "https://files.pythonhosted.org/packages/9c/d9/5f7756922cdd676869eca1c4e3c0cd0df60ed30199ffd775e319089cb3ed/markupsafe-3.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:116bb52f642a37c115f517494ea5feb03889e04df47eeff5b130b1808ce7c219", size = 12029, upload-time = "2025-09-27T18:36:43.257Z" }, + { url = "https://files.pythonhosted.org/packages/00/07/575a68c754943058c78f30db02ee03a64b3c638586fba6a6dd56830b30a3/markupsafe-3.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:133a43e73a802c5562be9bbcd03d090aa5a1fe899db609c29e8c8d815c5f6de6", size = 24374, upload-time = "2025-09-27T18:36:44.508Z" }, + { url = "https://files.pythonhosted.org/packages/a9/21/9b05698b46f218fc0e118e1f8168395c65c8a2c750ae2bab54fc4bd4e0e8/markupsafe-3.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ccfcd093f13f0f0b7fdd0f198b90053bf7b2f02a3927a30e63f3ccc9df56b676", size = 22980, upload-time = "2025-09-27T18:36:45.385Z" }, + { url = "https://files.pythonhosted.org/packages/7f/71/544260864f893f18b6827315b988c146b559391e6e7e8f7252839b1b846a/markupsafe-3.0.3-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:509fa21c6deb7a7a273d629cf5ec029bc209d1a51178615ddf718f5918992ab9", size = 21990, upload-time = "2025-09-27T18:36:46.916Z" }, + { url = "https://files.pythonhosted.org/packages/c2/28/b50fc2f74d1ad761af2f5dcce7492648b983d00a65b8c0e0cb457c82ebbe/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:a4afe79fb3de0b7097d81da19090f4df4f8d3a2b3adaa8764138aac2e44f3af1", size = 23784, upload-time = "2025-09-27T18:36:47.884Z" }, + { url = "https://files.pythonhosted.org/packages/ed/76/104b2aa106a208da8b17a2fb72e033a5a9d7073c68f7e508b94916ed47a9/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:795e7751525cae078558e679d646ae45574b47ed6e7771863fcc079a6171a0fc", size = 21588, upload-time = "2025-09-27T18:36:48.82Z" }, + { url = "https://files.pythonhosted.org/packages/b5/99/16a5eb2d140087ebd97180d95249b00a03aa87e29cc224056274f2e45fd6/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:8485f406a96febb5140bfeca44a73e3ce5116b2501ac54fe953e488fb1d03b12", size = 23041, upload-time = "2025-09-27T18:36:49.797Z" }, + { url = "https://files.pythonhosted.org/packages/19/bc/e7140ed90c5d61d77cea142eed9f9c303f4c4806f60a1044c13e3f1471d0/markupsafe-3.0.3-cp313-cp313-win32.whl", hash = "sha256:bdd37121970bfd8be76c5fb069c7751683bdf373db1ed6c010162b2a130248ed", size = 14543, upload-time = "2025-09-27T18:36:51.584Z" }, + { url = "https://files.pythonhosted.org/packages/05/73/c4abe620b841b6b791f2edc248f556900667a5a1cf023a6646967ae98335/markupsafe-3.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:9a1abfdc021a164803f4d485104931fb8f8c1efd55bc6b748d2f5774e78b62c5", size = 15113, upload-time = "2025-09-27T18:36:52.537Z" }, + { url = "https://files.pythonhosted.org/packages/f0/3a/fa34a0f7cfef23cf9500d68cb7c32dd64ffd58a12b09225fb03dd37d5b80/markupsafe-3.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:7e68f88e5b8799aa49c85cd116c932a1ac15caaa3f5db09087854d218359e485", size = 13911, upload-time = "2025-09-27T18:36:53.513Z" }, + { url = "https://files.pythonhosted.org/packages/e4/d7/e05cd7efe43a88a17a37b3ae96e79a19e846f3f456fe79c57ca61356ef01/markupsafe-3.0.3-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:218551f6df4868a8d527e3062d0fb968682fe92054e89978594c28e642c43a73", size = 11658, upload-time = "2025-09-27T18:36:54.819Z" }, + { url = "https://files.pythonhosted.org/packages/99/9e/e412117548182ce2148bdeacdda3bb494260c0b0184360fe0d56389b523b/markupsafe-3.0.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:3524b778fe5cfb3452a09d31e7b5adefeea8c5be1d43c4f810ba09f2ceb29d37", size = 12066, upload-time = "2025-09-27T18:36:55.714Z" }, + { url = "https://files.pythonhosted.org/packages/bc/e6/fa0ffcda717ef64a5108eaa7b4f5ed28d56122c9a6d70ab8b72f9f715c80/markupsafe-3.0.3-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4e885a3d1efa2eadc93c894a21770e4bc67899e3543680313b09f139e149ab19", size = 25639, upload-time = "2025-09-27T18:36:56.908Z" }, + { url = "https://files.pythonhosted.org/packages/96/ec/2102e881fe9d25fc16cb4b25d5f5cde50970967ffa5dddafdb771237062d/markupsafe-3.0.3-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8709b08f4a89aa7586de0aadc8da56180242ee0ada3999749b183aa23df95025", size = 23569, upload-time = "2025-09-27T18:36:57.913Z" }, + { url = "https://files.pythonhosted.org/packages/4b/30/6f2fce1f1f205fc9323255b216ca8a235b15860c34b6798f810f05828e32/markupsafe-3.0.3-cp313-cp313t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:b8512a91625c9b3da6f127803b166b629725e68af71f8184ae7e7d54686a56d6", size = 23284, upload-time = "2025-09-27T18:36:58.833Z" }, + { url = "https://files.pythonhosted.org/packages/58/47/4a0ccea4ab9f5dcb6f79c0236d954acb382202721e704223a8aafa38b5c8/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:9b79b7a16f7fedff2495d684f2b59b0457c3b493778c9eed31111be64d58279f", size = 24801, upload-time = "2025-09-27T18:36:59.739Z" }, + { url = "https://files.pythonhosted.org/packages/6a/70/3780e9b72180b6fecb83a4814d84c3bf4b4ae4bf0b19c27196104149734c/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_riscv64.whl", hash = "sha256:12c63dfb4a98206f045aa9563db46507995f7ef6d83b2f68eda65c307c6829eb", size = 22769, upload-time = "2025-09-27T18:37:00.719Z" }, + { url = "https://files.pythonhosted.org/packages/98/c5/c03c7f4125180fc215220c035beac6b9cb684bc7a067c84fc69414d315f5/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:8f71bc33915be5186016f675cd83a1e08523649b0e33efdb898db577ef5bb009", size = 23642, upload-time = "2025-09-27T18:37:01.673Z" }, + { url = "https://files.pythonhosted.org/packages/80/d6/2d1b89f6ca4bff1036499b1e29a1d02d282259f3681540e16563f27ebc23/markupsafe-3.0.3-cp313-cp313t-win32.whl", hash = "sha256:69c0b73548bc525c8cb9a251cddf1931d1db4d2258e9599c28c07ef3580ef354", size = 14612, upload-time = "2025-09-27T18:37:02.639Z" }, + { url = "https://files.pythonhosted.org/packages/2b/98/e48a4bfba0a0ffcf9925fe2d69240bfaa19c6f7507b8cd09c70684a53c1e/markupsafe-3.0.3-cp313-cp313t-win_amd64.whl", hash = "sha256:1b4b79e8ebf6b55351f0d91fe80f893b4743f104bff22e90697db1590e47a218", size = 15200, upload-time = "2025-09-27T18:37:03.582Z" }, + { url = "https://files.pythonhosted.org/packages/0e/72/e3cc540f351f316e9ed0f092757459afbc595824ca724cbc5a5d4263713f/markupsafe-3.0.3-cp313-cp313t-win_arm64.whl", hash = "sha256:ad2cf8aa28b8c020ab2fc8287b0f823d0a7d8630784c31e9ee5edea20f406287", size = 13973, upload-time = "2025-09-27T18:37:04.929Z" }, + { url = "https://files.pythonhosted.org/packages/33/8a/8e42d4838cd89b7dde187011e97fe6c3af66d8c044997d2183fbd6d31352/markupsafe-3.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:eaa9599de571d72e2daf60164784109f19978b327a3910d3e9de8c97b5b70cfe", size = 11619, upload-time = "2025-09-27T18:37:06.342Z" }, + { url = "https://files.pythonhosted.org/packages/b5/64/7660f8a4a8e53c924d0fa05dc3a55c9cee10bbd82b11c5afb27d44b096ce/markupsafe-3.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:c47a551199eb8eb2121d4f0f15ae0f923d31350ab9280078d1e5f12b249e0026", size = 12029, upload-time = "2025-09-27T18:37:07.213Z" }, + { url = "https://files.pythonhosted.org/packages/da/ef/e648bfd021127bef5fa12e1720ffed0c6cbb8310c8d9bea7266337ff06de/markupsafe-3.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f34c41761022dd093b4b6896d4810782ffbabe30f2d443ff5f083e0cbbb8c737", size = 24408, upload-time = "2025-09-27T18:37:09.572Z" }, + { url = "https://files.pythonhosted.org/packages/41/3c/a36c2450754618e62008bf7435ccb0f88053e07592e6028a34776213d877/markupsafe-3.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:457a69a9577064c05a97c41f4e65148652db078a3a509039e64d3467b9e7ef97", size = 23005, upload-time = "2025-09-27T18:37:10.58Z" }, + { url = "https://files.pythonhosted.org/packages/bc/20/b7fdf89a8456b099837cd1dc21974632a02a999ec9bf7ca3e490aacd98e7/markupsafe-3.0.3-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e8afc3f2ccfa24215f8cb28dcf43f0113ac3c37c2f0f0806d8c70e4228c5cf4d", size = 22048, upload-time = "2025-09-27T18:37:11.547Z" }, + { url = "https://files.pythonhosted.org/packages/9a/a7/591f592afdc734f47db08a75793a55d7fbcc6902a723ae4cfbab61010cc5/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:ec15a59cf5af7be74194f7ab02d0f59a62bdcf1a537677ce67a2537c9b87fcda", size = 23821, upload-time = "2025-09-27T18:37:12.48Z" }, + { url = "https://files.pythonhosted.org/packages/7d/33/45b24e4f44195b26521bc6f1a82197118f74df348556594bd2262bda1038/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:0eb9ff8191e8498cca014656ae6b8d61f39da5f95b488805da4bb029cccbfbaf", size = 21606, upload-time = "2025-09-27T18:37:13.485Z" }, + { url = "https://files.pythonhosted.org/packages/ff/0e/53dfaca23a69fbfbbf17a4b64072090e70717344c52eaaaa9c5ddff1e5f0/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:2713baf880df847f2bece4230d4d094280f4e67b1e813eec43b4c0e144a34ffe", size = 23043, upload-time = "2025-09-27T18:37:14.408Z" }, + { url = "https://files.pythonhosted.org/packages/46/11/f333a06fc16236d5238bfe74daccbca41459dcd8d1fa952e8fbd5dccfb70/markupsafe-3.0.3-cp314-cp314-win32.whl", hash = "sha256:729586769a26dbceff69f7a7dbbf59ab6572b99d94576a5592625d5b411576b9", size = 14747, upload-time = "2025-09-27T18:37:15.36Z" }, + { url = "https://files.pythonhosted.org/packages/28/52/182836104b33b444e400b14f797212f720cbc9ed6ba34c800639d154e821/markupsafe-3.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:bdc919ead48f234740ad807933cdf545180bfbe9342c2bb451556db2ed958581", size = 15341, upload-time = "2025-09-27T18:37:16.496Z" }, + { url = "https://files.pythonhosted.org/packages/6f/18/acf23e91bd94fd7b3031558b1f013adfa21a8e407a3fdb32745538730382/markupsafe-3.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:5a7d5dc5140555cf21a6fefbdbf8723f06fcd2f63ef108f2854de715e4422cb4", size = 14073, upload-time = "2025-09-27T18:37:17.476Z" }, + { url = "https://files.pythonhosted.org/packages/3c/f0/57689aa4076e1b43b15fdfa646b04653969d50cf30c32a102762be2485da/markupsafe-3.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:1353ef0c1b138e1907ae78e2f6c63ff67501122006b0f9abad68fda5f4ffc6ab", size = 11661, upload-time = "2025-09-27T18:37:18.453Z" }, + { url = "https://files.pythonhosted.org/packages/89/c3/2e67a7ca217c6912985ec766c6393b636fb0c2344443ff9d91404dc4c79f/markupsafe-3.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:1085e7fbddd3be5f89cc898938f42c0b3c711fdcb37d75221de2666af647c175", size = 12069, upload-time = "2025-09-27T18:37:19.332Z" }, + { url = "https://files.pythonhosted.org/packages/f0/00/be561dce4e6ca66b15276e184ce4b8aec61fe83662cce2f7d72bd3249d28/markupsafe-3.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1b52b4fb9df4eb9ae465f8d0c228a00624de2334f216f178a995ccdcf82c4634", size = 25670, upload-time = "2025-09-27T18:37:20.245Z" }, + { url = "https://files.pythonhosted.org/packages/50/09/c419f6f5a92e5fadde27efd190eca90f05e1261b10dbd8cbcb39cd8ea1dc/markupsafe-3.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:fed51ac40f757d41b7c48425901843666a6677e3e8eb0abcff09e4ba6e664f50", size = 23598, upload-time = "2025-09-27T18:37:21.177Z" }, + { url = "https://files.pythonhosted.org/packages/22/44/a0681611106e0b2921b3033fc19bc53323e0b50bc70cffdd19f7d679bb66/markupsafe-3.0.3-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:f190daf01f13c72eac4efd5c430a8de82489d9cff23c364c3ea822545032993e", size = 23261, upload-time = "2025-09-27T18:37:22.167Z" }, + { url = "https://files.pythonhosted.org/packages/5f/57/1b0b3f100259dc9fffe780cfb60d4be71375510e435efec3d116b6436d43/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:e56b7d45a839a697b5eb268c82a71bd8c7f6c94d6fd50c3d577fa39a9f1409f5", size = 24835, upload-time = "2025-09-27T18:37:23.296Z" }, + { url = "https://files.pythonhosted.org/packages/26/6a/4bf6d0c97c4920f1597cc14dd720705eca0bf7c787aebc6bb4d1bead5388/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:f3e98bb3798ead92273dc0e5fd0f31ade220f59a266ffd8a4f6065e0a3ce0523", size = 22733, upload-time = "2025-09-27T18:37:24.237Z" }, + { url = "https://files.pythonhosted.org/packages/14/c7/ca723101509b518797fedc2fdf79ba57f886b4aca8a7d31857ba3ee8281f/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:5678211cb9333a6468fb8d8be0305520aa073f50d17f089b5b4b477ea6e67fdc", size = 23672, upload-time = "2025-09-27T18:37:25.271Z" }, + { url = "https://files.pythonhosted.org/packages/fb/df/5bd7a48c256faecd1d36edc13133e51397e41b73bb77e1a69deab746ebac/markupsafe-3.0.3-cp314-cp314t-win32.whl", hash = "sha256:915c04ba3851909ce68ccc2b8e2cd691618c4dc4c4232fb7982bca3f41fd8c3d", size = 14819, upload-time = "2025-09-27T18:37:26.285Z" }, + { url = "https://files.pythonhosted.org/packages/1a/8a/0402ba61a2f16038b48b39bccca271134be00c5c9f0f623208399333c448/markupsafe-3.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4faffd047e07c38848ce017e8725090413cd80cbc23d86e55c587bf979e579c9", size = 15426, upload-time = "2025-09-27T18:37:27.316Z" }, + { url = "https://files.pythonhosted.org/packages/70/bc/6f1c2f612465f5fa89b95bead1f44dcb607670fd42891d8fdcd5d039f4f4/markupsafe-3.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:32001d6a8fc98c8cb5c947787c5d08b0a50663d139f1305bac5885d98d9b40fa", size = 14146, upload-time = "2025-09-27T18:37:28.327Z" }, +] + +[[package]] +name = "narwhals" +version = "2.20.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/e9/f3/257adc69a71011b4c8cda321b00f02c5bf1980ae38ffd05a58d9632d4de8/narwhals-2.20.0.tar.gz", hash = "sha256:c10994975fa7dc5a68c2cffcddbd5908fc8ebb2d463c5bab085309c0ee1f551e", size = 627848, upload-time = "2026-04-20T12:11:45.427Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d0/69/f24d3d1c38ad69e256138b4ec2452a8c7cf66be49dc214771ae99dd4f0a0/narwhals-2.20.0-py3-none-any.whl", hash = "sha256:16e750ea5507d4ba6e8d03455b5f93a535e0405976561baea235bca5dc9f475d", size = 449373, upload-time = "2026-04-20T12:11:43.596Z" }, +] + +[[package]] +name = "numpy" +version = "2.4.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d7/9f/b8cef5bffa569759033adda9481211426f12f53299629b410340795c2514/numpy-2.4.4.tar.gz", hash = "sha256:2d390634c5182175533585cc89f3608a4682ccb173cc9bb940b2881c8d6f8fa0", size = 20731587, upload-time = "2026-03-29T13:22:01.298Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/14/1d/d0a583ce4fefcc3308806a749a536c201ed6b5ad6e1322e227ee4848979d/numpy-2.4.4-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:08f2e31ed5e6f04b118e49821397f12767934cfdd12a1ce86a058f91e004ee50", size = 16684933, upload-time = "2026-03-29T13:19:22.47Z" }, + { url = "https://files.pythonhosted.org/packages/c1/62/2b7a48fbb745d344742c0277f01286dead15f3f68e4f359fbfcf7b48f70f/numpy-2.4.4-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:e823b8b6edc81e747526f70f71a9c0a07ac4e7ad13020aa736bb7c9d67196115", size = 14694532, upload-time = "2026-03-29T13:19:25.581Z" }, + { url = "https://files.pythonhosted.org/packages/e5/87/499737bfba066b4a3bebff24a8f1c5b2dee410b209bc6668c9be692580f0/numpy-2.4.4-cp313-cp313-macosx_14_0_arm64.whl", hash = "sha256:4a19d9dba1a76618dd86b164d608566f393f8ec6ac7c44f0cc879011c45e65af", size = 5199661, upload-time = "2026-03-29T13:19:28.31Z" }, + { url = "https://files.pythonhosted.org/packages/cd/da/464d551604320d1491bc345efed99b4b7034143a85787aab78d5691d5a0e/numpy-2.4.4-cp313-cp313-macosx_14_0_x86_64.whl", hash = "sha256:d2a8490669bfe99a233298348acc2d824d496dee0e66e31b66a6022c2ad74a5c", size = 6547539, upload-time = "2026-03-29T13:19:30.97Z" }, + { url = "https://files.pythonhosted.org/packages/7d/90/8d23e3b0dafd024bf31bdec225b3bb5c2dbfa6912f8a53b8659f21216cbf/numpy-2.4.4-cp313-cp313-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:45dbed2ab436a9e826e302fcdcbe9133f9b0006e5af7168afb8963a6520da103", size = 15668806, upload-time = "2026-03-29T13:19:33.887Z" }, + { url = "https://files.pythonhosted.org/packages/d1/73/a9d864e42a01896bb5974475438f16086be9ba1f0d19d0bb7a07427c4a8b/numpy-2.4.4-cp313-cp313-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c901b15172510173f5cb310eae652908340f8dede90fff9e3bf6c0d8dfd92f83", size = 16632682, upload-time = "2026-03-29T13:19:37.336Z" }, + { url = "https://files.pythonhosted.org/packages/34/fb/14570d65c3bde4e202a031210475ae9cde9b7686a2e7dc97ee67d2833b35/numpy-2.4.4-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:99d838547ace2c4aace6c4f76e879ddfe02bb58a80c1549928477862b7a6d6ed", size = 17019810, upload-time = "2026-03-29T13:19:40.963Z" }, + { url = "https://files.pythonhosted.org/packages/8a/77/2ba9d87081fd41f6d640c83f26fb7351e536b7ce6dd9061b6af5904e8e46/numpy-2.4.4-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:0aec54fd785890ecca25a6003fd9a5aed47ad607bbac5cd64f836ad8666f4959", size = 18357394, upload-time = "2026-03-29T13:19:44.859Z" }, + { url = "https://files.pythonhosted.org/packages/a2/23/52666c9a41708b0853fa3b1a12c90da38c507a3074883823126d4e9d5b30/numpy-2.4.4-cp313-cp313-win32.whl", hash = "sha256:07077278157d02f65c43b1b26a3886bce886f95d20aabd11f87932750dfb14ed", size = 5959556, upload-time = "2026-03-29T13:19:47.661Z" }, + { url = "https://files.pythonhosted.org/packages/57/fb/48649b4971cde70d817cf97a2a2fdc0b4d8308569f1dd2f2611959d2e0cf/numpy-2.4.4-cp313-cp313-win_amd64.whl", hash = "sha256:5c70f1cc1c4efbe316a572e2d8b9b9cc44e89b95f79ca3331553fbb63716e2bf", size = 12317311, upload-time = "2026-03-29T13:19:50.67Z" }, + { url = "https://files.pythonhosted.org/packages/ba/d8/11490cddd564eb4de97b4579ef6bfe6a736cc07e94c1598590ae25415e01/numpy-2.4.4-cp313-cp313-win_arm64.whl", hash = "sha256:ef4059d6e5152fa1a39f888e344c73fdc926e1b2dd58c771d67b0acfbf2aa67d", size = 10222060, upload-time = "2026-03-29T13:19:54.229Z" }, + { url = "https://files.pythonhosted.org/packages/99/5d/dab4339177a905aad3e2221c915b35202f1ec30d750dd2e5e9d9a72b804b/numpy-2.4.4-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:4bbc7f303d125971f60ec0aaad5e12c62d0d2c925f0ab1273debd0e4ba37aba5", size = 14822302, upload-time = "2026-03-29T13:19:57.585Z" }, + { url = "https://files.pythonhosted.org/packages/eb/e4/0564a65e7d3d97562ed6f9b0fd0fb0a6f559ee444092f105938b50043876/numpy-2.4.4-cp313-cp313t-macosx_14_0_arm64.whl", hash = "sha256:4d6d57903571f86180eb98f8f0c839fa9ebbfb031356d87f1361be91e433f5b7", size = 5327407, upload-time = "2026-03-29T13:20:00.601Z" }, + { url = "https://files.pythonhosted.org/packages/29/8d/35a3a6ce5ad371afa58b4700f1c820f8f279948cca32524e0a695b0ded83/numpy-2.4.4-cp313-cp313t-macosx_14_0_x86_64.whl", hash = "sha256:4636de7fd195197b7535f231b5de9e4b36d2c440b6e566d2e4e4746e6af0ca93", size = 6647631, upload-time = "2026-03-29T13:20:02.855Z" }, + { url = "https://files.pythonhosted.org/packages/f4/da/477731acbd5a58a946c736edfdabb2ac5b34c3d08d1ba1a7b437fa0884df/numpy-2.4.4-cp313-cp313t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ad2e2ef14e0b04e544ea2fa0a36463f847f113d314aa02e5b402fdf910ef309e", size = 15727691, upload-time = "2026-03-29T13:20:06.004Z" }, + { url = "https://files.pythonhosted.org/packages/e6/db/338535d9b152beabeb511579598418ba0212ce77cf9718edd70262cc4370/numpy-2.4.4-cp313-cp313t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:5a285b3b96f951841799528cd1f4f01cd70e7e0204b4abebac9463eecfcf2a40", size = 16681241, upload-time = "2026-03-29T13:20:09.417Z" }, + { url = "https://files.pythonhosted.org/packages/e2/a9/ad248e8f58beb7a0219b413c9c7d8151c5d285f7f946c3e26695bdbbe2df/numpy-2.4.4-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:f8474c4241bc18b750be2abea9d7a9ec84f46ef861dbacf86a4f6e043401f79e", size = 17085767, upload-time = "2026-03-29T13:20:13.126Z" }, + { url = "https://files.pythonhosted.org/packages/b5/1a/3b88ccd3694681356f70da841630e4725a7264d6a885c8d442a697e1146b/numpy-2.4.4-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:4e874c976154687c1f71715b034739b45c7711bec81db01914770373d125e392", size = 18403169, upload-time = "2026-03-29T13:20:17.096Z" }, + { url = "https://files.pythonhosted.org/packages/c2/c9/fcfd5d0639222c6eac7f304829b04892ef51c96a75d479214d77e3ce6e33/numpy-2.4.4-cp313-cp313t-win32.whl", hash = "sha256:9c585a1790d5436a5374bac930dad6ed244c046ed91b2b2a3634eb2971d21008", size = 6083477, upload-time = "2026-03-29T13:20:20.195Z" }, + { url = "https://files.pythonhosted.org/packages/d5/e3/3938a61d1c538aaec8ed6fd6323f57b0c2d2d2219512434c5c878db76553/numpy-2.4.4-cp313-cp313t-win_amd64.whl", hash = "sha256:93e15038125dc1e5345d9b5b68aa7f996ec33b98118d18c6ca0d0b7d6198b7e8", size = 12457487, upload-time = "2026-03-29T13:20:22.946Z" }, + { url = "https://files.pythonhosted.org/packages/97/6a/7e345032cc60501721ef94e0e30b60f6b0bd601f9174ebd36389a2b86d40/numpy-2.4.4-cp313-cp313t-win_arm64.whl", hash = "sha256:0dfd3f9d3adbe2920b68b5cd3d51444e13a10792ec7154cd0a2f6e74d4ab3233", size = 10292002, upload-time = "2026-03-29T13:20:25.909Z" }, + { url = "https://files.pythonhosted.org/packages/6e/06/c54062f85f673dd5c04cbe2f14c3acb8c8b95e3384869bb8cc9bff8cb9df/numpy-2.4.4-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:f169b9a863d34f5d11b8698ead99febeaa17a13ca044961aa8e2662a6c7766a0", size = 16684353, upload-time = "2026-03-29T13:20:29.504Z" }, + { url = "https://files.pythonhosted.org/packages/4c/39/8a320264a84404c74cc7e79715de85d6130fa07a0898f67fb5cd5bd79908/numpy-2.4.4-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:2483e4584a1cb3092da4470b38866634bafb223cbcd551ee047633fd2584599a", size = 14704914, upload-time = "2026-03-29T13:20:33.547Z" }, + { url = "https://files.pythonhosted.org/packages/91/fb/287076b2614e1d1044235f50f03748f31fa287e3dbe6abeb35cdfa351eca/numpy-2.4.4-cp314-cp314-macosx_14_0_arm64.whl", hash = "sha256:2d19e6e2095506d1736b7d80595e0f252d76b89f5e715c35e06e937679ea7d7a", size = 5210005, upload-time = "2026-03-29T13:20:36.45Z" }, + { url = "https://files.pythonhosted.org/packages/63/eb/fcc338595309910de6ecabfcef2419a9ce24399680bfb149421fa2df1280/numpy-2.4.4-cp314-cp314-macosx_14_0_x86_64.whl", hash = "sha256:6a246d5914aa1c820c9443ddcee9c02bec3e203b0c080349533fae17727dfd1b", size = 6544974, upload-time = "2026-03-29T13:20:39.014Z" }, + { url = "https://files.pythonhosted.org/packages/44/5d/e7e9044032a716cdfaa3fba27a8e874bf1c5f1912a1ddd4ed071bf8a14a6/numpy-2.4.4-cp314-cp314-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:989824e9faf85f96ec9c7761cd8d29c531ad857bfa1daa930cba85baaecf1a9a", size = 15684591, upload-time = "2026-03-29T13:20:42.146Z" }, + { url = "https://files.pythonhosted.org/packages/98/7c/21252050676612625449b4807d6b695b9ce8a7c9e1c197ee6216c8a65c7c/numpy-2.4.4-cp314-cp314-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:27a8d92cd10f1382a67d7cf4db7ce18341b66438bdd9f691d7b0e48d104c2a9d", size = 16637700, upload-time = "2026-03-29T13:20:46.204Z" }, + { url = "https://files.pythonhosted.org/packages/b1/29/56d2bbef9465db24ef25393383d761a1af4f446a1df9b8cded4fe3a5a5d7/numpy-2.4.4-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:e44319a2953c738205bf3354537979eaa3998ed673395b964c1176083dd46252", size = 17035781, upload-time = "2026-03-29T13:20:50.242Z" }, + { url = "https://files.pythonhosted.org/packages/e3/2b/a35a6d7589d21f44cea7d0a98de5ddcbb3d421b2622a5c96b1edf18707c3/numpy-2.4.4-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:e892aff75639bbef0d2a2cfd55535510df26ff92f63c92cd84ef8d4ba5a5557f", size = 18362959, upload-time = "2026-03-29T13:20:54.019Z" }, + { url = "https://files.pythonhosted.org/packages/64/c9/d52ec581f2390e0f5f85cbfd80fb83d965fc15e9f0e1aec2195faa142cde/numpy-2.4.4-cp314-cp314-win32.whl", hash = "sha256:1378871da56ca8943c2ba674530924bb8ca40cd228358a3b5f302ad60cf875fc", size = 6008768, upload-time = "2026-03-29T13:20:56.912Z" }, + { url = "https://files.pythonhosted.org/packages/fa/22/4cc31a62a6c7b74a8730e31a4274c5dc80e005751e277a2ce38e675e4923/numpy-2.4.4-cp314-cp314-win_amd64.whl", hash = "sha256:715d1c092715954784bc79e1174fc2a90093dc4dc84ea15eb14dad8abdcdeb74", size = 12449181, upload-time = "2026-03-29T13:20:59.548Z" }, + { url = "https://files.pythonhosted.org/packages/70/2e/14cda6f4d8e396c612d1bf97f22958e92148801d7e4f110cabebdc0eef4b/numpy-2.4.4-cp314-cp314-win_arm64.whl", hash = "sha256:2c194dd721e54ecad9ad387c1d35e63dce5c4450c6dc7dd5611283dda239aabb", size = 10496035, upload-time = "2026-03-29T13:21:02.524Z" }, + { url = "https://files.pythonhosted.org/packages/b1/e8/8fed8c8d848d7ecea092dc3469643f9d10bc3a134a815a3b033da1d2039b/numpy-2.4.4-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:2aa0613a5177c264ff5921051a5719d20095ea586ca88cc802c5c218d1c67d3e", size = 14824958, upload-time = "2026-03-29T13:21:05.671Z" }, + { url = "https://files.pythonhosted.org/packages/05/1a/d8007a5138c179c2bf33ef44503e83d70434d2642877ee8fbb230e7c0548/numpy-2.4.4-cp314-cp314t-macosx_14_0_arm64.whl", hash = "sha256:42c16925aa5a02362f986765f9ebabf20de75cdefdca827d14315c568dcab113", size = 5330020, upload-time = "2026-03-29T13:21:08.635Z" }, + { url = "https://files.pythonhosted.org/packages/99/64/ffb99ac6ae93faf117bcbd5c7ba48a7f45364a33e8e458545d3633615dda/numpy-2.4.4-cp314-cp314t-macosx_14_0_x86_64.whl", hash = "sha256:874f200b2a981c647340f841730fc3a2b54c9d940566a3c4149099591e2c4c3d", size = 6650758, upload-time = "2026-03-29T13:21:10.949Z" }, + { url = "https://files.pythonhosted.org/packages/6e/6e/795cc078b78a384052e73b2f6281ff7a700e9bf53bcce2ee579d4f6dd879/numpy-2.4.4-cp314-cp314t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c9b39d38a9bd2ae1becd7eac1303d031c5c110ad31f2b319c6e7d98b135c934d", size = 15729948, upload-time = "2026-03-29T13:21:14.047Z" }, + { url = "https://files.pythonhosted.org/packages/5f/86/2acbda8cc2af5f3d7bfc791192863b9e3e19674da7b5e533fded124d1299/numpy-2.4.4-cp314-cp314t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b268594bccac7d7cf5844c7732e3f20c50921d94e36d7ec9b79e9857694b1b2f", size = 16679325, upload-time = "2026-03-29T13:21:17.561Z" }, + { url = "https://files.pythonhosted.org/packages/bc/59/cafd83018f4aa55e0ac6fa92aa066c0a1877b77a615ceff1711c260ffae8/numpy-2.4.4-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:ac6b31e35612a26483e20750126d30d0941f949426974cace8e6b5c58a3657b0", size = 17084883, upload-time = "2026-03-29T13:21:21.106Z" }, + { url = "https://files.pythonhosted.org/packages/f0/85/a42548db84e65ece46ab2caea3d3f78b416a47af387fcbb47ec28e660dc2/numpy-2.4.4-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:8e3ed142f2728df44263aaf5fb1f5b0b99f4070c553a0d7f033be65338329150", size = 18403474, upload-time = "2026-03-29T13:21:24.828Z" }, + { url = "https://files.pythonhosted.org/packages/ed/ad/483d9e262f4b831000062e5d8a45e342166ec8aaa1195264982bca267e62/numpy-2.4.4-cp314-cp314t-win32.whl", hash = "sha256:dddbbd259598d7240b18c9d87c56a9d2fb3b02fe266f49a7c101532e78c1d871", size = 6155500, upload-time = "2026-03-29T13:21:28.205Z" }, + { url = "https://files.pythonhosted.org/packages/c7/03/2fc4e14c7bd4ff2964b74ba90ecb8552540b6315f201df70f137faa5c589/numpy-2.4.4-cp314-cp314t-win_amd64.whl", hash = "sha256:a7164afb23be6e37ad90b2f10426149fd75aee07ca55653d2aa41e66c4ef697e", size = 12637755, upload-time = "2026-03-29T13:21:31.107Z" }, + { url = "https://files.pythonhosted.org/packages/58/78/548fb8e07b1a341746bfbecb32f2c268470f45fa028aacdbd10d9bc73aab/numpy-2.4.4-cp314-cp314t-win_arm64.whl", hash = "sha256:ba203255017337d39f89bdd58417f03c4426f12beed0440cfd933cb15f8669c7", size = 10566643, upload-time = "2026-03-29T13:21:34.339Z" }, +] + +[[package]] +name = "packaging" +version = "26.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d7/f1/e7a6dd94a8d4a5626c03e4e99c87f241ba9e350cd9e6d75123f992427270/packaging-26.2.tar.gz", hash = "sha256:ff452ff5a3e828ce110190feff1178bb1f2ea2281fa2075aadb987c2fb221661", size = 228134, upload-time = "2026-04-24T20:15:23.917Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/df/b2/87e62e8c3e2f4b32e5fe99e0b86d576da1312593b39f47d8ceef365e95ed/packaging-26.2-py3-none-any.whl", hash = "sha256:5fc45236b9446107ff2415ce77c807cee2862cb6fac22b8a73826d0693b0980e", size = 100195, upload-time = "2026-04-24T20:15:22.081Z" }, +] + +[[package]] +name = "pandas" +version = "3.0.2" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "numpy" }, + { name = "python-dateutil" }, + { name = "tzdata", marker = "sys_platform == 'emscripten' or sys_platform == 'win32'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/da/99/b342345300f13440fe9fe385c3c481e2d9a595ee3bab4d3219247ac94e9a/pandas-3.0.2.tar.gz", hash = "sha256:f4753e73e34c8d83221ba58f232433fca2748be8b18dbca02d242ed153945043", size = 4645855, upload-time = "2026-03-31T06:48:30.816Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/bf/ca/3e639a1ea6fcd0617ca4e8ca45f62a74de33a56ae6cd552735470b22c8d3/pandas-3.0.2-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:b5918ba197c951dec132b0c5929a00c0bf05d5942f590d3c10a807f6e15a57d3", size = 10321105, upload-time = "2026-03-31T06:46:57.327Z" }, + { url = "https://files.pythonhosted.org/packages/0b/77/dbc82ff2fb0e63c6564356682bf201edff0ba16c98630d21a1fb312a8182/pandas-3.0.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:d606a041c89c0a474a4702d532ab7e73a14fe35c8d427b972a625c8e46373668", size = 9864088, upload-time = "2026-03-31T06:46:59.935Z" }, + { url = "https://files.pythonhosted.org/packages/5c/2b/341f1b04bbca2e17e13cd3f08c215b70ef2c60c5356ef1e8c6857449edc7/pandas-3.0.2-cp313-cp313-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:710246ba0616e86891b58ab95f2495143bb2bc83ab6b06747c74216f583a6ac9", size = 10369066, upload-time = "2026-03-31T06:47:02.792Z" }, + { url = "https://files.pythonhosted.org/packages/12/c5/cbb1ffefb20a93d3f0e1fdcda699fb84976210d411b008f97f48bf6ce27e/pandas-3.0.2-cp313-cp313-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:5d3cfe227c725b1f3dff4278b43d8c784656a42a9325b63af6b1492a8232209e", size = 10876780, upload-time = "2026-03-31T06:47:06.205Z" }, + { url = "https://files.pythonhosted.org/packages/98/fe/2249ae5e0a69bd0ddf17353d0a5d26611d70970111f5b3600cdc8be883e7/pandas-3.0.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:c3b723df9087a9a9a840e263ebd9f88b64a12075d1bf2ea401a5a42f254f084d", size = 11375181, upload-time = "2026-03-31T06:47:09.383Z" }, + { url = "https://files.pythonhosted.org/packages/de/64/77a38b09e70b6464883b8d7584ab543e748e42c1b5d337a2ee088e0df741/pandas-3.0.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:a3096110bf9eac0070b7208465f2740e2d8a670d5cb6530b5bb884eca495fd39", size = 11928899, upload-time = "2026-03-31T06:47:12.686Z" }, + { url = "https://files.pythonhosted.org/packages/5e/52/42855bf626868413f761addd574acc6195880ae247a5346477a4361c3acb/pandas-3.0.2-cp313-cp313-win_amd64.whl", hash = "sha256:07a10f5c36512eead51bc578eb3354ad17578b22c013d89a796ab5eee90cd991", size = 9746574, upload-time = "2026-03-31T06:47:15.64Z" }, + { url = "https://files.pythonhosted.org/packages/88/39/21304ae06a25e8bf9fc820d69b29b2c495b2ae580d1e143146c309941760/pandas-3.0.2-cp313-cp313-win_arm64.whl", hash = "sha256:5fdbfa05931071aba28b408e59226186b01eb5e92bea2ab78b65863ca3228d84", size = 9047156, upload-time = "2026-03-31T06:47:18.595Z" }, + { url = "https://files.pythonhosted.org/packages/72/20/7defa8b27d4f330a903bb68eea33be07d839c5ea6bdda54174efcec0e1d2/pandas-3.0.2-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:dbc20dea3b9e27d0e66d74c42b2d0c1bed9c2ffe92adea33633e3bedeb5ac235", size = 10756238, upload-time = "2026-03-31T06:47:22.012Z" }, + { url = "https://files.pythonhosted.org/packages/e9/95/49433c14862c636afc0e9b2db83ff16b3ad92959364e52b2955e44c8e94c/pandas-3.0.2-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:b75c347eff42497452116ce05ef461822d97ce5b9ff8df6edacb8076092c855d", size = 10408520, upload-time = "2026-03-31T06:47:25.197Z" }, + { url = "https://files.pythonhosted.org/packages/3b/f8/462ad2b5881d6b8ec8e5f7ed2ea1893faa02290d13870a1600fe72ad8efc/pandas-3.0.2-cp313-cp313t-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:d1478075142e83a5571782ad007fb201ed074bdeac7ebcc8890c71442e96adf7", size = 10324154, upload-time = "2026-03-31T06:47:28.097Z" }, + { url = "https://files.pythonhosted.org/packages/0a/65/d1e69b649cbcddda23ad6e4c40ef935340f6f652a006e5cbc3555ac8adb3/pandas-3.0.2-cp313-cp313t-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:5880314e69e763d4c8b27937090de570f1fb8d027059a7ada3f7f8e98bdcb677", size = 10714449, upload-time = "2026-03-31T06:47:30.85Z" }, + { url = "https://files.pythonhosted.org/packages/47/a4/85b59bc65b8190ea3689882db6cdf32a5003c0ccd5a586c30fdcc3ffc4fc/pandas-3.0.2-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:b5329e26898896f06035241a626d7c335daa479b9bbc82be7c2742d048e41172", size = 11338475, upload-time = "2026-03-31T06:47:34.026Z" }, + { url = "https://files.pythonhosted.org/packages/1e/c4/bc6966c6e38e5d9478b935272d124d80a589511ed1612a5d21d36f664c68/pandas-3.0.2-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:81526c4afd31971f8b62671442a4b2b51e0aa9acc3819c9f0f12a28b6fcf85f1", size = 11786568, upload-time = "2026-03-31T06:47:36.941Z" }, + { url = "https://files.pythonhosted.org/packages/e8/74/09298ca9740beed1d3504e073d67e128aa07e5ca5ca2824b0c674c0b8676/pandas-3.0.2-cp313-cp313t-win_amd64.whl", hash = "sha256:7cadd7e9a44ec13b621aec60f9150e744cfc7a3dd32924a7e2f45edff31823b0", size = 10488652, upload-time = "2026-03-31T06:47:40.612Z" }, + { url = "https://files.pythonhosted.org/packages/bb/40/c6ea527147c73b24fc15c891c3fcffe9c019793119c5742b8784a062c7db/pandas-3.0.2-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:db0dbfd2a6cdf3770aa60464d50333d8f3d9165b2f2671bcc299b72de5a6677b", size = 10326084, upload-time = "2026-03-31T06:47:43.834Z" }, + { url = "https://files.pythonhosted.org/packages/95/25/bdb9326c3b5455f8d4d3549fce7abcf967259de146fe2cf7a82368141948/pandas-3.0.2-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:0555c5882688a39317179ab4a0ed41d3ebc8812ab14c69364bbee8fb7a3f6288", size = 9914146, upload-time = "2026-03-31T06:47:46.67Z" }, + { url = "https://files.pythonhosted.org/packages/8d/77/3a227ff3337aa376c60d288e1d61c5d097131d0ac71f954d90a8f369e422/pandas-3.0.2-cp314-cp314-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:01f31a546acd5574ef77fe199bc90b55527c225c20ccda6601cf6b0fd5ed597c", size = 10444081, upload-time = "2026-03-31T06:47:49.681Z" }, + { url = "https://files.pythonhosted.org/packages/15/88/3cdd54fa279341afa10acf8d2b503556b1375245dccc9315659f795dd2e9/pandas-3.0.2-cp314-cp314-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:deeca1b5a931fdf0c2212c8a659ade6d3b1edc21f0914ce71ef24456ca7a6535", size = 10897535, upload-time = "2026-03-31T06:47:53.033Z" }, + { url = "https://files.pythonhosted.org/packages/06/9d/98cc7a7624f7932e40f434299260e2917b090a579d75937cb8a57b9d2de3/pandas-3.0.2-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:0f48afd9bb13300ffb5a3316973324c787054ba6665cda0da3fbd67f451995db", size = 11446992, upload-time = "2026-03-31T06:47:56.193Z" }, + { url = "https://files.pythonhosted.org/packages/9a/cd/19ff605cc3760e80602e6826ddef2824d8e7050ed80f2e11c4b079741dc3/pandas-3.0.2-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:6c4d8458b97a35717b62469a4ea0e85abd5ed8687277f5ccfc67f8a5126f8c53", size = 11968257, upload-time = "2026-03-31T06:47:59.137Z" }, + { url = "https://files.pythonhosted.org/packages/db/60/aba6a38de456e7341285102bede27514795c1eaa353bc0e7638b6b785356/pandas-3.0.2-cp314-cp314-win_amd64.whl", hash = "sha256:b35d14bb5d8285d9494fe93815a9e9307c0876e10f1e8e89ac5b88f728ec8dcf", size = 9865893, upload-time = "2026-03-31T06:48:02.038Z" }, + { url = "https://files.pythonhosted.org/packages/08/71/e5ec979dd2e8a093dacb8864598c0ff59a0cee0bbcdc0bfec16a51684d4f/pandas-3.0.2-cp314-cp314-win_arm64.whl", hash = "sha256:63d141b56ef686f7f0d714cfb8de4e320475b86bf4b620aa0b7da89af8cbdbbb", size = 9188644, upload-time = "2026-03-31T06:48:05.045Z" }, + { url = "https://files.pythonhosted.org/packages/f1/6c/7b45d85db19cae1eb524f2418ceaa9d85965dcf7b764ed151386b7c540f0/pandas-3.0.2-cp314-cp314t-macosx_10_15_x86_64.whl", hash = "sha256:140f0cffb1fa2524e874dde5b477d9defe10780d8e9e220d259b2c0874c89d9d", size = 10776246, upload-time = "2026-03-31T06:48:07.789Z" }, + { url = "https://files.pythonhosted.org/packages/a8/3e/7b00648b086c106e81766f25322b48aa8dfa95b55e621dbdf2fdd413a117/pandas-3.0.2-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:ae37e833ff4fed0ba352f6bdd8b73ba3ab3256a85e54edfd1ab51ae40cca0af8", size = 10424801, upload-time = "2026-03-31T06:48:10.897Z" }, + { url = "https://files.pythonhosted.org/packages/da/6e/558dd09a71b53b4008e7fc8a98ec6d447e9bfb63cdaeea10e5eb9b2dabe8/pandas-3.0.2-cp314-cp314t-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4d888a5c678a419a5bb41a2a93818e8ed9fd3172246555c0b37b7cc27027effd", size = 10345643, upload-time = "2026-03-31T06:48:13.7Z" }, + { url = "https://files.pythonhosted.org/packages/be/e3/921c93b4d9a280409451dc8d07b062b503bbec0531d2627e73a756e99a82/pandas-3.0.2-cp314-cp314t-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b444dc64c079e84df91baa8bf613d58405645461cabca929d9178f2cd392398d", size = 10743641, upload-time = "2026-03-31T06:48:16.659Z" }, + { url = "https://files.pythonhosted.org/packages/56/ca/fd17286f24fa3b4d067965d8d5d7e14fe557dd4f979a0b068ac0deaf8228/pandas-3.0.2-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:4544c7a54920de8eeacaa1466a6b7268ecfbc9bc64ab4dbb89c6bbe94d5e0660", size = 11361993, upload-time = "2026-03-31T06:48:19.475Z" }, + { url = "https://files.pythonhosted.org/packages/e4/a5/2f6ed612056819de445a433ca1f2821ac3dab7f150d569a59e9cc105de1d/pandas-3.0.2-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:734be7551687c00fbd760dc0522ed974f82ad230d4a10f54bf51b80d44a08702", size = 11815274, upload-time = "2026-03-31T06:48:22.695Z" }, + { url = "https://files.pythonhosted.org/packages/00/2f/b622683e99ec3ce00b0854bac9e80868592c5b051733f2cf3a868e5fea26/pandas-3.0.2-cp314-cp314t-win_amd64.whl", hash = "sha256:57a07209bebcbcf768d2d13c9b78b852f9a15978dac41b9e6421a81ad4cdd276", size = 10888530, upload-time = "2026-03-31T06:48:25.806Z" }, + { url = "https://files.pythonhosted.org/packages/cb/2b/f8434233fab2bd66a02ec014febe4e5adced20e2693e0e90a07d118ed30e/pandas-3.0.2-cp314-cp314t-win_arm64.whl", hash = "sha256:5371b72c2d4d415d08765f32d689217a43227484e81b2305b52076e328f6f482", size = 9455341, upload-time = "2026-03-31T06:48:28.418Z" }, +] + +[[package]] +name = "pillow" +version = "12.2.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/8c/21/c2bcdd5906101a30244eaffc1b6e6ce71a31bd0742a01eb89e660ebfac2d/pillow-12.2.0.tar.gz", hash = "sha256:a830b1a40919539d07806aa58e1b114df53ddd43213d9c8b75847eee6c0182b5", size = 46987819, upload-time = "2026-04-01T14:46:17.687Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/4a/01/53d10cf0dbad820a8db274d259a37ba50b88b24768ddccec07355382d5ad/pillow-12.2.0-cp313-cp313-ios_13_0_arm64_iphoneos.whl", hash = "sha256:8297651f5b5679c19968abefd6bb84d95fe30ef712eb1b2d9b2d31ca61267f4c", size = 4100837, upload-time = "2026-04-01T14:43:41.506Z" }, + { url = "https://files.pythonhosted.org/packages/0f/98/f3a6657ecb698c937f6c76ee564882945f29b79bad496abcba0e84659ec5/pillow-12.2.0-cp313-cp313-ios_13_0_arm64_iphonesimulator.whl", hash = "sha256:50d8520da2a6ce0af445fa6d648c4273c3eeefbc32d7ce049f22e8b5c3daecc2", size = 4176528, upload-time = "2026-04-01T14:43:43.773Z" }, + { url = "https://files.pythonhosted.org/packages/69/bc/8986948f05e3ea490b8442ea1c1d4d990b24a7e43d8a51b2c7d8b1dced36/pillow-12.2.0-cp313-cp313-ios_13_0_x86_64_iphonesimulator.whl", hash = "sha256:766cef22385fa1091258ad7e6216792b156dc16d8d3fa607e7545b2b72061f1c", size = 3640401, upload-time = "2026-04-01T14:43:45.87Z" }, + { url = "https://files.pythonhosted.org/packages/34/46/6c717baadcd62bc8ed51d238d521ab651eaa74838291bda1f86fe1f864c9/pillow-12.2.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:5d2fd0fa6b5d9d1de415060363433f28da8b1526c1c129020435e186794b3795", size = 5308094, upload-time = "2026-04-01T14:43:48.438Z" }, + { url = "https://files.pythonhosted.org/packages/71/43/905a14a8b17fdb1ccb58d282454490662d2cb89a6bfec26af6d3520da5ec/pillow-12.2.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:56b25336f502b6ed02e889f4ece894a72612fe885889a6e8c4c80239ff6e5f5f", size = 4695402, upload-time = "2026-04-01T14:43:51.292Z" }, + { url = "https://files.pythonhosted.org/packages/73/dd/42107efcb777b16fa0393317eac58f5b5cf30e8392e266e76e51cff28c3d/pillow-12.2.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:f1c943e96e85df3d3478f7b691f229887e143f81fedab9b20205349ab04d73ed", size = 6280005, upload-time = "2026-04-01T14:43:54.242Z" }, + { url = "https://files.pythonhosted.org/packages/a8/68/b93e09e5e8549019e61acf49f65b1a8530765a7f812c77a7461bca7e4494/pillow-12.2.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:03f6fab9219220f041c74aeaa2939ff0062bd5c364ba9ce037197f4c6d498cd9", size = 8090669, upload-time = "2026-04-01T14:43:57.335Z" }, + { url = "https://files.pythonhosted.org/packages/4b/6e/3ccb54ce8ec4ddd1accd2d89004308b7b0b21c4ac3d20fa70af4760a4330/pillow-12.2.0-cp313-cp313-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5cdfebd752ec52bf5bb4e35d9c64b40826bc5b40a13df7c3cda20a2c03a0f5ed", size = 6395194, upload-time = "2026-04-01T14:43:59.864Z" }, + { url = "https://files.pythonhosted.org/packages/67/ee/21d4e8536afd1a328f01b359b4d3997b291ffd35a237c877b331c1c3b71c/pillow-12.2.0-cp313-cp313-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:eedf4b74eda2b5a4b2b2fb4c006d6295df3bf29e459e198c90ea48e130dc75c3", size = 7082423, upload-time = "2026-04-01T14:44:02.74Z" }, + { url = "https://files.pythonhosted.org/packages/78/5f/e9f86ab0146464e8c133fe85df987ed9e77e08b29d8d35f9f9f4d6f917ba/pillow-12.2.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:00a2865911330191c0b818c59103b58a5e697cae67042366970a6b6f1b20b7f9", size = 6505667, upload-time = "2026-04-01T14:44:05.381Z" }, + { url = "https://files.pythonhosted.org/packages/ed/1e/409007f56a2fdce61584fd3acbc2bbc259857d555196cedcadc68c015c82/pillow-12.2.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:1e1757442ed87f4912397c6d35a0db6a7b52592156014706f17658ff58bbf795", size = 7208580, upload-time = "2026-04-01T14:44:08.39Z" }, + { url = "https://files.pythonhosted.org/packages/23/c4/7349421080b12fb35414607b8871e9534546c128a11965fd4a7002ccfbee/pillow-12.2.0-cp313-cp313-win32.whl", hash = "sha256:144748b3af2d1b358d41286056d0003f47cb339b8c43a9ea42f5fea4d8c66b6e", size = 6375896, upload-time = "2026-04-01T14:44:11.197Z" }, + { url = "https://files.pythonhosted.org/packages/3f/82/8a3739a5e470b3c6cbb1d21d315800d8e16bff503d1f16b03a4ec3212786/pillow-12.2.0-cp313-cp313-win_amd64.whl", hash = "sha256:390ede346628ccc626e5730107cde16c42d3836b89662a115a921f28440e6a3b", size = 7081266, upload-time = "2026-04-01T14:44:13.947Z" }, + { url = "https://files.pythonhosted.org/packages/c3/25/f968f618a062574294592f668218f8af564830ccebdd1fa6200f598e65c5/pillow-12.2.0-cp313-cp313-win_arm64.whl", hash = "sha256:8023abc91fba39036dbce14a7d6535632f99c0b857807cbbbf21ecc9f4717f06", size = 2463508, upload-time = "2026-04-01T14:44:16.312Z" }, + { url = "https://files.pythonhosted.org/packages/4d/a4/b342930964e3cb4dce5038ae34b0eab4653334995336cd486c5a8c25a00c/pillow-12.2.0-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:042db20a421b9bafecc4b84a8b6e444686bd9d836c7fd24542db3e7df7baad9b", size = 5309927, upload-time = "2026-04-01T14:44:18.89Z" }, + { url = "https://files.pythonhosted.org/packages/9f/de/23198e0a65a9cf06123f5435a5d95cea62a635697f8f03d134d3f3a96151/pillow-12.2.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:dd025009355c926a84a612fecf58bb315a3f6814b17ead51a8e48d3823d9087f", size = 4698624, upload-time = "2026-04-01T14:44:21.115Z" }, + { url = "https://files.pythonhosted.org/packages/01/a6/1265e977f17d93ea37aa28aa81bad4fa597933879fac2520d24e021c8da3/pillow-12.2.0-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:88ddbc66737e277852913bd1e07c150cc7bb124539f94c4e2df5344494e0a612", size = 6321252, upload-time = "2026-04-01T14:44:23.663Z" }, + { url = "https://files.pythonhosted.org/packages/3c/83/5982eb4a285967baa70340320be9f88e57665a387e3a53a7f0db8231a0cd/pillow-12.2.0-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:d362d1878f00c142b7e1a16e6e5e780f02be8195123f164edf7eddd911eefe7c", size = 8126550, upload-time = "2026-04-01T14:44:26.772Z" }, + { url = "https://files.pythonhosted.org/packages/4e/48/6ffc514adce69f6050d0753b1a18fd920fce8cac87620d5a31231b04bfc5/pillow-12.2.0-cp313-cp313t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:2c727a6d53cb0018aadd8018c2b938376af27914a68a492f59dfcaca650d5eea", size = 6433114, upload-time = "2026-04-01T14:44:29.615Z" }, + { url = "https://files.pythonhosted.org/packages/36/a3/f9a77144231fb8d40ee27107b4463e205fa4677e2ca2548e14da5cf18dce/pillow-12.2.0-cp313-cp313t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:efd8c21c98c5cc60653bcb311bef2ce0401642b7ce9d09e03a7da87c878289d4", size = 7115667, upload-time = "2026-04-01T14:44:32.773Z" }, + { url = "https://files.pythonhosted.org/packages/c1/fc/ac4ee3041e7d5a565e1c4fd72a113f03b6394cc72ab7089d27608f8aaccb/pillow-12.2.0-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:9f08483a632889536b8139663db60f6724bfcb443c96f1b18855860d7d5c0fd4", size = 6538966, upload-time = "2026-04-01T14:44:35.252Z" }, + { url = "https://files.pythonhosted.org/packages/c0/a8/27fb307055087f3668f6d0a8ccb636e7431d56ed0750e07a60547b1e083e/pillow-12.2.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:dac8d77255a37e81a2efcbd1fc05f1c15ee82200e6c240d7e127e25e365c39ea", size = 7238241, upload-time = "2026-04-01T14:44:37.875Z" }, + { url = "https://files.pythonhosted.org/packages/ad/4b/926ab182c07fccae9fcb120043464e1ff1564775ec8864f21a0ebce6ac25/pillow-12.2.0-cp313-cp313t-win32.whl", hash = "sha256:ee3120ae9dff32f121610bb08e4313be87e03efeadfc6c0d18f89127e24d0c24", size = 6379592, upload-time = "2026-04-01T14:44:40.336Z" }, + { url = "https://files.pythonhosted.org/packages/c2/c4/f9e476451a098181b30050cc4c9a3556b64c02cf6497ea421ac047e89e4b/pillow-12.2.0-cp313-cp313t-win_amd64.whl", hash = "sha256:325ca0528c6788d2a6c3d40e3568639398137346c3d6e66bb61db96b96511c98", size = 7085542, upload-time = "2026-04-01T14:44:43.251Z" }, + { url = "https://files.pythonhosted.org/packages/00/a4/285f12aeacbe2d6dc36c407dfbbe9e96d4a80b0fb710a337f6d2ad978c75/pillow-12.2.0-cp313-cp313t-win_arm64.whl", hash = "sha256:2e5a76d03a6c6dcef67edabda7a52494afa4035021a79c8558e14af25313d453", size = 2465765, upload-time = "2026-04-01T14:44:45.996Z" }, + { url = "https://files.pythonhosted.org/packages/bf/98/4595daa2365416a86cb0d495248a393dfc84e96d62ad080c8546256cb9c0/pillow-12.2.0-cp314-cp314-ios_13_0_arm64_iphoneos.whl", hash = "sha256:3adc9215e8be0448ed6e814966ecf3d9952f0ea40eb14e89a102b87f450660d8", size = 4100848, upload-time = "2026-04-01T14:44:48.48Z" }, + { url = "https://files.pythonhosted.org/packages/0b/79/40184d464cf89f6663e18dfcf7ca21aae2491fff1a16127681bf1fa9b8cf/pillow-12.2.0-cp314-cp314-ios_13_0_arm64_iphonesimulator.whl", hash = "sha256:6a9adfc6d24b10f89588096364cc726174118c62130c817c2837c60cf08a392b", size = 4176515, upload-time = "2026-04-01T14:44:51.353Z" }, + { url = "https://files.pythonhosted.org/packages/b0/63/703f86fd4c422a9cf722833670f4f71418fb116b2853ff7da722ea43f184/pillow-12.2.0-cp314-cp314-ios_13_0_x86_64_iphonesimulator.whl", hash = "sha256:6a6e67ea2e6feda684ed370f9a1c52e7a243631c025ba42149a2cc5934dec295", size = 3640159, upload-time = "2026-04-01T14:44:53.588Z" }, + { url = "https://files.pythonhosted.org/packages/71/e0/fb22f797187d0be2270f83500aab851536101b254bfa1eae10795709d283/pillow-12.2.0-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:2bb4a8d594eacdfc59d9e5ad972aa8afdd48d584ffd5f13a937a664c3e7db0ed", size = 5312185, upload-time = "2026-04-01T14:44:56.039Z" }, + { url = "https://files.pythonhosted.org/packages/ba/8c/1a9e46228571de18f8e28f16fabdfc20212a5d019f3e3303452b3f0a580d/pillow-12.2.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:80b2da48193b2f33ed0c32c38140f9d3186583ce7d516526d462645fd98660ae", size = 4695386, upload-time = "2026-04-01T14:44:58.663Z" }, + { url = "https://files.pythonhosted.org/packages/70/62/98f6b7f0c88b9addd0e87c217ded307b36be024d4ff8869a812b241d1345/pillow-12.2.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:22db17c68434de69d8ecfc2fe821569195c0c373b25cccb9cbdacf2c6e53c601", size = 6280384, upload-time = "2026-04-01T14:45:01.5Z" }, + { url = "https://files.pythonhosted.org/packages/5e/03/688747d2e91cfbe0e64f316cd2e8005698f76ada3130d0194664174fa5de/pillow-12.2.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:7b14cc0106cd9aecda615dd6903840a058b4700fcb817687d0ee4fc8b6e389be", size = 8091599, upload-time = "2026-04-01T14:45:04.5Z" }, + { url = "https://files.pythonhosted.org/packages/f6/35/577e22b936fcdd66537329b33af0b4ccfefaeabd8aec04b266528cddb33c/pillow-12.2.0-cp314-cp314-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8cbeb542b2ebc6fcdacabf8aca8c1a97c9b3ad3927d46b8723f9d4f033288a0f", size = 6396021, upload-time = "2026-04-01T14:45:07.117Z" }, + { url = "https://files.pythonhosted.org/packages/11/8d/d2532ad2a603ca2b93ad9f5135732124e57811d0168155852f37fbce2458/pillow-12.2.0-cp314-cp314-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4bfd07bc812fbd20395212969e41931001fd59eb55a60658b0e5710872e95286", size = 7083360, upload-time = "2026-04-01T14:45:09.763Z" }, + { url = "https://files.pythonhosted.org/packages/5e/26/d325f9f56c7e039034897e7380e9cc202b1e368bfd04d4cbe6a441f02885/pillow-12.2.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:9aba9a17b623ef750a4d11b742cbafffeb48a869821252b30ee21b5e91392c50", size = 6507628, upload-time = "2026-04-01T14:45:12.378Z" }, + { url = "https://files.pythonhosted.org/packages/5f/f7/769d5632ffb0988f1c5e7660b3e731e30f7f8ec4318e94d0a5d674eb65a4/pillow-12.2.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:deede7c263feb25dba4e82ea23058a235dcc2fe1f6021025dc71f2b618e26104", size = 7209321, upload-time = "2026-04-01T14:45:15.122Z" }, + { url = "https://files.pythonhosted.org/packages/6a/7a/c253e3c645cd47f1aceea6a8bacdba9991bf45bb7dfe927f7c893e89c93c/pillow-12.2.0-cp314-cp314-win32.whl", hash = "sha256:632ff19b2778e43162304d50da0181ce24ac5bb8180122cbe1bf4673428328c7", size = 6479723, upload-time = "2026-04-01T14:45:17.797Z" }, + { url = "https://files.pythonhosted.org/packages/cd/8b/601e6566b957ca50e28725cb6c355c59c2c8609751efbecd980db44e0349/pillow-12.2.0-cp314-cp314-win_amd64.whl", hash = "sha256:4e6c62e9d237e9b65fac06857d511e90d8461a32adcc1b9065ea0c0fa3a28150", size = 7217400, upload-time = "2026-04-01T14:45:20.529Z" }, + { url = "https://files.pythonhosted.org/packages/d6/94/220e46c73065c3e2951bb91c11a1fb636c8c9ad427ac3ce7d7f3359b9b2f/pillow-12.2.0-cp314-cp314-win_arm64.whl", hash = "sha256:b1c1fbd8a5a1af3412a0810d060a78b5136ec0836c8a4ef9aa11807f2a22f4e1", size = 2554835, upload-time = "2026-04-01T14:45:23.162Z" }, + { url = "https://files.pythonhosted.org/packages/b6/ab/1b426a3974cb0e7da5c29ccff4807871d48110933a57207b5a676cccc155/pillow-12.2.0-cp314-cp314t-macosx_10_15_x86_64.whl", hash = "sha256:57850958fe9c751670e49b2cecf6294acc99e562531f4bd317fa5ddee2068463", size = 5314225, upload-time = "2026-04-01T14:45:25.637Z" }, + { url = "https://files.pythonhosted.org/packages/19/1e/dce46f371be2438eecfee2a1960ee2a243bbe5e961890146d2dee1ff0f12/pillow-12.2.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:d5d38f1411c0ed9f97bcb49b7bd59b6b7c314e0e27420e34d99d844b9ce3b6f3", size = 4698541, upload-time = "2026-04-01T14:45:28.355Z" }, + { url = "https://files.pythonhosted.org/packages/55/c3/7fbecf70adb3a0c33b77a300dc52e424dc22ad8cdc06557a2e49523b703d/pillow-12.2.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:5c0a9f29ca8e79f09de89293f82fc9b0270bb4af1d58bc98f540cc4aedf03166", size = 6322251, upload-time = "2026-04-01T14:45:30.924Z" }, + { url = "https://files.pythonhosted.org/packages/1c/3c/7fbc17cfb7e4fe0ef1642e0abc17fc6c94c9f7a16be41498e12e2ba60408/pillow-12.2.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:1610dd6c61621ae1cf811bef44d77e149ce3f7b95afe66a4512f8c59f25d9ebe", size = 8127807, upload-time = "2026-04-01T14:45:33.908Z" }, + { url = "https://files.pythonhosted.org/packages/ff/c3/a8ae14d6defd2e448493ff512fae903b1e9bd40b72efb6ec55ce0048c8ce/pillow-12.2.0-cp314-cp314t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0a34329707af4f73cf1782a36cd2289c0368880654a2c11f027bcee9052d35dd", size = 6433935, upload-time = "2026-04-01T14:45:36.623Z" }, + { url = "https://files.pythonhosted.org/packages/6e/32/2880fb3a074847ac159d8f902cb43278a61e85f681661e7419e6596803ed/pillow-12.2.0-cp314-cp314t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8e9c4f5b3c546fa3458a29ab22646c1c6c787ea8f5ef51300e5a60300736905e", size = 7116720, upload-time = "2026-04-01T14:45:39.258Z" }, + { url = "https://files.pythonhosted.org/packages/46/87/495cc9c30e0129501643f24d320076f4cc54f718341df18cc70ec94c44e1/pillow-12.2.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:fb043ee2f06b41473269765c2feae53fc2e2fbf96e5e22ca94fb5ad677856f06", size = 6540498, upload-time = "2026-04-01T14:45:41.879Z" }, + { url = "https://files.pythonhosted.org/packages/18/53/773f5edca692009d883a72211b60fdaf8871cbef075eaa9d577f0a2f989e/pillow-12.2.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:f278f034eb75b4e8a13a54a876cc4a5ab39173d2cdd93a638e1b467fc545ac43", size = 7239413, upload-time = "2026-04-01T14:45:44.705Z" }, + { url = "https://files.pythonhosted.org/packages/c9/e4/4b64a97d71b2a83158134abbb2f5bd3f8a2ea691361282f010998f339ec7/pillow-12.2.0-cp314-cp314t-win32.whl", hash = "sha256:6bb77b2dcb06b20f9f4b4a8454caa581cd4dd0643a08bacf821216a16d9c8354", size = 6482084, upload-time = "2026-04-01T14:45:47.568Z" }, + { url = "https://files.pythonhosted.org/packages/ba/13/306d275efd3a3453f72114b7431c877d10b1154014c1ebbedd067770d629/pillow-12.2.0-cp314-cp314t-win_amd64.whl", hash = "sha256:6562ace0d3fb5f20ed7290f1f929cae41b25ae29528f2af1722966a0a02e2aa1", size = 7225152, upload-time = "2026-04-01T14:45:50.032Z" }, + { url = "https://files.pythonhosted.org/packages/ff/6e/cf826fae916b8658848d7b9f38d88da6396895c676e8086fc0988073aaf8/pillow-12.2.0-cp314-cp314t-win_arm64.whl", hash = "sha256:aa88ccfe4e32d362816319ed727a004423aab09c5cea43c01a4b435643fa34eb", size = 2556579, upload-time = "2026-04-01T14:45:52.529Z" }, +] + +[[package]] +name = "protobuf" +version = "7.34.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/6b/6b/a0e95cad1ad7cc3f2c6821fcab91671bd5b78bd42afb357bb4765f29bc41/protobuf-7.34.1.tar.gz", hash = "sha256:9ce42245e704cc5027be797c1db1eb93184d44d1cdd71811fb2d9b25ad541280", size = 454708, upload-time = "2026-03-20T17:34:47.036Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ec/11/3325d41e6ee15bf1125654301211247b042563bcc898784351252549a8ad/protobuf-7.34.1-cp310-abi3-macosx_10_9_universal2.whl", hash = "sha256:d8b2cc79c4d8f62b293ad9b11ec3aebce9af481fa73e64556969f7345ebf9fc7", size = 429247, upload-time = "2026-03-20T17:34:37.024Z" }, + { url = "https://files.pythonhosted.org/packages/eb/9d/aa69df2724ff63efa6f72307b483ce0827f4347cc6d6df24b59e26659fef/protobuf-7.34.1-cp310-abi3-manylinux2014_aarch64.whl", hash = "sha256:5185e0e948d07abe94bb76ec9b8416b604cfe5da6f871d67aad30cbf24c3110b", size = 325753, upload-time = "2026-03-20T17:34:38.751Z" }, + { url = "https://files.pythonhosted.org/packages/92/e8/d174c91fd48e50101943f042b09af9029064810b734e4160bbe282fa1caa/protobuf-7.34.1-cp310-abi3-manylinux2014_s390x.whl", hash = "sha256:403b093a6e28a960372b44e5eb081775c9b056e816a8029c61231743d63f881a", size = 340198, upload-time = "2026-03-20T17:34:39.871Z" }, + { url = "https://files.pythonhosted.org/packages/53/1b/3b431694a4dc6d37b9f653f0c64b0a0d9ec074ee810710c0c3da21d67ba7/protobuf-7.34.1-cp310-abi3-manylinux2014_x86_64.whl", hash = "sha256:8ff40ce8cd688f7265326b38d5a1bed9bfdf5e6723d49961432f83e21d5713e4", size = 324267, upload-time = "2026-03-20T17:34:41.1Z" }, + { url = "https://files.pythonhosted.org/packages/85/29/64de04a0ac142fb685fd09999bc3d337943fb386f3a0ec57f92fd8203f97/protobuf-7.34.1-cp310-abi3-win32.whl", hash = "sha256:34b84ce27680df7cca9f231043ada0daa55d0c44a2ddfaa58ec1d0d89d8bf60a", size = 426628, upload-time = "2026-03-20T17:34:42.536Z" }, + { url = "https://files.pythonhosted.org/packages/4d/87/cb5e585192a22b8bd457df5a2c16a75ea0db9674c3a0a39fc9347d84e075/protobuf-7.34.1-cp310-abi3-win_amd64.whl", hash = "sha256:e97b55646e6ce5cbb0954a8c28cd39a5869b59090dfaa7df4598a7fba869468c", size = 437901, upload-time = "2026-03-20T17:34:44.112Z" }, + { url = "https://files.pythonhosted.org/packages/88/95/608f665226bca68b736b79e457fded9a2a38c4f4379a4a7614303d9db3bc/protobuf-7.34.1-py3-none-any.whl", hash = "sha256:bb3812cd53aefea2b028ef42bd780f5b96407247f20c6ef7c679807e9d188f11", size = 170715, upload-time = "2026-03-20T17:34:45.384Z" }, +] + +[[package]] +name = "pyarrow" +version = "24.0.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/91/13/13e1069b351bdc3881266e11147ffccf687505dbb0ea74036237f5d454a5/pyarrow-24.0.0.tar.gz", hash = "sha256:85fe721a14dd823aca09127acbb06c3ca723efbd436c004f16bca601b04dcc83", size = 1180261, upload-time = "2026-04-21T10:51:25.837Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/6f/d3/a1abf004482026ddc17f4503db227787fa3cfe41ec5091ff20e4fea55e57/pyarrow-24.0.0-cp313-cp313-macosx_12_0_arm64.whl", hash = "sha256:02b001b3ed4723caa44f6cd1af2d5c86aa2cf9971dacc2ffa55b21237713dfba", size = 34976759, upload-time = "2026-04-21T10:48:07.258Z" }, + { url = "https://files.pythonhosted.org/packages/4f/4a/34f0a36d28a2dd32225301b79daad44e243dc1a2bb77d43b60749be255c4/pyarrow-24.0.0-cp313-cp313-macosx_12_0_x86_64.whl", hash = "sha256:04920d6a71aabd08a0417709efce97d45ea8e6fb733d9ca9ecffb13c67839f68", size = 36658471, upload-time = "2026-04-21T10:48:13.347Z" }, + { url = "https://files.pythonhosted.org/packages/1f/78/543b94712ae8bb1a6023bcc1acf1a740fbff8286747c289cd9468fced2a5/pyarrow-24.0.0-cp313-cp313-manylinux_2_28_aarch64.whl", hash = "sha256:a964266397740257f16f7bb2e4f08a0c81454004beab8ff59dd531b73610e9f2", size = 45675981, upload-time = "2026-04-21T10:48:20.201Z" }, + { url = "https://files.pythonhosted.org/packages/84/9f/8fb7c222b100d314137fa40ec050de56cd8c6d957d1cfff685ce72f15b17/pyarrow-24.0.0-cp313-cp313-manylinux_2_28_x86_64.whl", hash = "sha256:6f066b179d68c413374294bc1735f68475457c933258df594443bb9d88ddc2a0", size = 48859172, upload-time = "2026-04-21T10:48:27.541Z" }, + { url = "https://files.pythonhosted.org/packages/a7/d3/1ea72538e6c8b3b475ed78d1049a2c518e655761ea50fe1171fc855fcab7/pyarrow-24.0.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:1183baeb14c5f587b1ec52831e665718ce632caab84b7cd6b85fd44f96114495", size = 49385733, upload-time = "2026-04-21T10:48:34.7Z" }, + { url = "https://files.pythonhosted.org/packages/c3/be/c3d8b06a1ba35f2260f8e1f771abbee7d5e345c0937aab90675706b1690a/pyarrow-24.0.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:806f24b4085453c197a5078218d1ee08783ebbba271badd153d1ae22a3ee804f", size = 51934335, upload-time = "2026-04-21T10:48:42.099Z" }, + { url = "https://files.pythonhosted.org/packages/9c/62/89e07a1e7329d2cde3e3c6994ba0839a24977a2beda8be6005ea3d860b99/pyarrow-24.0.0-cp313-cp313-win_amd64.whl", hash = "sha256:e4505fc6583f7b05ab854934896bcac8253b04ac1171a77dfb73efef92076d91", size = 27271748, upload-time = "2026-04-21T10:49:42.532Z" }, + { url = "https://files.pythonhosted.org/packages/17/1a/cff3a59f80b5b1658549d46611b67163f65e0664431c076ad728bf9d5af4/pyarrow-24.0.0-cp313-cp313t-macosx_12_0_arm64.whl", hash = "sha256:1a4e45017efbf115032e4475ee876d525e0e36c742214fbe405332480ecd6275", size = 35238554, upload-time = "2026-04-21T10:48:48.526Z" }, + { url = "https://files.pythonhosted.org/packages/a8/99/cce0f42a327bfef2c420fb6078a3eb834826e5d6697bf3009fe11d2ad051/pyarrow-24.0.0-cp313-cp313t-macosx_12_0_x86_64.whl", hash = "sha256:7986f1fa71cee060ad00758bcc79d3a93bab8559bf978fab9e53472a2e25a17b", size = 36782301, upload-time = "2026-04-21T10:48:55.181Z" }, + { url = "https://files.pythonhosted.org/packages/2a/66/8e560d5ff6793ca29aca213c53eec0dd482dd46cb93b2819e5aab52e4252/pyarrow-24.0.0-cp313-cp313t-manylinux_2_28_aarch64.whl", hash = "sha256:d3e0b61e8efb24ed38898e5cdc5fffa9124be480008d401a1f8071500494ae42", size = 45721929, upload-time = "2026-04-21T10:49:03.676Z" }, + { url = "https://files.pythonhosted.org/packages/27/0c/a26e25505d030716e078d9f16eb74973cbf0b33b672884e9f9da1c83b871/pyarrow-24.0.0-cp313-cp313t-manylinux_2_28_x86_64.whl", hash = "sha256:55a3bc1e3df3b5567b7d27ef551b2283f0c68a5e86f1cd56abc569da4f31335b", size = 48825365, upload-time = "2026-04-21T10:49:11.714Z" }, + { url = "https://files.pythonhosted.org/packages/5f/eb/771f9ecb0c65e73fe9dccdd1717901b9594f08c4515d000c7c62df573811/pyarrow-24.0.0-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:641f795b361874ac9da5294f8f443dfdbee355cf2bd9e3b8d97aaac2306b9b37", size = 49451819, upload-time = "2026-04-21T10:49:21.474Z" }, + { url = "https://files.pythonhosted.org/packages/48/da/61ae89a88732f5a785646f3ec6125dbb640fa98a540eb2b9889caa561403/pyarrow-24.0.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:8adc8e6ce5fccf5dc707046ae4914fd537def529709cc0d285d37a7f9cd442ca", size = 51909252, upload-time = "2026-04-21T10:49:31.164Z" }, + { url = "https://files.pythonhosted.org/packages/cb/1a/8dd5cafab7b66573fa91c03d06d213356ad4edd71813aa75e08ce2b3a844/pyarrow-24.0.0-cp313-cp313t-win_amd64.whl", hash = "sha256:9b18371ad2f44044b81a8d23bc2d8a9b6a6226dca775e8e16cfee640473d6c5d", size = 27388127, upload-time = "2026-04-21T10:49:37.334Z" }, + { url = "https://files.pythonhosted.org/packages/ad/80/d022a34ff05d2cbedd8ccf841fc1f532ecfa9eb5ed1711b56d0e0ea71fc9/pyarrow-24.0.0-cp314-cp314-macosx_12_0_arm64.whl", hash = "sha256:1cc9057f0319e26333b357e17f3c2c022f1a83739b48a88b25bfd5fa2dc18838", size = 35007997, upload-time = "2026-04-21T10:49:48.796Z" }, + { url = "https://files.pythonhosted.org/packages/1a/ff/f01485fda6f4e5d441afb8dd5e7681e4db18826c1e271852f5d3957d6a80/pyarrow-24.0.0-cp314-cp314-macosx_12_0_x86_64.whl", hash = "sha256:e6f1278ee4785b6db21229374a1c9e54ec7c549de5d1efc9630b6207de7e170b", size = 36678720, upload-time = "2026-04-21T10:49:55.858Z" }, + { url = "https://files.pythonhosted.org/packages/9e/c2/2d2d5fea814237923f71b36495211f20b43a1576f9a4d6da7e751a64ec6f/pyarrow-24.0.0-cp314-cp314-manylinux_2_28_aarch64.whl", hash = "sha256:adbbedc55506cbdabb830890444fb856bfb0060c46c6f8026c6c2f2cf86ae795", size = 45741852, upload-time = "2026-04-21T10:50:04.624Z" }, + { url = "https://files.pythonhosted.org/packages/8e/3a/28ba9c1c1ebdbb5f1b94dfebb46f207e52e6a554b7fe4132540fde29a3a0/pyarrow-24.0.0-cp314-cp314-manylinux_2_28_x86_64.whl", hash = "sha256:ae8a1145af31d903fa9bb166824d7abe9b4681a000b0159c9fb99c11bc11ad26", size = 48889852, upload-time = "2026-04-21T10:50:12.293Z" }, + { url = "https://files.pythonhosted.org/packages/df/51/4a389acfd31dca009f8fb82d7f510bb4130f2b3a8e18cf00194d0687d8ac/pyarrow-24.0.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:d7027eba1df3b2069e2e8d80f644fa0918b68c46432af3d088ddd390d063ecde", size = 49445207, upload-time = "2026-04-21T10:50:20.677Z" }, + { url = "https://files.pythonhosted.org/packages/19/4b/0bab2b23d2ae901b1b9a03c0efd4b2d070256f8ce3fc43f6e58c167b2081/pyarrow-24.0.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:e56a1ffe9bf7b727432b89104cc0849c21582949dd7bdcb34f17b2001a351a76", size = 51954117, upload-time = "2026-04-21T10:50:29.14Z" }, + { url = "https://files.pythonhosted.org/packages/29/88/f4e9145da0417b3d2c12035a8492b35ff4a3dbc653e614fcfb51d9dedb38/pyarrow-24.0.0-cp314-cp314-win_amd64.whl", hash = "sha256:38be1808cdd068605b787e6ca9119b27eb275a0234e50212c3492331680c3b1e", size = 28001155, upload-time = "2026-04-21T10:51:22.337Z" }, + { url = "https://files.pythonhosted.org/packages/79/4f/46a49a63f43526da895b1a45bbb51d5baf8e4d77159f8528fc3e5490007f/pyarrow-24.0.0-cp314-cp314t-macosx_12_0_arm64.whl", hash = "sha256:418e48ce50a45a6a6c73c454677203a9c75c966cb1e92ca3370959185f197a05", size = 35250387, upload-time = "2026-04-21T10:50:35.552Z" }, + { url = "https://files.pythonhosted.org/packages/a0/da/d5e0cd5ef00796922404806d5f00325cdadc3441ce2c13fe7115f2df9a64/pyarrow-24.0.0-cp314-cp314t-macosx_12_0_x86_64.whl", hash = "sha256:2f16197705a230a78270cdd4ea8a1d57e86b2fdcbc34a1f6aebc72e65c986f9a", size = 36797102, upload-time = "2026-04-21T10:50:42.417Z" }, + { url = "https://files.pythonhosted.org/packages/34/c7/5904145b0a593a05236c882933d439b5720f0a145381179063722fbfc123/pyarrow-24.0.0-cp314-cp314t-manylinux_2_28_aarch64.whl", hash = "sha256:fb24ac194bfc5e86839d7dcd52092ee31e5fe6733fe11f5e3b06ef0812b20072", size = 45745118, upload-time = "2026-04-21T10:50:49.324Z" }, + { url = "https://files.pythonhosted.org/packages/13/d3/cca42fe166d1c6e4d5b80e530b7949104d10e17508a90ae202dac205ce2a/pyarrow-24.0.0-cp314-cp314t-manylinux_2_28_x86_64.whl", hash = "sha256:9700ebd9a51f5895ce75ff4ac4b3c47a7d4b42bc618be8e713e5d56bacf5f931", size = 48844765, upload-time = "2026-04-21T10:50:55.579Z" }, + { url = "https://files.pythonhosted.org/packages/b0/49/942c3b79878ba928324d1e17c274ed84581db8c0a749b24bcf4cbdf15bd3/pyarrow-24.0.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:d8ddd2768da81d3ee08cfea9b597f4abb4e8e1dc8ae7e204b608d23a0d3ab699", size = 49471890, upload-time = "2026-04-21T10:51:02.439Z" }, + { url = "https://files.pythonhosted.org/packages/76/97/ff71431000a75d84135a1ace5ca4ba11726a231a8007bbb320a4c54075d5/pyarrow-24.0.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:61a3d7eaa97a14768b542f3d284dc6400dd2470d9f080708b13cd46b6ae18136", size = 51932250, upload-time = "2026-04-21T10:51:10.576Z" }, + { url = "https://files.pythonhosted.org/packages/51/be/6f79d55816d5c22557cf27533543d5d70dfe692adfbee4b99f2760674f38/pyarrow-24.0.0-cp314-cp314t-win_amd64.whl", hash = "sha256:c91d00057f23b8d353039520dc3a6c09d8608164c692e9f59a175a42b2ae0c19", size = 28131282, upload-time = "2026-04-21T10:51:16.815Z" }, +] + +[[package]] +name = "pydeck" +version = "0.9.2" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "jinja2" }, + { name = "numpy" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/eb/df/4e9e7f20f8034a37c6571c93809f6d22388c39978c98d174d656c1a18fd2/pydeck-0.9.2.tar.gz", hash = "sha256:c10d9035e81ead6385264cac8d19402471f6866a15ca1f7df1400f52142bcf87", size = 5849672, upload-time = "2026-04-16T18:30:30.089Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/88/24/b30ee7d723100fd822de1bb4c0adea62f3419884a75a536f35f355d1e7c0/pydeck-0.9.2-py2.py3-none-any.whl", hash = "sha256:8213dfeacc5f6bfe6825f61c8ee34e3850e8a31fc43924379ec98edb34a75b25", size = 11305615, upload-time = "2026-04-16T18:30:28.133Z" }, +] + +[[package]] +name = "python-dateutil" +version = "2.9.0.post0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "six" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/66/c0/0c8b6ad9f17a802ee498c46e004a0eb49bc148f2fd230864601a86dcf6db/python-dateutil-2.9.0.post0.tar.gz", hash = "sha256:37dd54208da7e1cd875388217d5e00ebd4179249f90fb72437e91a35459a0ad3", size = 342432, upload-time = "2024-03-01T18:36:20.211Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ec/57/56b9bcc3c9c6a792fcbaf139543cee77261f3651ca9da0c93f5c1221264b/python_dateutil-2.9.0.post0-py2.py3-none-any.whl", hash = "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427", size = 229892, upload-time = "2024-03-01T18:36:18.57Z" }, +] + +[[package]] +name = "referencing" +version = "0.37.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "attrs" }, + { name = "rpds-py" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/22/f5/df4e9027acead3ecc63e50fe1e36aca1523e1719559c499951bb4b53188f/referencing-0.37.0.tar.gz", hash = "sha256:44aefc3142c5b842538163acb373e24cce6632bd54bdb01b21ad5863489f50d8", size = 78036, upload-time = "2025-10-13T15:30:48.871Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/2c/58/ca301544e1fa93ed4f80d724bf5b194f6e4b945841c5bfd555878eea9fcb/referencing-0.37.0-py3-none-any.whl", hash = "sha256:381329a9f99628c9069361716891d34ad94af76e461dcb0335825aecc7692231", size = 26766, upload-time = "2025-10-13T15:30:47.625Z" }, +] + +[[package]] +name = "requests" +version = "2.33.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "certifi" }, + { name = "charset-normalizer" }, + { name = "idna" }, + { name = "urllib3" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/5f/a4/98b9c7c6428a668bf7e42ebb7c79d576a1c3c1e3ae2d47e674b468388871/requests-2.33.1.tar.gz", hash = "sha256:18817f8c57c6263968bc123d237e3b8b08ac046f5456bd1e307ee8f4250d3517", size = 134120, upload-time = "2026-03-30T16:09:15.531Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d7/8e/7540e8a2036f79a125c1d2ebadf69ed7901608859186c856fa0388ef4197/requests-2.33.1-py3-none-any.whl", hash = "sha256:4e6d1ef462f3626a1f0a0a9c42dd93c63bad33f9f1c1937509b8c5c8718ab56a", size = 64947, upload-time = "2026-03-30T16:09:13.83Z" }, +] + +[[package]] +name = "rpds-py" +version = "0.30.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/20/af/3f2f423103f1113b36230496629986e0ef7e199d2aa8392452b484b38ced/rpds_py-0.30.0.tar.gz", hash = "sha256:dd8ff7cf90014af0c0f787eea34794ebf6415242ee1d6fa91eaba725cc441e84", size = 69469, upload-time = "2025-11-30T20:24:38.837Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ed/dc/d61221eb88ff410de3c49143407f6f3147acf2538c86f2ab7ce65ae7d5f9/rpds_py-0.30.0-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:f83424d738204d9770830d35290ff3273fbb02b41f919870479fab14b9d303b2", size = 374887, upload-time = "2025-11-30T20:22:41.812Z" }, + { url = "https://files.pythonhosted.org/packages/fd/32/55fb50ae104061dbc564ef15cc43c013dc4a9f4527a1f4d99baddf56fe5f/rpds_py-0.30.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:e7536cd91353c5273434b4e003cbda89034d67e7710eab8761fd918ec6c69cf8", size = 358904, upload-time = "2025-11-30T20:22:43.479Z" }, + { url = "https://files.pythonhosted.org/packages/58/70/faed8186300e3b9bdd138d0273109784eea2396c68458ed580f885dfe7ad/rpds_py-0.30.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2771c6c15973347f50fece41fc447c054b7ac2ae0502388ce3b6738cd366e3d4", size = 389945, upload-time = "2025-11-30T20:22:44.819Z" }, + { url = "https://files.pythonhosted.org/packages/bd/a8/073cac3ed2c6387df38f71296d002ab43496a96b92c823e76f46b8af0543/rpds_py-0.30.0-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:0a59119fc6e3f460315fe9d08149f8102aa322299deaa5cab5b40092345c2136", size = 407783, upload-time = "2025-11-30T20:22:46.103Z" }, + { url = "https://files.pythonhosted.org/packages/77/57/5999eb8c58671f1c11eba084115e77a8899d6e694d2a18f69f0ba471ec8b/rpds_py-0.30.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:76fec018282b4ead0364022e3c54b60bf368b9d926877957a8624b58419169b7", size = 515021, upload-time = "2025-11-30T20:22:47.458Z" }, + { url = "https://files.pythonhosted.org/packages/e0/af/5ab4833eadc36c0a8ed2bc5c0de0493c04f6c06de223170bd0798ff98ced/rpds_py-0.30.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:692bef75a5525db97318e8cd061542b5a79812d711ea03dbc1f6f8dbb0c5f0d2", size = 414589, upload-time = "2025-11-30T20:22:48.872Z" }, + { url = "https://files.pythonhosted.org/packages/b7/de/f7192e12b21b9e9a68a6d0f249b4af3fdcdff8418be0767a627564afa1f1/rpds_py-0.30.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9027da1ce107104c50c81383cae773ef5c24d296dd11c99e2629dbd7967a20c6", size = 394025, upload-time = "2025-11-30T20:22:50.196Z" }, + { url = "https://files.pythonhosted.org/packages/91/c4/fc70cd0249496493500e7cc2de87504f5aa6509de1e88623431fec76d4b6/rpds_py-0.30.0-cp313-cp313-manylinux_2_31_riscv64.whl", hash = "sha256:9cf69cdda1f5968a30a359aba2f7f9aa648a9ce4b580d6826437f2b291cfc86e", size = 408895, upload-time = "2025-11-30T20:22:51.87Z" }, + { url = "https://files.pythonhosted.org/packages/58/95/d9275b05ab96556fefff73a385813eb66032e4c99f411d0795372d9abcea/rpds_py-0.30.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a4796a717bf12b9da9d3ad002519a86063dcac8988b030e405704ef7d74d2d9d", size = 422799, upload-time = "2025-11-30T20:22:53.341Z" }, + { url = "https://files.pythonhosted.org/packages/06/c1/3088fc04b6624eb12a57eb814f0d4997a44b0d208d6cace713033ff1a6ba/rpds_py-0.30.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:5d4c2aa7c50ad4728a094ebd5eb46c452e9cb7edbfdb18f9e1221f597a73e1e7", size = 572731, upload-time = "2025-11-30T20:22:54.778Z" }, + { url = "https://files.pythonhosted.org/packages/d8/42/c612a833183b39774e8ac8fecae81263a68b9583ee343db33ab571a7ce55/rpds_py-0.30.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:ba81a9203d07805435eb06f536d95a266c21e5b2dfbf6517748ca40c98d19e31", size = 599027, upload-time = "2025-11-30T20:22:56.212Z" }, + { url = "https://files.pythonhosted.org/packages/5f/60/525a50f45b01d70005403ae0e25f43c0384369ad24ffe46e8d9068b50086/rpds_py-0.30.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:945dccface01af02675628334f7cf49c2af4c1c904748efc5cf7bbdf0b579f95", size = 563020, upload-time = "2025-11-30T20:22:58.2Z" }, + { url = "https://files.pythonhosted.org/packages/0b/5d/47c4655e9bcd5ca907148535c10e7d489044243cc9941c16ed7cd53be91d/rpds_py-0.30.0-cp313-cp313-win32.whl", hash = "sha256:b40fb160a2db369a194cb27943582b38f79fc4887291417685f3ad693c5a1d5d", size = 223139, upload-time = "2025-11-30T20:23:00.209Z" }, + { url = "https://files.pythonhosted.org/packages/f2/e1/485132437d20aa4d3e1d8b3fb5a5e65aa8139f1e097080c2a8443201742c/rpds_py-0.30.0-cp313-cp313-win_amd64.whl", hash = "sha256:806f36b1b605e2d6a72716f321f20036b9489d29c51c91f4dd29a3e3afb73b15", size = 240224, upload-time = "2025-11-30T20:23:02.008Z" }, + { url = "https://files.pythonhosted.org/packages/24/95/ffd128ed1146a153d928617b0ef673960130be0009c77d8fbf0abe306713/rpds_py-0.30.0-cp313-cp313-win_arm64.whl", hash = "sha256:d96c2086587c7c30d44f31f42eae4eac89b60dabbac18c7669be3700f13c3ce1", size = 230645, upload-time = "2025-11-30T20:23:03.43Z" }, + { url = "https://files.pythonhosted.org/packages/ff/1b/b10de890a0def2a319a2626334a7f0ae388215eb60914dbac8a3bae54435/rpds_py-0.30.0-cp313-cp313t-macosx_10_12_x86_64.whl", hash = "sha256:eb0b93f2e5c2189ee831ee43f156ed34e2a89a78a66b98cadad955972548be5a", size = 364443, upload-time = "2025-11-30T20:23:04.878Z" }, + { url = "https://files.pythonhosted.org/packages/0d/bf/27e39f5971dc4f305a4fb9c672ca06f290f7c4e261c568f3dea16a410d47/rpds_py-0.30.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:922e10f31f303c7c920da8981051ff6d8c1a56207dbdf330d9047f6d30b70e5e", size = 353375, upload-time = "2025-11-30T20:23:06.342Z" }, + { url = "https://files.pythonhosted.org/packages/40/58/442ada3bba6e8e6615fc00483135c14a7538d2ffac30e2d933ccf6852232/rpds_py-0.30.0-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cdc62c8286ba9bf7f47befdcea13ea0e26bf294bda99758fd90535cbaf408000", size = 383850, upload-time = "2025-11-30T20:23:07.825Z" }, + { url = "https://files.pythonhosted.org/packages/14/14/f59b0127409a33c6ef6f5c1ebd5ad8e32d7861c9c7adfa9a624fc3889f6c/rpds_py-0.30.0-cp313-cp313t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:47f9a91efc418b54fb8190a6b4aa7813a23fb79c51f4bb84e418f5476c38b8db", size = 392812, upload-time = "2025-11-30T20:23:09.228Z" }, + { url = "https://files.pythonhosted.org/packages/b3/66/e0be3e162ac299b3a22527e8913767d869e6cc75c46bd844aa43fb81ab62/rpds_py-0.30.0-cp313-cp313t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1f3587eb9b17f3789ad50824084fa6f81921bbf9a795826570bda82cb3ed91f2", size = 517841, upload-time = "2025-11-30T20:23:11.186Z" }, + { url = "https://files.pythonhosted.org/packages/3d/55/fa3b9cf31d0c963ecf1ba777f7cf4b2a2c976795ac430d24a1f43d25a6ba/rpds_py-0.30.0-cp313-cp313t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:39c02563fc592411c2c61d26b6c5fe1e51eaa44a75aa2c8735ca88b0d9599daa", size = 408149, upload-time = "2025-11-30T20:23:12.864Z" }, + { url = "https://files.pythonhosted.org/packages/60/ca/780cf3b1a32b18c0f05c441958d3758f02544f1d613abf9488cd78876378/rpds_py-0.30.0-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:51a1234d8febafdfd33a42d97da7a43f5dcb120c1060e352a3fbc0c6d36e2083", size = 383843, upload-time = "2025-11-30T20:23:14.638Z" }, + { url = "https://files.pythonhosted.org/packages/82/86/d5f2e04f2aa6247c613da0c1dd87fcd08fa17107e858193566048a1e2f0a/rpds_py-0.30.0-cp313-cp313t-manylinux_2_31_riscv64.whl", hash = "sha256:eb2c4071ab598733724c08221091e8d80e89064cd472819285a9ab0f24bcedb9", size = 396507, upload-time = "2025-11-30T20:23:16.105Z" }, + { url = "https://files.pythonhosted.org/packages/4b/9a/453255d2f769fe44e07ea9785c8347edaf867f7026872e76c1ad9f7bed92/rpds_py-0.30.0-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:6bdfdb946967d816e6adf9a3d8201bfad269c67efe6cefd7093ef959683c8de0", size = 414949, upload-time = "2025-11-30T20:23:17.539Z" }, + { url = "https://files.pythonhosted.org/packages/a3/31/622a86cdc0c45d6df0e9ccb6becdba5074735e7033c20e401a6d9d0e2ca0/rpds_py-0.30.0-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:c77afbd5f5250bf27bf516c7c4a016813eb2d3e116139aed0096940c5982da94", size = 565790, upload-time = "2025-11-30T20:23:19.029Z" }, + { url = "https://files.pythonhosted.org/packages/1c/5d/15bbf0fb4a3f58a3b1c67855ec1efcc4ceaef4e86644665fff03e1b66d8d/rpds_py-0.30.0-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:61046904275472a76c8c90c9ccee9013d70a6d0f73eecefd38c1ae7c39045a08", size = 590217, upload-time = "2025-11-30T20:23:20.885Z" }, + { url = "https://files.pythonhosted.org/packages/6d/61/21b8c41f68e60c8cc3b2e25644f0e3681926020f11d06ab0b78e3c6bbff1/rpds_py-0.30.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:4c5f36a861bc4b7da6516dbdf302c55313afa09b81931e8280361a4f6c9a2d27", size = 555806, upload-time = "2025-11-30T20:23:22.488Z" }, + { url = "https://files.pythonhosted.org/packages/f9/39/7e067bb06c31de48de3eb200f9fc7c58982a4d3db44b07e73963e10d3be9/rpds_py-0.30.0-cp313-cp313t-win32.whl", hash = "sha256:3d4a69de7a3e50ffc214ae16d79d8fbb0922972da0356dcf4d0fdca2878559c6", size = 211341, upload-time = "2025-11-30T20:23:24.449Z" }, + { url = "https://files.pythonhosted.org/packages/0a/4d/222ef0b46443cf4cf46764d9c630f3fe4abaa7245be9417e56e9f52b8f65/rpds_py-0.30.0-cp313-cp313t-win_amd64.whl", hash = "sha256:f14fc5df50a716f7ece6a80b6c78bb35ea2ca47c499e422aa4463455dd96d56d", size = 225768, upload-time = "2025-11-30T20:23:25.908Z" }, + { url = "https://files.pythonhosted.org/packages/86/81/dad16382ebbd3d0e0328776d8fd7ca94220e4fa0798d1dc5e7da48cb3201/rpds_py-0.30.0-cp314-cp314-macosx_10_12_x86_64.whl", hash = "sha256:68f19c879420aa08f61203801423f6cd5ac5f0ac4ac82a2368a9fcd6a9a075e0", size = 362099, upload-time = "2025-11-30T20:23:27.316Z" }, + { url = "https://files.pythonhosted.org/packages/2b/60/19f7884db5d5603edf3c6bce35408f45ad3e97e10007df0e17dd57af18f8/rpds_py-0.30.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:ec7c4490c672c1a0389d319b3a9cfcd098dcdc4783991553c332a15acf7249be", size = 353192, upload-time = "2025-11-30T20:23:29.151Z" }, + { url = "https://files.pythonhosted.org/packages/bf/c4/76eb0e1e72d1a9c4703c69607cec123c29028bff28ce41588792417098ac/rpds_py-0.30.0-cp314-cp314-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f251c812357a3fed308d684a5079ddfb9d933860fc6de89f2b7ab00da481e65f", size = 384080, upload-time = "2025-11-30T20:23:30.785Z" }, + { url = "https://files.pythonhosted.org/packages/72/87/87ea665e92f3298d1b26d78814721dc39ed8d2c74b86e83348d6b48a6f31/rpds_py-0.30.0-cp314-cp314-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:ac98b175585ecf4c0348fd7b29c3864bda53b805c773cbf7bfdaffc8070c976f", size = 394841, upload-time = "2025-11-30T20:23:32.209Z" }, + { url = "https://files.pythonhosted.org/packages/77/ad/7783a89ca0587c15dcbf139b4a8364a872a25f861bdb88ed99f9b0dec985/rpds_py-0.30.0-cp314-cp314-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3e62880792319dbeb7eb866547f2e35973289e7d5696c6e295476448f5b63c87", size = 516670, upload-time = "2025-11-30T20:23:33.742Z" }, + { url = "https://files.pythonhosted.org/packages/5b/3c/2882bdac942bd2172f3da574eab16f309ae10a3925644e969536553cb4ee/rpds_py-0.30.0-cp314-cp314-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4e7fc54e0900ab35d041b0601431b0a0eb495f0851a0639b6ef90f7741b39a18", size = 408005, upload-time = "2025-11-30T20:23:35.253Z" }, + { url = "https://files.pythonhosted.org/packages/ce/81/9a91c0111ce1758c92516a3e44776920b579d9a7c09b2b06b642d4de3f0f/rpds_py-0.30.0-cp314-cp314-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47e77dc9822d3ad616c3d5759ea5631a75e5809d5a28707744ef79d7a1bcfcad", size = 382112, upload-time = "2025-11-30T20:23:36.842Z" }, + { url = "https://files.pythonhosted.org/packages/cf/8e/1da49d4a107027e5fbc64daeab96a0706361a2918da10cb41769244b805d/rpds_py-0.30.0-cp314-cp314-manylinux_2_31_riscv64.whl", hash = "sha256:b4dc1a6ff022ff85ecafef7979a2c6eb423430e05f1165d6688234e62ba99a07", size = 399049, upload-time = "2025-11-30T20:23:38.343Z" }, + { url = "https://files.pythonhosted.org/packages/df/5a/7ee239b1aa48a127570ec03becbb29c9d5a9eb092febbd1699d567cae859/rpds_py-0.30.0-cp314-cp314-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4559c972db3a360808309e06a74628b95eaccbf961c335c8fe0d590cf587456f", size = 415661, upload-time = "2025-11-30T20:23:40.263Z" }, + { url = "https://files.pythonhosted.org/packages/70/ea/caa143cf6b772f823bc7929a45da1fa83569ee49b11d18d0ada7f5ee6fd6/rpds_py-0.30.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:0ed177ed9bded28f8deb6ab40c183cd1192aa0de40c12f38be4d59cd33cb5c65", size = 565606, upload-time = "2025-11-30T20:23:42.186Z" }, + { url = "https://files.pythonhosted.org/packages/64/91/ac20ba2d69303f961ad8cf55bf7dbdb4763f627291ba3d0d7d67333cced9/rpds_py-0.30.0-cp314-cp314-musllinux_1_2_i686.whl", hash = "sha256:ad1fa8db769b76ea911cb4e10f049d80bf518c104f15b3edb2371cc65375c46f", size = 591126, upload-time = "2025-11-30T20:23:44.086Z" }, + { url = "https://files.pythonhosted.org/packages/21/20/7ff5f3c8b00c8a95f75985128c26ba44503fb35b8e0259d812766ea966c7/rpds_py-0.30.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:46e83c697b1f1c72b50e5ee5adb4353eef7406fb3f2043d64c33f20ad1c2fc53", size = 553371, upload-time = "2025-11-30T20:23:46.004Z" }, + { url = "https://files.pythonhosted.org/packages/72/c7/81dadd7b27c8ee391c132a6b192111ca58d866577ce2d9b0ca157552cce0/rpds_py-0.30.0-cp314-cp314-win32.whl", hash = "sha256:ee454b2a007d57363c2dfd5b6ca4a5d7e2c518938f8ed3b706e37e5d470801ed", size = 215298, upload-time = "2025-11-30T20:23:47.696Z" }, + { url = "https://files.pythonhosted.org/packages/3e/d2/1aaac33287e8cfb07aab2e6b8ac1deca62f6f65411344f1433c55e6f3eb8/rpds_py-0.30.0-cp314-cp314-win_amd64.whl", hash = "sha256:95f0802447ac2d10bcc69f6dc28fe95fdf17940367b21d34e34c737870758950", size = 228604, upload-time = "2025-11-30T20:23:49.501Z" }, + { url = "https://files.pythonhosted.org/packages/e8/95/ab005315818cc519ad074cb7784dae60d939163108bd2b394e60dc7b5461/rpds_py-0.30.0-cp314-cp314-win_arm64.whl", hash = "sha256:613aa4771c99f03346e54c3f038e4cc574ac09a3ddfb0e8878487335e96dead6", size = 222391, upload-time = "2025-11-30T20:23:50.96Z" }, + { url = "https://files.pythonhosted.org/packages/9e/68/154fe0194d83b973cdedcdcc88947a2752411165930182ae41d983dcefa6/rpds_py-0.30.0-cp314-cp314t-macosx_10_12_x86_64.whl", hash = "sha256:7e6ecfcb62edfd632e56983964e6884851786443739dbfe3582947e87274f7cb", size = 364868, upload-time = "2025-11-30T20:23:52.494Z" }, + { url = "https://files.pythonhosted.org/packages/83/69/8bbc8b07ec854d92a8b75668c24d2abcb1719ebf890f5604c61c9369a16f/rpds_py-0.30.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:a1d0bc22a7cdc173fedebb73ef81e07faef93692b8c1ad3733b67e31e1b6e1b8", size = 353747, upload-time = "2025-11-30T20:23:54.036Z" }, + { url = "https://files.pythonhosted.org/packages/ab/00/ba2e50183dbd9abcce9497fa5149c62b4ff3e22d338a30d690f9af970561/rpds_py-0.30.0-cp314-cp314t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0d08f00679177226c4cb8c5265012eea897c8ca3b93f429e546600c971bcbae7", size = 383795, upload-time = "2025-11-30T20:23:55.556Z" }, + { url = "https://files.pythonhosted.org/packages/05/6f/86f0272b84926bcb0e4c972262f54223e8ecc556b3224d281e6598fc9268/rpds_py-0.30.0-cp314-cp314t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5965af57d5848192c13534f90f9dd16464f3c37aaf166cc1da1cae1fd5a34898", size = 393330, upload-time = "2025-11-30T20:23:57.033Z" }, + { url = "https://files.pythonhosted.org/packages/cb/e9/0e02bb2e6dc63d212641da45df2b0bf29699d01715913e0d0f017ee29438/rpds_py-0.30.0-cp314-cp314t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9a4e86e34e9ab6b667c27f3211ca48f73dba7cd3d90f8d5b11be56e5dbc3fb4e", size = 518194, upload-time = "2025-11-30T20:23:58.637Z" }, + { url = "https://files.pythonhosted.org/packages/ee/ca/be7bca14cf21513bdf9c0606aba17d1f389ea2b6987035eb4f62bd923f25/rpds_py-0.30.0-cp314-cp314t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e5d3e6b26f2c785d65cc25ef1e5267ccbe1b069c5c21b8cc724efee290554419", size = 408340, upload-time = "2025-11-30T20:24:00.2Z" }, + { url = "https://files.pythonhosted.org/packages/c2/c7/736e00ebf39ed81d75544c0da6ef7b0998f8201b369acf842f9a90dc8fce/rpds_py-0.30.0-cp314-cp314t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:626a7433c34566535b6e56a1b39a7b17ba961e97ce3b80ec62e6f1312c025551", size = 383765, upload-time = "2025-11-30T20:24:01.759Z" }, + { url = "https://files.pythonhosted.org/packages/4a/3f/da50dfde9956aaf365c4adc9533b100008ed31aea635f2b8d7b627e25b49/rpds_py-0.30.0-cp314-cp314t-manylinux_2_31_riscv64.whl", hash = "sha256:acd7eb3f4471577b9b5a41baf02a978e8bdeb08b4b355273994f8b87032000a8", size = 396834, upload-time = "2025-11-30T20:24:03.687Z" }, + { url = "https://files.pythonhosted.org/packages/4e/00/34bcc2565b6020eab2623349efbdec810676ad571995911f1abdae62a3a0/rpds_py-0.30.0-cp314-cp314t-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:fe5fa731a1fa8a0a56b0977413f8cacac1768dad38d16b3a296712709476fbd5", size = 415470, upload-time = "2025-11-30T20:24:05.232Z" }, + { url = "https://files.pythonhosted.org/packages/8c/28/882e72b5b3e6f718d5453bd4d0d9cf8df36fddeb4ddbbab17869d5868616/rpds_py-0.30.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:74a3243a411126362712ee1524dfc90c650a503502f135d54d1b352bd01f2404", size = 565630, upload-time = "2025-11-30T20:24:06.878Z" }, + { url = "https://files.pythonhosted.org/packages/3b/97/04a65539c17692de5b85c6e293520fd01317fd878ea1995f0367d4532fb1/rpds_py-0.30.0-cp314-cp314t-musllinux_1_2_i686.whl", hash = "sha256:3e8eeb0544f2eb0d2581774be4c3410356eba189529a6b3e36bbbf9696175856", size = 591148, upload-time = "2025-11-30T20:24:08.445Z" }, + { url = "https://files.pythonhosted.org/packages/85/70/92482ccffb96f5441aab93e26c4d66489eb599efdcf96fad90c14bbfb976/rpds_py-0.30.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:dbd936cde57abfee19ab3213cf9c26be06d60750e60a8e4dd85d1ab12c8b1f40", size = 556030, upload-time = "2025-11-30T20:24:10.956Z" }, + { url = "https://files.pythonhosted.org/packages/20/53/7c7e784abfa500a2b6b583b147ee4bb5a2b3747a9166bab52fec4b5b5e7d/rpds_py-0.30.0-cp314-cp314t-win32.whl", hash = "sha256:dc824125c72246d924f7f796b4f63c1e9dc810c7d9e2355864b3c3a73d59ade0", size = 211570, upload-time = "2025-11-30T20:24:12.735Z" }, + { url = "https://files.pythonhosted.org/packages/d0/02/fa464cdfbe6b26e0600b62c528b72d8608f5cc49f96b8d6e38c95d60c676/rpds_py-0.30.0-cp314-cp314t-win_amd64.whl", hash = "sha256:27f4b0e92de5bfbc6f86e43959e6edd1425c33b5e69aab0984a72047f2bcf1e3", size = 226532, upload-time = "2025-11-30T20:24:14.634Z" }, +] + +[[package]] +name = "six" +version = "1.17.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/94/e7/b2c673351809dca68a0e064b6af791aa332cf192da575fd474ed7d6f16a2/six-1.17.0.tar.gz", hash = "sha256:ff70335d468e7eb6ec65b95b99d3a2836546063f63acc5171de367e834932a81", size = 34031, upload-time = "2024-12-04T17:35:28.174Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b7/ce/149a00dd41f10bc29e5921b496af8b574d8413afcd5e30dfa0ed46c2cc5e/six-1.17.0-py2.py3-none-any.whl", hash = "sha256:4721f391ed90541fddacab5acf947aa0d3dc7d27b2e1e8eda2be8970586c3274", size = 11050, upload-time = "2024-12-04T17:35:26.475Z" }, +] + +[[package]] +name = "smmap" +version = "5.0.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/1f/ea/49c993d6dfdd7338c9b1000a0f36817ed7ec84577ae2e52f890d1a4ff909/smmap-5.0.3.tar.gz", hash = "sha256:4d9debb8b99007ae47165abc08670bd74cb74b5227dda7f643eccc4e9eb5642c", size = 22506, upload-time = "2026-03-09T03:43:26.1Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/c1/d4/59e74daffcb57a07668852eeeb6035af9f32cbfd7a1d2511f17d2fe6a738/smmap-5.0.3-py3-none-any.whl", hash = "sha256:c106e05d5a61449cf6ba9a1e650227ecfb141590d2a98412103ff35d89fc7b2f", size = 24390, upload-time = "2026-03-09T03:43:24.361Z" }, +] + +[[package]] +name = "sqlingo" +version = "0.1.0" +source = { virtual = "." } +dependencies = [ + { name = "duckdb" }, + { name = "streamlit" }, +] + +[package.metadata] +requires-dist = [ + { name = "duckdb", specifier = ">=1.5.2" }, + { name = "streamlit", specifier = ">=1.56.0" }, +] + +[[package]] +name = "streamlit" +version = "1.56.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "altair" }, + { name = "blinker" }, + { name = "cachetools" }, + { name = "click" }, + { name = "gitpython" }, + { name = "numpy" }, + { name = "packaging" }, + { name = "pandas" }, + { name = "pillow" }, + { name = "protobuf" }, + { name = "pyarrow" }, + { name = "pydeck" }, + { name = "requests" }, + { name = "tenacity" }, + { name = "toml" }, + { name = "tornado" }, + { name = "typing-extensions" }, + { name = "watchdog", marker = "sys_platform != 'darwin'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/03/85/7c669b3a1336d34ef39fa9760fbd343185f3b15db2ad0838fd78423d1c7f/streamlit-1.56.0.tar.gz", hash = "sha256:1176acfa89ae1318b79078e8efe689a9d02e8d58e325c00fc0e55fa2f3fe8d6a", size = 8559239, upload-time = "2026-03-31T22:29:38.59Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/e4/91/cb6f13a89e376ef179309d74f37a70ea0041d5e4b5ba5c4836dbf6e020ad/streamlit-1.56.0-py3-none-any.whl", hash = "sha256:8677a335734a30a51bc57ad0ec910e365d95f7c456fc02c60032927cd0729dc5", size = 9052089, upload-time = "2026-03-31T22:29:36.342Z" }, +] + +[[package]] +name = "tenacity" +version = "9.1.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/47/c6/ee486fd809e357697ee8a44d3d69222b344920433d3b6666ccd9b374630c/tenacity-9.1.4.tar.gz", hash = "sha256:adb31d4c263f2bd041081ab33b498309a57c77f9acf2db65aadf0898179cf93a", size = 49413, upload-time = "2026-02-07T10:45:33.841Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d7/c1/eb8f9debc45d3b7918a32ab756658a0904732f75e555402972246b0b8e71/tenacity-9.1.4-py3-none-any.whl", hash = "sha256:6095a360c919085f28c6527de529e76a06ad89b23659fa881ae0649b867a9d55", size = 28926, upload-time = "2026-02-07T10:45:32.24Z" }, +] + +[[package]] +name = "toml" +version = "0.10.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/be/ba/1f744cdc819428fc6b5084ec34d9b30660f6f9daaf70eead706e3203ec3c/toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f", size = 22253, upload-time = "2020-11-01T01:40:22.204Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/44/6f/7120676b6d73228c96e17f1f794d8ab046fc910d781c8d151120c3f1569e/toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b", size = 16588, upload-time = "2020-11-01T01:40:20.672Z" }, +] + +[[package]] +name = "tornado" +version = "6.5.5" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/f8/f1/3173dfa4a18db4a9b03e5d55325559dab51ee653763bb8745a75af491286/tornado-6.5.5.tar.gz", hash = "sha256:192b8f3ea91bd7f1f50c06955416ed76c6b72f96779b962f07f911b91e8d30e9", size = 516006, upload-time = "2026-03-10T21:31:02.067Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/59/8c/77f5097695f4dd8255ecbd08b2a1ed8ba8b953d337804dd7080f199e12bf/tornado-6.5.5-cp39-abi3-macosx_10_9_universal2.whl", hash = "sha256:487dc9cc380e29f58c7ab88f9e27cdeef04b2140862e5076a66fb6bb68bb1bfa", size = 445983, upload-time = "2026-03-10T21:30:44.28Z" }, + { url = "https://files.pythonhosted.org/packages/ab/5e/7625b76cd10f98f1516c36ce0346de62061156352353ef2da44e5c21523c/tornado-6.5.5-cp39-abi3-macosx_10_9_x86_64.whl", hash = "sha256:65a7f1d46d4bb41df1ac99f5fcb685fb25c7e61613742d5108b010975a9a6521", size = 444246, upload-time = "2026-03-10T21:30:46.571Z" }, + { url = "https://files.pythonhosted.org/packages/b2/04/7b5705d5b3c0fab088f434f9c83edac1573830ca49ccf29fb83bf7178eec/tornado-6.5.5-cp39-abi3-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:e74c92e8e65086b338fd56333fb9a68b9f6f2fe7ad532645a290a464bcf46be5", size = 447229, upload-time = "2026-03-10T21:30:48.273Z" }, + { url = "https://files.pythonhosted.org/packages/34/01/74e034a30ef59afb4097ef8659515e96a39d910b712a89af76f5e4e1f93c/tornado-6.5.5-cp39-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:435319e9e340276428bbdb4e7fa732c2d399386d1de5686cb331ec8eee754f07", size = 448192, upload-time = "2026-03-10T21:30:51.22Z" }, + { url = "https://files.pythonhosted.org/packages/be/00/fe9e02c5a96429fce1a1d15a517f5d8444f9c412e0bb9eadfbe3b0fc55bf/tornado-6.5.5-cp39-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:3f54aa540bdbfee7b9eb268ead60e7d199de5021facd276819c193c0fb28ea4e", size = 448039, upload-time = "2026-03-10T21:30:53.52Z" }, + { url = "https://files.pythonhosted.org/packages/82/9e/656ee4cec0398b1d18d0f1eb6372c41c6b889722641d84948351ae19556d/tornado-6.5.5-cp39-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:36abed1754faeb80fbd6e64db2758091e1320f6bba74a4cf8c09cd18ccce8aca", size = 447445, upload-time = "2026-03-10T21:30:55.541Z" }, + { url = "https://files.pythonhosted.org/packages/5a/76/4921c00511f88af86a33de770d64141170f1cfd9c00311aea689949e274e/tornado-6.5.5-cp39-abi3-win32.whl", hash = "sha256:dd3eafaaeec1c7f2f8fdcd5f964e8907ad788fe8a5a32c4426fbbdda621223b7", size = 448582, upload-time = "2026-03-10T21:30:57.142Z" }, + { url = "https://files.pythonhosted.org/packages/2c/23/f6c6112a04d28eed765e374435fb1a9198f73e1ec4b4024184f21faeb1ad/tornado-6.5.5-cp39-abi3-win_amd64.whl", hash = "sha256:6443a794ba961a9f619b1ae926a2e900ac20c34483eea67be4ed8f1e58d3ef7b", size = 448990, upload-time = "2026-03-10T21:30:58.857Z" }, + { url = "https://files.pythonhosted.org/packages/b7/c8/876602cbc96469911f0939f703453c1157b0c826ecb05bdd32e023397d4e/tornado-6.5.5-cp39-abi3-win_arm64.whl", hash = "sha256:2c9a876e094109333f888539ddb2de4361743e5d21eece20688e3e351e4990a6", size = 448016, upload-time = "2026-03-10T21:31:00.43Z" }, +] + +[[package]] +name = "typing-extensions" +version = "4.15.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391, upload-time = "2025-08-25T13:49:26.313Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614, upload-time = "2025-08-25T13:49:24.86Z" }, +] + +[[package]] +name = "tzdata" +version = "2026.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ba/19/1b9b0e29f30c6d35cb345486df41110984ea67ae69dddbc0e8a100999493/tzdata-2026.2.tar.gz", hash = "sha256:9173fde7d80d9018e02a662e168e5a2d04f87c41ea174b139fbef642eda62d10", size = 198254, upload-time = "2026-04-24T15:22:08.651Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ce/e4/dccd7f47c4b64213ac01ef921a1337ee6e30e8c6466046018326977efd95/tzdata-2026.2-py2.py3-none-any.whl", hash = "sha256:bbe9af844f658da81a5f95019480da3a89415801f6cc966806612cc7169bffe7", size = 349321, upload-time = "2026-04-24T15:22:05.876Z" }, +] + +[[package]] +name = "urllib3" +version = "2.6.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/c7/24/5f1b3bdffd70275f6661c76461e25f024d5a38a46f04aaca912426a2b1d3/urllib3-2.6.3.tar.gz", hash = "sha256:1b62b6884944a57dbe321509ab94fd4d3b307075e0c2eae991ac71ee15ad38ed", size = 435556, upload-time = "2026-01-07T16:24:43.925Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/39/08/aaaad47bc4e9dc8c725e68f9d04865dbcb2052843ff09c97b08904852d84/urllib3-2.6.3-py3-none-any.whl", hash = "sha256:bf272323e553dfb2e87d9bfd225ca7b0f467b919d7bbd355436d3fd37cb0acd4", size = 131584, upload-time = "2026-01-07T16:24:42.685Z" }, +] + +[[package]] +name = "watchdog" +version = "6.0.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/db/7d/7f3d619e951c88ed75c6037b246ddcf2d322812ee8ea189be89511721d54/watchdog-6.0.0.tar.gz", hash = "sha256:9ddf7c82fda3ae8e24decda1338ede66e1c99883db93711d8fb941eaa2d8c282", size = 131220, upload-time = "2024-11-01T14:07:13.037Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a9/c7/ca4bf3e518cb57a686b2feb4f55a1892fd9a3dd13f470fca14e00f80ea36/watchdog-6.0.0-py3-none-manylinux2014_aarch64.whl", hash = "sha256:7607498efa04a3542ae3e05e64da8202e58159aa1fa4acddf7678d34a35d4f13", size = 79079, upload-time = "2024-11-01T14:06:59.472Z" }, + { url = "https://files.pythonhosted.org/packages/5c/51/d46dc9332f9a647593c947b4b88e2381c8dfc0942d15b8edc0310fa4abb1/watchdog-6.0.0-py3-none-manylinux2014_armv7l.whl", hash = "sha256:9041567ee8953024c83343288ccc458fd0a2d811d6a0fd68c4c22609e3490379", size = 79078, upload-time = "2024-11-01T14:07:01.431Z" }, + { url = "https://files.pythonhosted.org/packages/d4/57/04edbf5e169cd318d5f07b4766fee38e825d64b6913ca157ca32d1a42267/watchdog-6.0.0-py3-none-manylinux2014_i686.whl", hash = "sha256:82dc3e3143c7e38ec49d61af98d6558288c415eac98486a5c581726e0737c00e", size = 79076, upload-time = "2024-11-01T14:07:02.568Z" }, + { url = "https://files.pythonhosted.org/packages/ab/cc/da8422b300e13cb187d2203f20b9253e91058aaf7db65b74142013478e66/watchdog-6.0.0-py3-none-manylinux2014_ppc64.whl", hash = "sha256:212ac9b8bf1161dc91bd09c048048a95ca3a4c4f5e5d4a7d1b1a7d5752a7f96f", size = 79077, upload-time = "2024-11-01T14:07:03.893Z" }, + { url = "https://files.pythonhosted.org/packages/2c/3b/b8964e04ae1a025c44ba8e4291f86e97fac443bca31de8bd98d3263d2fcf/watchdog-6.0.0-py3-none-manylinux2014_ppc64le.whl", hash = "sha256:e3df4cbb9a450c6d49318f6d14f4bbc80d763fa587ba46ec86f99f9e6876bb26", size = 79078, upload-time = "2024-11-01T14:07:05.189Z" }, + { url = "https://files.pythonhosted.org/packages/62/ae/a696eb424bedff7407801c257d4b1afda455fe40821a2be430e173660e81/watchdog-6.0.0-py3-none-manylinux2014_s390x.whl", hash = "sha256:2cce7cfc2008eb51feb6aab51251fd79b85d9894e98ba847408f662b3395ca3c", size = 79077, upload-time = "2024-11-01T14:07:06.376Z" }, + { url = "https://files.pythonhosted.org/packages/b5/e8/dbf020b4d98251a9860752a094d09a65e1b436ad181faf929983f697048f/watchdog-6.0.0-py3-none-manylinux2014_x86_64.whl", hash = "sha256:20ffe5b202af80ab4266dcd3e91aae72bf2da48c0d33bdb15c66658e685e94e2", size = 79078, upload-time = "2024-11-01T14:07:07.547Z" }, + { url = "https://files.pythonhosted.org/packages/07/f6/d0e5b343768e8bcb4cda79f0f2f55051bf26177ecd5651f84c07567461cf/watchdog-6.0.0-py3-none-win32.whl", hash = "sha256:07df1fdd701c5d4c8e55ef6cf55b8f0120fe1aef7ef39a1c6fc6bc2e606d517a", size = 79065, upload-time = "2024-11-01T14:07:09.525Z" }, + { url = "https://files.pythonhosted.org/packages/db/d9/c495884c6e548fce18a8f40568ff120bc3a4b7b99813081c8ac0c936fa64/watchdog-6.0.0-py3-none-win_amd64.whl", hash = "sha256:cbafb470cf848d93b5d013e2ecb245d4aa1c8fd0504e863ccefa32445359d680", size = 79070, upload-time = "2024-11-01T14:07:10.686Z" }, + { url = "https://files.pythonhosted.org/packages/33/e8/e40370e6d74ddba47f002a32919d91310d6074130fe4e17dabcafc15cbf1/watchdog-6.0.0-py3-none-win_ia64.whl", hash = "sha256:a1914259fa9e1454315171103c6a30961236f508b9b623eae470268bbcc6a22f", size = 79067, upload-time = "2024-11-01T14:07:11.845Z" }, +] From 3ef281a644757f83bd39852083ebd109b7e34d45 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:34:03 +0000 Subject: [PATCH 02/18] chore: ignore *.duckdb --- .gitignore | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index 20e3a62..28d6b50 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,5 @@ venv/ data/ __pycache__ -test.py \ No newline at end of file +test.py +*.duckdb \ No newline at end of file From 3d9a9d62b327bec0fcee73f87126dbc93d5ea224 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:34:54 +0000 Subject: [PATCH 03/18] chore: ignore .venv/ --- .gitignore | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index 28d6b50..d06d94f 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,4 @@ -venv/ +.venv/ data/ __pycache__ test.py From b49f313774faf25e7407c6ef2044bc3e94d024cf Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:35:43 +0000 Subject: [PATCH 04/18] feat(config): add a configuration file to maintain constants --- config.py | 13 +++++++++++++ 1 file changed, 13 insertions(+) create mode 100644 config.py diff --git a/config.py b/config.py new file mode 100644 index 0000000..9cbf5d2 --- /dev/null +++ b/config.py @@ -0,0 +1,13 @@ +"""Constantes globales pour le projet SQLingo.""" + +from pathlib import Path + +# Paramètres chemin d'accès +DB_PATH = Path("exercices_tables.duckdb") +EXERCICES_PATH = Path("Exercices.yaml") +QUESTIONS_PATH = Path("Exercices/questions") +ANSWERS_PATH = Path("Exercices/answers") + +# Paramètres SRS +SRS_CORRECT_DELAY = None +SRS_INCORRECT_DELAY = 3 From a2f350ef2ea3b7f97f5b49a0496b40fafada9839 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:36:44 +0000 Subject: [PATCH 05/18] feat(exercices): exercices are described in a yaml file so its easier to maintain --- Exercices.yaml | 93 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 93 insertions(+) create mode 100644 Exercices.yaml diff --git a/Exercices.yaml b/Exercices.yaml new file mode 100644 index 0000000..f3ead13 --- /dev/null +++ b/Exercices.yaml @@ -0,0 +1,93 @@ +# Source de vérité unique pour tous les exercices SQLingo. +# Pour ajouter un exercice : ajouter un bloc ici + le fichier .sql correspondant. +# +# Champs : +# theme : catégorie de l'exercice (utilisé pour grouper) +# name : identifiant unique de l'exercice +# tables : tables DuckDB nécessaires (pour affichage uniquement) +# answer : nom du fichier SQL solution dans Exercices/answers// + +- theme: cross_joins + name: cross_joins_1 + tables: [beverages, food_items] + answer: cross_joins_1.sql + +- theme: cross_joins + name: cross_joins_2 + tables: [sizes, trademarks] + answer: cross_joins_2.sql + +- theme: cross_joins + name: cross_joins_3 + tables: [hours, quarters] + answer: cross_joins_3.sql + +- theme: inner_joins + name: inner_joins_1 + tables: [salaries, seniorities] + answer: inner_joins_1.sql + +- theme: left_joins + name: left_joins_1 + tables: [products, order_details] + answer: left_joins_1.sql + +- theme: left_joins + name: left_joins_2 + tables: [orders, customers, products, order_details] + answer: left_joins_2.sql + +- theme: full_outer_joins + name: full_outer_joins_1 + tables: [df_store_products, df_products] + answer: full_outer_joins_1.sql + +- theme: full_outer_joins + name: full_outer_joins_2 + tables: [df_customers, df_stores, df_store_products, df_products] + answer: full_outer_joins_2.sql + +- theme: self_joins + name: self_joins_1 + tables: [employees] + answer: self_joins_1.sql + +- theme: self_joins + name: self_joins_2 + tables: [sales] + answer: self_joins_2.sql + +- theme: group_by + name: group_by_1 + tables: [ventes_immo] + answer: group_by_1.sql + +- theme: group_by + name: group_by_2 + tables: [ventes] + answer: group_by_2.sql + +- theme: case_when + name: case_when_1 + tables: [salaires] + answer: case_when_1.sql + +- theme: case_when + name: case_when_2 + tables: [discount] + answer: case_when_2.sql + +- theme: case_when + name: case_when_3 + tables: [salaires] + answer: case_when_3.sql + +- theme: grouping_set + name: grouping_set_1 + tables: [redbull] + answer: grouping_set_1.sql + +- theme: grouping_set + name: grouping_set_2 + tables: [datapop] + answer: grouping_set_2.sql \ No newline at end of file From 1b7dbf558cd731efd77ca1872b70d4735d5191d4 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:37:21 +0000 Subject: [PATCH 06/18] feat(db): add a db file to insert data into tables --- db.py | 218 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 218 insertions(+) create mode 100644 db.py diff --git a/db.py b/db.py new file mode 100644 index 0000000..0c6709d --- /dev/null +++ b/db.py @@ -0,0 +1,218 @@ +""" +db.py - Initialisation de la base de données et connexion partagée. + +Ce module a deux responsabilités : + 1. Créer les tables de données des exercices (à lancer une seule fois). + 2. Exposer get_connection() pour partager une connexion DuckDB dans toute l'app. +""" + +import io +import random +import numpy as np +import pandas as pd +import duckdb +import streamlit as st +from pathlib import Path + +from config import DB_PATH + + +# ------------------------------------------------------------ +# CONNEXION PARTAGÉE +# ------------------------------------------------------------ + +@st.cache_resource +def get_connection() -> duckdb.DuckDBPyConnection: + """Retourne une connexion DuckDB partagée pour toute la session Streamlit.""" + return duckdb.connect(DB_PATH) + + +# ------------------------------------------------------------ +# CRÉATION DES TABLES D'EXERCICES +# ------------------------------------------------------------ + +def create_cross_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: + beverages_df = pd.read_csv(io.StringIO("beverage,price\norange juice,2.5\nExpresso,2\nTea,3\n")) + con.execute("CREATE TABLE IF NOT EXISTS beverages AS SELECT * FROM beverages_df") + + food_items_df = pd.read_csv(io.StringIO("food_item,food_price\ncookie,2.5\ncholatine,2\nmuffin,3\n")) + con.execute("CREATE TABLE IF NOT EXISTS food_items AS SELECT * FROM food_items_df") + + sizes_df = pd.read_csv(io.StringIO("size\nXS\nM\nL\nXL\n")) + con.execute("CREATE TABLE IF NOT EXISTS sizes AS SELECT * FROM sizes_df") + + trademarks_df = pd.read_csv(io.StringIO("trademark\nNike\nAsphalte\nAbercrombie\nLewis\n")) + con.execute("CREATE TABLE IF NOT EXISTS trademarks AS SELECT * FROM trademarks_df") + + hours_df = pd.read_csv(io.StringIO("hour\n08\n09\n10\n11\n12\n")) + con.execute("CREATE TABLE IF NOT EXISTS hours AS SELECT * FROM hours_df") + + quarters_df = pd.read_csv(io.StringIO("quarter\n00\n15\n30\n45\n")) + con.execute("CREATE TABLE IF NOT EXISTS quarters AS SELECT * FROM quarters_df") + + +def create_inner_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: + salaries_df = pd.read_csv(io.StringIO("salary,employee_id\n2000,1\n2500,2\n2200,3\n")) + con.execute("CREATE TABLE IF NOT EXISTS salaries AS SELECT * FROM salaries_df") + + seniorities_df = pd.read_csv(io.StringIO("employee_id,seniority\n1,2ans\n2,4ans\n")) + con.execute("CREATE TABLE IF NOT EXISTS seniorities AS SELECT * FROM seniorities_df") + + +def create_left_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: + orders_df = pd.DataFrame({"order_id": [1, 2, 3, 4, 5], "customer_id": [101, 102, 103, 104, 105]}) + con.execute("CREATE TABLE IF NOT EXISTS orders AS SELECT * FROM orders_df") + + customers_df = pd.DataFrame({ + "customer_id": [101, 102, 103, 104, 105, 106], + "customer_name": ["Toufik", "Daniel", "Tancrède", "Kaouter", "Jean-Nicolas", "David"], + }) + con.execute("CREATE TABLE IF NOT EXISTS customers AS SELECT * FROM customers_df") + + products_df = pd.DataFrame({ + "product_id": [101, 103, 104, 105], + "product_name": ["Laptop", "Ipad", "Livre", "Petitos"], + "product_price": [800, 400, 30, 2], + }) + con.execute("CREATE TABLE IF NOT EXISTS products AS SELECT * FROM products_df") + + order_details_df = pd.DataFrame({ + "order_id": [1, 2, 3, 4, 5], + "product_id": [102, 104, 101, 103, 105], + "quantity": [2, 1, 3, 2, 1], + }) + con.execute("CREATE TABLE IF NOT EXISTS order_details AS SELECT * FROM order_details_df") + + +def create_full_outer_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: + customers_df = pd.DataFrame({ + "customer_id": [11, 12, 13, 14, 15], + "customer_name": ["Zeinaba", "Tancrède", "Israel", "Kaouter", "Alan"], + }) + con.execute("CREATE TABLE IF NOT EXISTS df_customers AS SELECT * FROM customers_df") + + stores_df = pd.DataFrame({"store_id": [1, 2, 3, 4], "customer_id": [11, 12, 13, 15]}) + con.execute("CREATE TABLE IF NOT EXISTS df_stores AS SELECT * FROM stores_df") + + store_products_df = pd.DataFrame({ + "store_id": [1, 1, 1, 2, 2, 3, 4], + "product_id": [101, 103, 105, 101, 103, 104, 105], + }) + con.execute("CREATE TABLE IF NOT EXISTS df_store_products AS SELECT * FROM store_products_df") + + products_df = pd.DataFrame({ + "product_id": [100, 101, 103, 104], + "product_name": ["Cherry coke", "Laptop", "Ipad", "Livre"], + "product_price": [3, 800, 400, 30], + }) + con.execute("CREATE TABLE IF NOT EXISTS df_products AS SELECT * FROM products_df") + + +def create_self_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: + employees_df = pd.DataFrame({ + "employee_id": [11, 12, 13, 14, 15], + "employee_name": ["Sophie", "Sylvie", "Daniel", "Kaouter", "David"], + "manager_id": [13, None, 12, 13, 11], + }) + con.execute("CREATE TABLE IF NOT EXISTS employees AS SELECT * FROM employees_df") + + sales_df = pd.DataFrame({ + "order_id": list(range(1110, 1198)), + "customer_id": random.choices([11, 12, 13, 14, 15, 11, 12, 13, 14], k=88), + }) + sales_df["date"] = [d // 3 + 1 for d in range(1, 89)] + con.execute("CREATE TABLE IF NOT EXISTS sales AS SELECT * FROM sales_df") + + +def create_group_by_exercises(con: duckdb.DuckDBPyConnection) -> None: + ventes_immo_df = pd.DataFrame([ + [0, "vieux_lille", 460000], [1, "vieux_lille", 430000], [2, "vieux_lille", 450000], + [3, float("nan"), 470000], [4, "vieux_lille", 440000], [10, "gambetta", 336000], + [11, float("nan"), 333000], [12, "gambetta", 335000], [13, "gambetta", 337000], + [14, "gambetta", 334000], [20, "centre", 356000], [21, "centre", 353000], + [22, float("nan"), 355000], [23, "centre", 357000], [24, "centre", 354000], + [30, "wazemmes", 260000], [31, float("nan"), 230000], [32, "wazemmes", 250000], + [33, "wazemmes", 270000], [34, "wazemmes", 240000], + ], columns=["flat_id", "neighborhood", "price"]) + con.execute("CREATE TABLE IF NOT EXISTS ventes_immo AS SELECT * FROM ventes_immo_df") + + clients = ["Oussama", "Julie", "Chris", "Tom", "Jean-Nicolas", + "Aline", "Ben", "Toufik", "Sylvie", "David"] + ventes_df = pd.DataFrame( + [110, 49, 65, 23, 24, 3.99, 29, 48.77, 44, 10, 60, 12, 62, 19, 75] * 2, + columns=["montant"] + ) + ventes_df["client"] = clients * 3 + con.execute("CREATE TABLE IF NOT EXISTS ventes AS SELECT * FROM ventes_df") + + +def create_case_when_exercises(con: duckdb.DuckDBPyConnection) -> None: + salaires_df = pd.DataFrame({ + "name": ["Toufik", "Jean-Nicolas", "Daniel", "Kaouter", "Sylvie", "Sebastien", + "Diane", "Romain", "François", "Anna", "Zeinaba", "Gregory", + "Karima", "Arthur", "Benjamin"], + "wage": [60000, 75000, 55000, 80000, 70000, 90000, 65000, 72000, 68000, + 85000, 100000, 120000, 95000, 83000, 110000], + "department": ["IT", "HR", "SALES", "IT", "IT", "HR", "SALES", "IT", "HR", + "SALES", "IT", "IT", "HR", "SALES", "CEO"], + }) + con.execute("CREATE TABLE IF NOT EXISTS salaires AS SELECT * FROM salaires_df") + + discount_df = pd.DataFrame({ + "order_id": [1, 2, 3, 4, 5, 6], + "product_id": [101, 102, 101, 103, 102, 103], + "quantity": [5, 3, 2, 4, 6, 2], + "price_per_unit": [10.0, 25.0, 10.0, 8.0, 25.0, 8.0], + "discount_code": [None, "DISCOUNT10", "DISCOUNT20", None, None, "UNKNOWN"], + }) + con.execute("CREATE TABLE IF NOT EXISTS discount AS SELECT * FROM discount_df") + + +def create_grouping_set_exercises(con: duckdb.DuckDBPyConnection) -> None: + redbull_df = pd.DataFrame({ + "store_id": ["Armentieres"] * 4 + ["Lille"] * 4 + ["Douai"] * 4, + "product_name": ["redbull", "chips", "wine", "redbull", "redbull", "chips", + "wine", "icecream", "redbull", "chips", "wine", "icecream"], + "amount": [45, 60, 60, 45, 100, 140, 190, 170, 55, 70, 20, 45], + }) + con.execute("CREATE TABLE IF NOT EXISTS redbull AS SELECT * FROM redbull_df") + + datapop_df = pd.DataFrame({ + "year": [2016, 2017, 2018, 2019, 2020] * 3, + "region": ["IDF"] * 5 + ["HDF"] * 5 + ["PACA"] * 5, + "population": [1010000, 1020000, 1030000, 1040000, 1000000, + 910000, 920000, 930000, 940000, 900000, + 810000, 820000, 830000, 840000, 950000], + }) + con.execute("CREATE TABLE IF NOT EXISTS datapop AS SELECT * FROM datapop_df") + + +# ------------------------------------------------------------ +# POINT D'ENTRÉE +# ------------------------------------------------------------ + +def init_db() -> None: + """Crée toutes les tables si elles n'existent pas encore.""" + + Path(DB_PATH).parent.mkdir(parents=True, exist_ok=True) + + con = duckdb.connect(DB_PATH) + create_cross_joins_exercises(con) + create_inner_joins_exercises(con) + create_left_joins_exercises(con) + create_full_outer_joins_exercises(con) + create_self_joins_exercises(con) + create_group_by_exercises(con) + create_case_when_exercises(con) + create_grouping_set_exercises(con) + con.close() + + +if __name__ == "__main__": + init_db() + print("Base de données initialisée.") + con = duckdb.connect(DB_PATH) + + with con: + tables = con.execute("SHOW TABLES").fetchall() + print("Tables créées :", tables) From 2387f1206218e69e0a4c9d826f40dd9a7d6ab1b3 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:42:54 +0000 Subject: [PATCH 07/18] delete: db.py is the way now --- init_db.py | 352 ----------------------------------------------------- 1 file changed, 352 deletions(-) delete mode 100644 init_db.py diff --git a/init_db.py b/init_db.py deleted file mode 100644 index 0b85016..0000000 --- a/init_db.py +++ /dev/null @@ -1,352 +0,0 @@ -""" -Ce module contient les fonctions permettant d'initialiser la base de données de l'application. -""" - -import io -import random -import numpy as np -import pandas as pd -import duckdb - - -def create_memory_state_table(con): - """ - Créer la table vide des exercices - """ - con.execute( - """ - CREATE TABLE IF NOT EXISTS memory_state ( - user_id TEXT, - theme TEXT, - exercise_name TEXT, - tables TEXT[], - last_reviewed DATE, - answer TEXT, - PRIMARY KEY (user_id, exercise_name) - ) - """ - ) - - -def create_cross_joins_exercises(con): - beverages_csv = "beverage,price\norange juice,2.5\nExpresso,2\nTea,3\n" - beverages_df = pd.read_csv(io.StringIO(beverages_csv)) - con.execute("CREATE TABLE IF NOT EXISTS beverages AS SELECT * FROM beverages_df") - - food_items_csv = "food_item,food_price\ncookie juice,2.5\nchocolatine,2\nmuffin,3\n" - food_items_df = pd.read_csv(io.StringIO(food_items_csv)) - con.execute("CREATE TABLE IF NOT EXISTS food_items AS SELECT * FROM food_items_df") - - sizes_csv = "size\nXS\nM\nL\nXL\n" - sizes_df = pd.read_csv(io.StringIO(sizes_csv)) - con.execute("CREATE TABLE IF NOT EXISTS sizes AS SELECT * FROM sizes_df") - - trademarks_csv = "trademark\nNike\nAsphalte\nAbercrombie\nLewis\n" - trademarks_df = pd.read_csv(io.StringIO(trademarks_csv)) - con.execute("CREATE TABLE IF NOT EXISTS trademarks AS SELECT * FROM trademarks_df") - - hours_csv = "hour\n08\n09\n10\n11\n12\n" - hours_df = pd.read_csv(io.StringIO(hours_csv)) - con.execute("CREATE TABLE IF NOT EXISTS hours AS SELECT * FROM hours_df") - - quarters_csv = "quarter\n00\n15\n30\n45\n" - quarters_df = pd.read_csv(io.StringIO(quarters_csv)) - con.execute("CREATE TABLE IF NOT EXISTS quarters AS SELECT * FROM quarters_df") - - -def create_inner_joins_exercises(con): - salaries_csv = "salary,employee_id\n2000,1\n2500,2\n2200,3\n" - salaries_df = pd.read_csv(io.StringIO(salaries_csv)) - con.execute("CREATE TABLE IF NOT EXISTS salaries AS SELECT * FROM salaries_df") - - seniorities_csv = "employee_id,seniority\n1,2ans\n2,4ans\n" - seniorities_df = pd.read_csv(io.StringIO(seniorities_csv)) - con.execute( - "CREATE TABLE IF NOT EXISTS seniorities AS SELECT * FROM seniorities_df" - ) - - -def create_left_joins_exercises(con): - orders_data = { - "order_id": [1, 2, 3, 4, 5], - "customer_id": [101, 102, 103, 104, 105], - } - df_orders = pd.DataFrame(orders_data) - con.execute("CREATE TABLE IF NOT EXISTS orders AS SELECT * FROM df_orders") - - customers_data = { - "customer_id": [101, 102, 103, 104, 105, 106], - "customer_name": [ - "Toufik", - "Daniel", - "Tancrède", - "Kaouter", - "Jean-Nicolas", - "David", - ], - } - df_customers = pd.DataFrame(customers_data) - con.execute("CREATE TABLE IF NOT EXISTS customers AS SELECT * FROM df_customers") - - products_data = { - "product_id": [101, 103, 104, 105], - "product_name": ["Laptop", "Ipad", "Livre", "Petitos"], - "product_price": [800, 400, 30, 2], - } - df_products = pd.DataFrame(products_data) - con.execute("CREATE TABLE IF NOT EXISTS products AS SELECT * FROM df_products") - - order_details_data = { - "order_id": [1, 2, 3, 4, 5], - "product_id": [102, 104, 101, 103, 105], - "quantity": [2, 1, 3, 2, 1], - } - df_order_details = pd.DataFrame(order_details_data) - con.execute( - "CREATE TABLE IF NOT EXISTS order_details AS SELECT * FROM df_order_details" - ) - - -def create_full_outer_joins_exercises(con): - customers_data = { - "customer_id": [11, 12, 13, 14, 15], - "customer_name": ["Zeinaba", "Tancrède", "Israel", "Kaouter", "Alan"], - } - customers_df = pd.DataFrame(customers_data) - con.execute("CREATE TABLE IF NOT EXISTS df_customers AS SELECT * FROM customers_df") - - stores_data = {"store_id": [1, 2, 3, 4], "customer_id": [11, 12, 13, 15]} - stores_df = pd.DataFrame(stores_data) - con.execute("CREATE TABLE IF NOT EXISTS df_stores AS SELECT * FROM stores_df") - - store_products_data = { - "store_id": [1, 1, 1, 2, 2, 3, 4], - "product_id": [101, 103, 105, 101, 103, 104, 105], - } - store_products_df = pd.DataFrame(store_products_data) - con.execute( - "CREATE TABLE IF NOT EXISTS df_store_products AS SELECT * FROM store_products_df" - ) - - products_data = { - "product_id": [100, 101, 103, 104], - "product_name": ["Cherry coke", "Laptop", "Ipad", "Livre"], - "product_price": [3, 800, 400, 30], - } - products_df = pd.DataFrame(products_data) - con.execute("CREATE TABLE IF NOT EXISTS df_products AS SELECT * FROM products_df") - - -def create_self_joins_exercises(con): - employees_data = { - "employee_id": [11, 12, 13, 14, 15], - "employee_name": ["Sophie", "Sylvie", "Daniel", "Kaouter", "David"], - "manager_id": [13, None, 12, 13, 11], - } - employees_df = pd.DataFrame(employees_data) - con.execute("CREATE TABLE IF NOT EXISTS employees AS SELECT * FROM employees_df") - - sales_data = { - "order_id": list(range(1110, 1198)), - "customer_id": random.choices([11, 12, 13, 14, 15, 11, 12, 13, 14], k=88), - } - sales_df = pd.DataFrame(sales_data) - sales_df["date"] = [d // 3 + 1 for d in range(1, 89)] - con.execute("CREATE TABLE IF NOT EXISTS sales AS SELECT * FROM sales_df") - - -def create_group_by_exercises(con): - ventes_immo_data = [ - [0, "vieux_lille", 460000], - [1, "vieux_lille", 430000], - [2, "vieux_lille", 450000], - [3, np.NaN, 470000], - [4, "vieux_lille", 440000], - [10, "gambetta", 336000], - [11, np.NaN, 333000], - [12, "gambetta", 335000], - [13, "gambetta", 337000], - [14, "gambetta", 334000], - [20, "centre", 356000], - [21, "centre", 353000], - [22, np.NaN, 355000], - [23, "centre", 357000], - [24, "centre", 354000], - [30, "wazemmes", 260000], - [31, np.NaN, 230000], - [32, "wazemmes", 250000], - [33, "wazemmes", 270000], - [34, "wazemmes", 240000], - ] - ventes_immo_df = pd.DataFrame( - ventes_immo_data, columns=["flat_id", "neighborhood", "price"] - ) - con.execute( - "CREATE TABLE IF NOT EXISTS ventes_immo AS SELECT * FROM ventes_immo_df" - ) - - clients = [ - "Oussama", - "Julie", - "Chris", - "Tom", - "Jean-Nicolas", - "Aline", - "Ben", - "Toufik", - "Sylvie", - "David", - ] - ventes_data = [110, 49, 65, 23, 24, 3.99, 29, 48.77, 44, 10, 60, 12, 62, 19, 75] * 2 - ventes_df = pd.DataFrame(ventes_data, columns=["montant"]) - ventes_df["client"] = clients * 3 - con.execute("CREATE TABLE IF NOT EXISTS ventes AS SELECT * FROM ventes_df") - - -def create_case_when_exercises(con): - salaires_data = { - "name": [ - "Toufik", - "Jean-Nicolas", - "Daniel", - "Kaouter", - "Sylvie", - "Sebastien", - "Diane", - "Romain", - "François", - "Anna", - "Zeinaba", - "Gregory", - "Karima", - "Arthur", - "Benjamin", - ], - "wage": [ - 60000, - 75000, - 55000, - 80000, - 70000, - 90000, - 65000, - 72000, - 68000, - 85000, - 100000, - 120000, - 95000, - 83000, - 110000, - ], - "department": [ - "IT", - "HR", - "SALES", - "IT", - "IT", - "HR", - "SALES", - "IT", - "HR", - "SALES", - "IT", - "IT", - "HR", - "SALES", - "CEO", - ], - } - salaires_df = pd.DataFrame(salaires_data) - con.execute("CREATE TABLE IF NOT EXISTS salaires AS SELECT * FROM salaires_df") - - discount_data = { - "order_id": [1, 2, 3, 4, 5, 6], - "product_id": [101, 102, 101, 103, 102, 103], - "quantity": [5, 3, 2, 4, 6, 2], - "price_per_unit": [10.0, 25.0, 10.0, 8.0, 25.0, 8.0], - "discount_code": [None, "DISCOUNT10", "DISCOUNT20", None, None, "UNKNOWN"], - } - discount_df = pd.DataFrame(discount_data) - con.execute("CREATE TABLE IF NOT EXISTS discount AS SELECT * FROM discount_df") - - -def create_grouping_set_exercises(con): - redbull_data = { - "store_id": ["Armentieres"] * 4 + ["Lille"] * 4 + ["Douai"] * 4, - "product_name": [ - "redbull", - "chips", - "wine", - "redbull", - "redbull", - "chips", - "wine", - "icecream", - "redbull", - "chips", - "wine", - "icecream", - ], - "amount": [45, 60, 60, 45, 100, 140, 190, 170, 55, 70, 20, 45], - } - redbull_df = pd.DataFrame(redbull_data) - con.execute("CREATE TABLE IF NOT EXISTS redbull AS SELECT * FROM redbull_df") - - datapop_data = { - "year": [2016, 2017, 2018, 2019, 2020] * 3, - "region": ["IDF"] * 5 + ["HDF"] * 5 + ["PACA"] * 5, - "population": [ - 1010000, - 1020000, - 1030000, - 1040000, - 1000000, - 910000, - 920000, - 930000, - 940000, - 900000, - 810000, - 820000, - 830000, - 840000, - 950000, - ], - } - datapop_df = pd.DataFrame(datapop_data) - con.execute("CREATE TABLE IF NOT EXISTS datapop AS SELECT * FROM datapop_df") - - -def create_users_table(con): - con.execute("CREATE SEQUENCE IF NOT EXISTS user_id_seq") - con.execute( - """ - CREATE TABLE IF NOT EXISTS users ( - id INT DEFAULT nextval('user_id_seq') PRIMARY KEY, - username TEXT UNIQUE NOT NULL, - password TEXT NOT NULL - ) - """ - ) - - -def main(): - con = duckdb.connect(database="data/exercises_sql_tables.duckdb", read_only=False) - - create_memory_state_table(con) - create_cross_joins_exercises(con) - create_inner_joins_exercises(con) - create_left_joins_exercises(con) - create_full_outer_joins_exercises(con) - create_self_joins_exercises(con) - create_group_by_exercises(con) - create_case_when_exercises(con) - create_grouping_set_exercises(con) - create_users_table(con) - - con.close() - - -if __name__ == "__main__": - main() From ec5211680c9841dc397e71fc52ca7a818f8b857f Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:47:32 +0000 Subject: [PATCH 08/18] REFACTOR: delete functions.py, introduce utils.py Simplify functions, delete user account features, add a random queue in session state --- functions.py | 416 --------------------------------------------------- utils.py | 197 ++++++++++++++++++++++++ 2 files changed, 197 insertions(+), 416 deletions(-) delete mode 100644 functions.py create mode 100644 utils.py diff --git a/functions.py b/functions.py deleted file mode 100644 index 71d57ee..0000000 --- a/functions.py +++ /dev/null @@ -1,416 +0,0 @@ -""" -Ce module contient toutes les fonctions nécessaires pour faire fonctionner l'application -""" - -import pandas as pd -import streamlit as st -import duckdb -import bcrypt -from pathlib import Path - - -# ------------------------------------------------------------ -# FUNCTIONS -# ------------------------------------------------------------ - - -def create_exercises_table(con, user_id): - """ - Initialise la table des exercices pour user - """ - data = { - "user_id": [user_id] * 17, - "theme": [ - "cross_joins", - "cross_joins", - "cross_joins", - "inner_joins", - "left_joins", - "left_joins", - "full_outer_joins", - "full_outer_joins", - "self_joins", - "self_joins", - "group_by", - "group_by", - "case_when", - "case_when", - "case_when", - "grouping_set", - "grouping_set", - # "grouping_set", - ], - "exercise_name": [ - "cross_joins_1", - "cross_joins_2", - "cross_joins_3", - "inner_joins_1", - "left_joins_1", - "left_joins_2", - "full_outer_joins_1", - "full_outer_joins_2", - "self_joins_1", - "self_joins_2", - "group_by_1", - "group_by_2", - "case_when_1", - "case_when_2", - "case_when_3", - "grouping_set_1", - "grouping_set_2", - # "grouping_set_3", - ], - "tables": [ - ["beverages", "food_items"], - ["sizes", "trademarks"], - ["hours", "quarters"], - ["salaries", "seniorities"], - ["products", "order_details"], - ["orders", "customers", "products", "order_details"], - ["df_store_products", "df_products"], - ["df_customers", "df_stores", "df_store_products", "df_products"], - ["employees"], - ["sales"], - ["ventes_immo"], - ["ventes"], - ["salaires"], - ["discount"], - ["salaires"], - ["redbull"], - ["datapop"], - # ["redbull"], - ], - "last_reviewed": ["1970-01-01"] * 17, - "answer": [ - "cross_joins_1.sql", - "cross_joins_2.sql", - "cross_joins_3.sql", - "inner_joins_1.sql", - "left_joins_1.sql", - "left_joins_2.sql", - "full_outer_joins_1.sql", - "full_outer_joins_2.sql", - "self_joins_1.sql", - "self_joins_2.sql", - "group_by_1.sql", - "group_by_2.sql", - "case_when_1.sql", - "case_when_2.sql", - "case_when_3.sql", - "grouping_set_1.sql", - "grouping_set_2.sql", - # "grouping_set_3.sql", - ], - } - memory_state_df = pd.DataFrame(data) - con.execute("INSERT INTO memory_state SELECT * FROM memory_state_df") - - -def hash_password(password: str) -> bytes: - """Hache un mot de passe en utilisant bcrypt. - - Cette fonction prend un mot de passe en entrée, l'encode en bytes, - génère un sel aléatoire, et retourne le mdp haché. - - Args: - password (str): le mot de passe à hacher - - Returns: - bytes : le mdp haché sous forme de bytes - """ - - return bcrypt.hashpw(password.encode(), bcrypt.gensalt()) - - -def check_password(hashed_password: bytes, user_password: str) -> bool: - """Vérifie que le mot de passe est valide. - - Cette fonction prend un mot de passe haché (généré par hash_password) et un mot de passe - en clair, puis vérifie que le hash du mot de passe en clair correspond au mot de passe haché. - - Args: - hashed_password (bytes): Le mot de passe haché (résultat de hash_password) - user_password (str): Le mot de passe en clair à vérifier - - Returns: - bool: True si le mot de passe est correct, False sinon - """ - return bcrypt.checkpw(user_password.encode(), hashed_password) - - -def signup_user(username: str, password: str) -> bool: - """Inscrit un nouvel utilisateur dans la base de données. - - Cette fonction prend un nom d'utilisateur et un mot de passe, vérifie si l'utilisateur - existe déjà, et si non, hache le mot de passe et insère ces informations dans la table 'users'. - - Args: - username (str): Le nom d'utilisateur du nouvel utilisateur. - password (str): Le mot de passe en clair du nouvel utilisateur. - - Returns: - bool: True si l'inscription a réussi, False si l'utilisateur existe déjà. - - Raises: - duckdb.Error: Si une erreur inattendue se produit lors de l'interaction avec la base de données. - """ - try: - with duckdb.connect("data/exercises_sql_tables.duckdb") as con: - # Vérifier si l'utilisateur existe déjà - result = con.execute( - "SELECT COUNT(*) FROM users WHERE username = ?", [username] - ).fetchone() - if result[0] > 0: - print(f"L'utilisateur '{username}' existe déjà.") - return False - - # Si l'utilisateur n'existe pas, procéder à l'inscription - hashed_password = hash_password(password) - con.execute( - """ - INSERT INTO users (username, password) VALUES (?, ?) - """, - (username, hashed_password), - ) - - create_exercises_table(con, username) - print(f"L'utilisateur '{username}' a été inscrit avec succès.") - return True - except duckdb.Error as e: - print(f"Une erreur s'est produite lors de l'inscription : {e}") - raise - - -def login_user(username: str, password: str) -> bool: - """Vérifie les informations de connexion de l'utilisateur. - - Cette fonction prend un nom d'utilisateur et un mot de passe, vérifie leur validité - dans la base de données, et retourne True si les informations sont correctes. - - Args: - username (str): Le nom d'utilisateur à vérifier. - password (str): Le mot de passe à vérifier. - - Returns: - bool: True si les informations de connexion sont valides, False sinon. - - Raises: - duckdb.Error: Si une erreur se produit lors de l'interaction avec la base de données. - """ - try: - with duckdb.connect(database="data/exercises_sql_tables.duckdb") as con: - result = con.execute( - """ - SELECT password FROM users WHERE username = ? - """, - (username,), - ).fetchone() - - if result: - hashed_password = result[0].encode() - return check_password(hashed_password, password) - return False - except duckdb.Error as e: - print(f"Erreur lors de la vérification des informations de connexion : {e}") - raise - - -def user_auth() -> None: - """Affiche le formulaire d'inscription ou de connexion et gère l'authentification de l'utilisateur.""" - - # Vérifier si l'utilisateur est déjà connecté - if st.session_state.get("logged_in", False): - # st.write(f"Vous êtes connecté en tant que {st.session_state['username']}") - # if st.button("Déconnexion"): - # st.session_state["logged_in"] = False - # st.session_state["username"] = None - # st.rerun() - return - - # Initialiser l'état d'inscription s'il n'existe pas - if "signup" not in st.session_state: - st.session_state["signup"] = False - - if st.session_state["signup"]: - st.title("Inscription") - signup_username = st.text_input("Nom d'utilisateur", key="signup_username") - signup_password = st.text_input( - "Mot de passe", type="password", key="signup_password" - ) - signup_password_confirm = st.text_input( - "Confirmer le mot de passe", type="password", key="signup_password_confirm" - ) - - if st.button("S'inscrire"): - if not signup_username or not signup_password: - st.error("Veuillez remplir tous les champs.") - elif signup_password != signup_password_confirm: - st.error("Les mots de passe ne correspondent pas.") - elif signup_user(signup_username, signup_password): - st.success( - "Inscription réussie. Vous pouvez maintenant vous connecter." - ) - st.session_state["signup"] = False - else: - st.error( - "L'utilisateur existe déjà. Essayez un autre nom d'utilisateur." - ) - - if st.button("Retour à la connexion"): - st.session_state["signup"] = False - else: - st.title("Connexion") - login_username = st.text_input("Nom d'utilisateur", key="login_username") - login_password = st.text_input( - "Mot de passe", type="password", key="login_password" - ) - - if st.button("Se connecter"): - if not login_username or not login_password: - st.error("Veuillez remplir tous les champs.") - elif login_user(login_username, login_password): - st.success("Connexion réussie") - st.session_state["logged_in"] = True - st.session_state["username"] = login_username - st.rerun() - else: - st.error("Nom d'utilisateur ou mot de passe incorrect") - - if st.button("S'inscrire"): - st.session_state["signup"] = True - - -def query_memory_df(con, user_id: str) -> pd.DataFrame: - with duckdb.connect(database="data/exercises_sql_tables.duckdb") as con: - memory_df = ( - con.execute("SELECT * FROM memory_state WHERE user_id = ?", [user_id]) - .df() - .sort_values("last_reviewed") - .reset_index(drop=True) - ) - - return memory_df - - -def check_users_solution(con, solution_df: pd.DataFrame, user_query: str) -> bool: - """ - Checks that user SQL query is correct by : - 1: checking the columns - 2: checking the values - :param con: connection duckdb - :param solution_df: THE TRUTH - :param user_query: string containing the query typed by the user - - returns : bool indicating wether the user's query matches the solution - """ - try: - # Exécute la requête utilisateur et récupère les résultats dans un DataFrame - result = con.execute(user_query).df() - st.dataframe(result) # Affiche le résultat pour l'utilisateur - - # Vérifie si les colonnes correspondent et dans le bon ordre - if not result.columns.equals(solution_df.columns): - st.write( - "Les colonnes de votre requête ne correspondent pas exactement à celles de la solution." - ) - return False - - # Vérifie si le nombre de lignes correspond - if result.shape[0] != solution_df.shape[0]: - st.write( - f"Votre requête retourne un nombre de lignes différent de la solution attendue ({result.shape[0]} vs {solution_df.shape[0]})." - ) - return False - - # Compare les valeurs des deux DataFrames - if not result.equals(solution_df): - differences = result.compare(solution_df) - st.write( - f"Il y a des différences dans les valeurs de votre requête : {differences}" - ) - return False - - st.balloons() # Affiche des ballons si la requête est correcte 8) - return True - - except Exception as e: - # Capture et affiche toute exception survenue lors de l'exécution de la requête ou de la comparaison - st.write(f"Erreur lors de l'exécution ou de la comparaison des requêtes : {e}") - return False - - -def get_selector_themes(memory_df: pd.DataFrame) -> str: - """ - Function to get themes selected by the user - - Args: - memory_df (pd.DataFrame): the memory state df from the database - - Returns: - str: the theme selected - """ - theme = st.selectbox( - "Sélectionner un thème", - options=memory_df["theme"].unique(), - index=0, - key="theme_selectbox", - ) - - return theme - - -def get_selector_exercises(memory_df: pd.DataFrame, theme: str) -> str: - filtered_exercises = memory_df[memory_df["theme"] == theme][ - "exercise_name" - ].to_list() - exercises_lst = st.selectbox( - "Sélectionner un exercice", - options=filtered_exercises, - index=0, - key="exercises_selectbox", - ) - - return exercises_lst - - -def get_selected_exercise(con, theme: str, exercises_lst: str, current_user: str): - exercise = con.execute( - f"SELECT * FROM memory_state where theme = '{theme}' and exercise_name = '{exercises_lst}' and user_id = '{current_user}'" - ).df() - # st.write('Vous avez sélectionné', exercise) - if not exercise.empty: - st.dataframe(exercise.iloc[:, 1:-2]) # On affiche pas la colonne réponse - elif exercise.empty: - st.write("Il faut sélectionner un thème") - else: - st.write("Pas d'exercice disponibles pour le thème sélectionné") - - # Récupérer la solution de l'exercice - answer_str = exercise.loc[0, "answer"] - try: - answer_file = Path(f"Exercices/answers/{theme}/{answer_str}") - with open(answer_file, "r") as f: - answer = f.read() - solution_df = con.execute(answer).df() - - except FileNotFoundError: - st.write("Fichier de réponse non trouvé") - return exercise, answer_str, answer, solution_df - - -def get_questions(theme: str, answer_str: str) -> None: - """Récupère les questions depuis un folder - - Args: - theme (str): thème sélectionné par l'utilisateur - answer_str (str): nom exercice sélectionné par l'utilisateur - """ - # print(f"{answer_str[:-4]}.txt") - try: - question_file = Path(f"Exercices/questions/{theme}/{answer_str[:-4]}.md") - with question_file.open("r") as f: - - question: str = f.read() - st.markdown(question) - except FileNotFoundError: - st.write("Fichier question absent") diff --git a/utils.py b/utils.py new file mode 100644 index 0000000..de1b93a --- /dev/null +++ b/utils.py @@ -0,0 +1,197 @@ +""" +utils.py — Fonctions utilitaires de SQLingo. + +Ce module contient : + - Le chargement des exercices depuis exercises.yaml + - L'initialisation de la session SRS-like + - La vérification des requêtes utilisateur + - L'affichage des questions et tables +""" + +import random + +import pandas as pd +import streamlit as st +import yaml + +from config import ( + EXERCISES_PATH, + ANSWERS_PATH, + QUESTIONS_PATH, + SRS_INCORRECT_DELAY, +) + + +# ------------------------------------------------------------ +# CHARGEMENT DES EXERCICES +# ------------------------------------------------------------ + +def load_exercises() -> list[dict]: + """Charge la liste des exercices depuis exercises.yaml. + + Returns: + list[dict]: liste de dicts avec les clés theme, name, tables, answer. + """ + with EXERCISES_PATH.open("r") as f: + return yaml.safe_load(f) + + +# ------------------------------------------------------------ +# SESSION SRS-LIKE +# ------------------------------------------------------------ + +def init_session(themes: list[str] | None = None) -> None: + """Initialise la session Streamlit avec une queue d'exercices mélangée. + + Args: + themes: si fourni, filtre les exercices sur ces thèmes uniquement. + si None, tous les exercices sont inclus. + """ + exercises = load_exercises() + + if themes: + exercises = [e for e in exercises if e["theme"] in themes] + + random.shuffle(exercises) + + st.session_state["queue"] = exercises + st.session_state["score"] = {"correct": 0, "incorrect": 0} + st.session_state["history"] = [] # liste de (exercise, "correct" | "incorrect") + st.session_state["session_started"] = True + + +def get_current_exercise() -> dict | None: + """Retourne l'exercice en tête de queue, ou None si la session est terminée.""" + queue = st.session_state.get("queue", []) + return queue[0] if queue else None + + +def record_result(exercise: dict, correct: bool) -> None: + """Enregistre le résultat et repositionne l'exercice dans la queue. + + - Bonne réponse → exercice retiré de la queue (ne revient pas) + - Mauvaise réponse → exercice réinséré à la position SRS_INCORRECT_DELAY + """ + queue = st.session_state["queue"] + result = "correct" if correct else "incorrect" + + st.session_state["history"].append((exercise, result)) + + if correct: + st.session_state["score"]["correct"] += 1 + queue.pop(0) + else: + st.session_state["score"]["incorrect"] += 1 + queue.pop(0) + insert_at = min(SRS_INCORRECT_DELAY, len(queue)) + queue.insert(insert_at, exercise) + + +# ------------------------------------------------------------ +# VÉRIFICATION DE LA REQUÊTE UTILISATEUR +# ------------------------------------------------------------ + +def check_users_solution(con, solution_df: pd.DataFrame, user_query: str) -> bool: + """Vérifie que la requête SQL de l'utilisateur produit le bon résultat. + + La comparaison est insensible à l'ordre des lignes : les deux DataFrames + sont triés avant d'être comparés. + + Args: + con: connexion DuckDB active + solution_df: DataFrame de référence (la solution attendue) + user_query: requête SQL saisie par l'utilisateur + + Returns: + bool: True si la requête est correcte, False sinon. + """ + try: + result = con.execute(user_query).df() + st.dataframe(result) + + # Colonnes + if list(result.columns) != list(solution_df.columns): + st.warning( + f"Colonnes incorrectes.\n\n" + f"**Attendu :** {list(solution_df.columns)}\n\n" + f"**Obtenu :** {list(result.columns)}" + ) + return False + + # Nombre de lignes + if len(result) != len(solution_df): + st.warning( + f"Nombre de lignes incorrect : " + f"ta requête retourne **{len(result)}** lignes, " + f"la solution en attend **{len(solution_df)}**." + ) + return False + + # Valeurs (order-insensitive) + result_sorted = result.sort_values(by=list(result.columns)).reset_index(drop=True) + solution_sorted = solution_df.sort_values(by=list(solution_df.columns)).reset_index(drop=True) + + if not result_sorted.equals(solution_sorted): + st.warning("Les colonnes et le nombre de lignes sont bons, mais certaines valeurs différent.") + return False + + st.balloons() + return True + + except Exception as e: + st.error(f"Erreur lors de l'exécution de ta requête : {e}") + return False + + +# ------------------------------------------------------------ +# AFFICHAGE +# ------------------------------------------------------------ + +def show_question(exercise: dict) -> None: + """Affiche l'énoncé markdown de l'exercice courant. + + Args: + exercise: dict d'un exercice chargé depuis exercises.yaml + """ + theme = exercise["theme"] + answer = exercise["answer"] + question_file = QUESTIONS_PATH / theme / f"{answer[:-4]}.md" + + try: + st.markdown(question_file.read_text()) + except FileNotFoundError: + st.warning(f"Énoncé introuvable : `{question_file}`") + + +def show_tables(con, exercise: dict) -> None: + """Affiche un aperçu des tables nécessaires à l'exercice. + + Args: + con: connexion DuckDB active + exercise: dict d'un exercice chargé depuis exercises.yaml + """ + for table in exercise["tables"]: + st.markdown(f"**`{table}`**") + try: + df = con.execute(f"SELECT * FROM {table} LIMIT 5").df() + st.dataframe(df) + except Exception as e: + st.error(f"Impossible de charger la table `{table}` : {e}") + + +def load_solution(con, exercise: dict) -> pd.DataFrame: + """Charge et exécute le fichier SQL solution, retourne le DataFrame résultat. + + Args: + con: connexion DuckDB active + exercise: dict d'un exercice chargé depuis exercises.yaml + + Returns: + pd.DataFrame: résultat de la requête solution. + + Raises: + FileNotFoundError: si le fichier SQL solution est absent. + """ + answer_file = ANSWERS_PATH / exercise["theme"] / exercise["answer"] + sql = answer_file.read_text() + return con.execute(sql).df() From 2bc2a2c8b9fa21dde4dfd11fdf9f82827ccf4dd5 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:55:25 +0000 Subject: [PATCH 09/18] style: ruff format + ignore F841 --- db.py | 337 ++++++++++++++++++++++++++++++++++++++++++---------------- 1 file changed, 243 insertions(+), 94 deletions(-) diff --git a/db.py b/db.py index 0c6709d..0ce0804 100644 --- a/db.py +++ b/db.py @@ -8,7 +8,6 @@ import io import random -import numpy as np import pandas as pd import duckdb import streamlit as st @@ -21,6 +20,7 @@ # CONNEXION PARTAGÉE # ------------------------------------------------------------ + @st.cache_resource def get_connection() -> duckdb.DuckDBPyConnection: """Retourne une connexion DuckDB partagée pour toute la session Streamlit.""" @@ -31,159 +31,307 @@ def get_connection() -> duckdb.DuckDBPyConnection: # CRÉATION DES TABLES D'EXERCICES # ------------------------------------------------------------ + def create_cross_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: - beverages_df = pd.read_csv(io.StringIO("beverage,price\norange juice,2.5\nExpresso,2\nTea,3\n")) + beverages_df = pd.read_csv( # noqa: F841 + io.StringIO("beverage,price\norange juice,2.5\nExpresso,2\nTea,3\n") + ) con.execute("CREATE TABLE IF NOT EXISTS beverages AS SELECT * FROM beverages_df") - food_items_df = pd.read_csv(io.StringIO("food_item,food_price\ncookie,2.5\ncholatine,2\nmuffin,3\n")) + food_items_df = pd.read_csv( # noqa: F841 + io.StringIO("food_item,food_price\ncookie,2.5\ncholatine,2\nmuffin,3\n") + ) con.execute("CREATE TABLE IF NOT EXISTS food_items AS SELECT * FROM food_items_df") - sizes_df = pd.read_csv(io.StringIO("size\nXS\nM\nL\nXL\n")) + sizes_df = pd.read_csv(io.StringIO("size\nXS\nM\nL\nXL\n")) # noqa: F841 con.execute("CREATE TABLE IF NOT EXISTS sizes AS SELECT * FROM sizes_df") - trademarks_df = pd.read_csv(io.StringIO("trademark\nNike\nAsphalte\nAbercrombie\nLewis\n")) + trademarks_df = pd.read_csv( # noqa: F841 + io.StringIO("trademark\nNike\nAsphalte\nAbercrombie\nLewis\n") + ) con.execute("CREATE TABLE IF NOT EXISTS trademarks AS SELECT * FROM trademarks_df") - hours_df = pd.read_csv(io.StringIO("hour\n08\n09\n10\n11\n12\n")) + hours_df = pd.read_csv(io.StringIO("hour\n08\n09\n10\n11\n12\n")) # noqa: F841 con.execute("CREATE TABLE IF NOT EXISTS hours AS SELECT * FROM hours_df") - quarters_df = pd.read_csv(io.StringIO("quarter\n00\n15\n30\n45\n")) + quarters_df = pd.read_csv(io.StringIO("quarter\n00\n15\n30\n45\n")) # noqa: F841 con.execute("CREATE TABLE IF NOT EXISTS quarters AS SELECT * FROM quarters_df") def create_inner_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: - salaries_df = pd.read_csv(io.StringIO("salary,employee_id\n2000,1\n2500,2\n2200,3\n")) + salaries_df = pd.read_csv( # noqa: F841 + io.StringIO("salary,employee_id\n2000,1\n2500,2\n2200,3\n") + ) con.execute("CREATE TABLE IF NOT EXISTS salaries AS SELECT * FROM salaries_df") - seniorities_df = pd.read_csv(io.StringIO("employee_id,seniority\n1,2ans\n2,4ans\n")) - con.execute("CREATE TABLE IF NOT EXISTS seniorities AS SELECT * FROM seniorities_df") + seniorities_df = pd.read_csv(io.StringIO("employee_id,seniority\n1,2ans\n2,4ans\n")) # noqa: F841, E501 + con.execute( + "CREATE TABLE IF NOT EXISTS seniorities AS SELECT * FROM seniorities_df" + ) def create_left_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: - orders_df = pd.DataFrame({"order_id": [1, 2, 3, 4, 5], "customer_id": [101, 102, 103, 104, 105]}) + orders_df = pd.DataFrame( # noqa: F841 + {"order_id": [1, 2, 3, 4, 5], "customer_id": [101, 102, 103, 104, 105]} + ) con.execute("CREATE TABLE IF NOT EXISTS orders AS SELECT * FROM orders_df") - customers_df = pd.DataFrame({ - "customer_id": [101, 102, 103, 104, 105, 106], - "customer_name": ["Toufik", "Daniel", "Tancrède", "Kaouter", "Jean-Nicolas", "David"], - }) + customers_df = pd.DataFrame( # noqa: F841 + { + "customer_id": [101, 102, 103, 104, 105, 106], + "customer_name": [ + "Toufik", + "Daniel", + "Tancrède", + "Kaouter", + "Jean-Nicolas", + "David", + ], + } + ) con.execute("CREATE TABLE IF NOT EXISTS customers AS SELECT * FROM customers_df") - products_df = pd.DataFrame({ - "product_id": [101, 103, 104, 105], - "product_name": ["Laptop", "Ipad", "Livre", "Petitos"], - "product_price": [800, 400, 30, 2], - }) + products_df = pd.DataFrame( # noqa: F841 + { + "product_id": [101, 103, 104, 105], + "product_name": ["Laptop", "Ipad", "Livre", "Petitos"], + "product_price": [800, 400, 30, 2], + } + ) con.execute("CREATE TABLE IF NOT EXISTS products AS SELECT * FROM products_df") - order_details_df = pd.DataFrame({ - "order_id": [1, 2, 3, 4, 5], - "product_id": [102, 104, 101, 103, 105], - "quantity": [2, 1, 3, 2, 1], - }) - con.execute("CREATE TABLE IF NOT EXISTS order_details AS SELECT * FROM order_details_df") + order_details_df = pd.DataFrame( # noqa: F841 + { + "order_id": [1, 2, 3, 4, 5], + "product_id": [102, 104, 101, 103, 105], + "quantity": [2, 1, 3, 2, 1], + } + ) + con.execute( + "CREATE TABLE IF NOT EXISTS order_details AS SELECT * FROM order_details_df" + ) def create_full_outer_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: - customers_df = pd.DataFrame({ - "customer_id": [11, 12, 13, 14, 15], - "customer_name": ["Zeinaba", "Tancrède", "Israel", "Kaouter", "Alan"], - }) + customers_df = pd.DataFrame( # noqa: F841 + { + "customer_id": [11, 12, 13, 14, 15], + "customer_name": ["Zeinaba", "Tancrède", "Israel", "Kaouter", "Alan"], + } + ) con.execute("CREATE TABLE IF NOT EXISTS df_customers AS SELECT * FROM customers_df") - stores_df = pd.DataFrame({"store_id": [1, 2, 3, 4], "customer_id": [11, 12, 13, 15]}) + stores_df = pd.DataFrame( # noqa: F841 + {"store_id": [1, 2, 3, 4], "customer_id": [11, 12, 13, 15]} + ) con.execute("CREATE TABLE IF NOT EXISTS df_stores AS SELECT * FROM stores_df") - store_products_df = pd.DataFrame({ - "store_id": [1, 1, 1, 2, 2, 3, 4], - "product_id": [101, 103, 105, 101, 103, 104, 105], - }) - con.execute("CREATE TABLE IF NOT EXISTS df_store_products AS SELECT * FROM store_products_df") - - products_df = pd.DataFrame({ - "product_id": [100, 101, 103, 104], - "product_name": ["Cherry coke", "Laptop", "Ipad", "Livre"], - "product_price": [3, 800, 400, 30], - }) + store_products_df = pd.DataFrame( # noqa: F841 + { + "store_id": [1, 1, 1, 2, 2, 3, 4], + "product_id": [101, 103, 105, 101, 103, 104, 105], + } + ) + con.execute( + "CREATE TABLE IF NOT EXISTS df_store_products AS SELECT * FROM store_products_df" + ) + + products_df = pd.DataFrame( # noqa: F841 + { + "product_id": [100, 101, 103, 104], + "product_name": ["Cherry coke", "Laptop", "Ipad", "Livre"], + "product_price": [3, 800, 400, 30], + } + ) con.execute("CREATE TABLE IF NOT EXISTS df_products AS SELECT * FROM products_df") def create_self_joins_exercises(con: duckdb.DuckDBPyConnection) -> None: - employees_df = pd.DataFrame({ - "employee_id": [11, 12, 13, 14, 15], - "employee_name": ["Sophie", "Sylvie", "Daniel", "Kaouter", "David"], - "manager_id": [13, None, 12, 13, 11], - }) + employees_df = pd.DataFrame( # noqa: F841 + { + "employee_id": [11, 12, 13, 14, 15], + "employee_name": ["Sophie", "Sylvie", "Daniel", "Kaouter", "David"], + "manager_id": [13, None, 12, 13, 11], + } + ) con.execute("CREATE TABLE IF NOT EXISTS employees AS SELECT * FROM employees_df") - sales_df = pd.DataFrame({ - "order_id": list(range(1110, 1198)), - "customer_id": random.choices([11, 12, 13, 14, 15, 11, 12, 13, 14], k=88), - }) + sales_df = pd.DataFrame( + { + "order_id": list(range(1110, 1198)), + "customer_id": random.choices([11, 12, 13, 14, 15, 11, 12, 13, 14], k=88), + } + ) sales_df["date"] = [d // 3 + 1 for d in range(1, 89)] con.execute("CREATE TABLE IF NOT EXISTS sales AS SELECT * FROM sales_df") def create_group_by_exercises(con: duckdb.DuckDBPyConnection) -> None: - ventes_immo_df = pd.DataFrame([ - [0, "vieux_lille", 460000], [1, "vieux_lille", 430000], [2, "vieux_lille", 450000], - [3, float("nan"), 470000], [4, "vieux_lille", 440000], [10, "gambetta", 336000], - [11, float("nan"), 333000], [12, "gambetta", 335000], [13, "gambetta", 337000], - [14, "gambetta", 334000], [20, "centre", 356000], [21, "centre", 353000], - [22, float("nan"), 355000], [23, "centre", 357000], [24, "centre", 354000], - [30, "wazemmes", 260000], [31, float("nan"), 230000], [32, "wazemmes", 250000], - [33, "wazemmes", 270000], [34, "wazemmes", 240000], - ], columns=["flat_id", "neighborhood", "price"]) - con.execute("CREATE TABLE IF NOT EXISTS ventes_immo AS SELECT * FROM ventes_immo_df") - - clients = ["Oussama", "Julie", "Chris", "Tom", "Jean-Nicolas", - "Aline", "Ben", "Toufik", "Sylvie", "David"] + ventes_immo_df = pd.DataFrame( # noqa: F841 + [ + [0, "vieux_lille", 460000], + [1, "vieux_lille", 430000], + [2, "vieux_lille", 450000], + [3, float("nan"), 470000], + [4, "vieux_lille", 440000], + [10, "gambetta", 336000], + [11, float("nan"), 333000], + [12, "gambetta", 335000], + [13, "gambetta", 337000], + [14, "gambetta", 334000], + [20, "centre", 356000], + [21, "centre", 353000], + [22, float("nan"), 355000], + [23, "centre", 357000], + [24, "centre", 354000], + [30, "wazemmes", 260000], + [31, float("nan"), 230000], + [32, "wazemmes", 250000], + [33, "wazemmes", 270000], + [34, "wazemmes", 240000], + ], + columns=["flat_id", "neighborhood", "price"], + ) + con.execute( + "CREATE TABLE IF NOT EXISTS ventes_immo AS SELECT * FROM ventes_immo_df" + ) + + clients = [ + "Oussama", + "Julie", + "Chris", + "Tom", + "Jean-Nicolas", + "Aline", + "Ben", + "Toufik", + "Sylvie", + "David", + ] ventes_df = pd.DataFrame( [110, 49, 65, 23, 24, 3.99, 29, 48.77, 44, 10, 60, 12, 62, 19, 75] * 2, - columns=["montant"] + columns=["montant"], ) ventes_df["client"] = clients * 3 con.execute("CREATE TABLE IF NOT EXISTS ventes AS SELECT * FROM ventes_df") def create_case_when_exercises(con: duckdb.DuckDBPyConnection) -> None: - salaires_df = pd.DataFrame({ - "name": ["Toufik", "Jean-Nicolas", "Daniel", "Kaouter", "Sylvie", "Sebastien", - "Diane", "Romain", "François", "Anna", "Zeinaba", "Gregory", - "Karima", "Arthur", "Benjamin"], - "wage": [60000, 75000, 55000, 80000, 70000, 90000, 65000, 72000, 68000, - 85000, 100000, 120000, 95000, 83000, 110000], - "department": ["IT", "HR", "SALES", "IT", "IT", "HR", "SALES", "IT", "HR", - "SALES", "IT", "IT", "HR", "SALES", "CEO"], - }) + salaires_df = pd.DataFrame( # noqa: F841 + { + "name": [ + "Toufik", + "Jean-Nicolas", + "Daniel", + "Kaouter", + "Sylvie", + "Sebastien", + "Diane", + "Romain", + "François", + "Anna", + "Zeinaba", + "Gregory", + "Karima", + "Arthur", + "Benjamin", + ], + "wage": [ + 60000, + 75000, + 55000, + 80000, + 70000, + 90000, + 65000, + 72000, + 68000, + 85000, + 100000, + 120000, + 95000, + 83000, + 110000, + ], + "department": [ + "IT", + "HR", + "SALES", + "IT", + "IT", + "HR", + "SALES", + "IT", + "HR", + "SALES", + "IT", + "IT", + "HR", + "SALES", + "CEO", + ], + } + ) con.execute("CREATE TABLE IF NOT EXISTS salaires AS SELECT * FROM salaires_df") - discount_df = pd.DataFrame({ - "order_id": [1, 2, 3, 4, 5, 6], - "product_id": [101, 102, 101, 103, 102, 103], - "quantity": [5, 3, 2, 4, 6, 2], - "price_per_unit": [10.0, 25.0, 10.0, 8.0, 25.0, 8.0], - "discount_code": [None, "DISCOUNT10", "DISCOUNT20", None, None, "UNKNOWN"], - }) + discount_df = pd.DataFrame( # noqa: F841 + { + "order_id": [1, 2, 3, 4, 5, 6], + "product_id": [101, 102, 101, 103, 102, 103], + "quantity": [5, 3, 2, 4, 6, 2], + "price_per_unit": [10.0, 25.0, 10.0, 8.0, 25.0, 8.0], + "discount_code": [None, "DISCOUNT10", "DISCOUNT20", None, None, "UNKNOWN"], + } + ) con.execute("CREATE TABLE IF NOT EXISTS discount AS SELECT * FROM discount_df") def create_grouping_set_exercises(con: duckdb.DuckDBPyConnection) -> None: - redbull_df = pd.DataFrame({ - "store_id": ["Armentieres"] * 4 + ["Lille"] * 4 + ["Douai"] * 4, - "product_name": ["redbull", "chips", "wine", "redbull", "redbull", "chips", - "wine", "icecream", "redbull", "chips", "wine", "icecream"], - "amount": [45, 60, 60, 45, 100, 140, 190, 170, 55, 70, 20, 45], - }) + redbull_df = pd.DataFrame( # noqa: F841 + { + "store_id": ["Armentieres"] * 4 + ["Lille"] * 4 + ["Douai"] * 4, + "product_name": [ + "redbull", + "chips", + "wine", + "redbull", + "redbull", + "chips", + "wine", + "icecream", + "redbull", + "chips", + "wine", + "icecream", + ], + "amount": [45, 60, 60, 45, 100, 140, 190, 170, 55, 70, 20, 45], + } + ) con.execute("CREATE TABLE IF NOT EXISTS redbull AS SELECT * FROM redbull_df") - datapop_df = pd.DataFrame({ - "year": [2016, 2017, 2018, 2019, 2020] * 3, - "region": ["IDF"] * 5 + ["HDF"] * 5 + ["PACA"] * 5, - "population": [1010000, 1020000, 1030000, 1040000, 1000000, - 910000, 920000, 930000, 940000, 900000, - 810000, 820000, 830000, 840000, 950000], - }) + datapop_df = pd.DataFrame( # noqa: F841 + { + "year": [2016, 2017, 2018, 2019, 2020] * 3, + "region": ["IDF"] * 5 + ["HDF"] * 5 + ["PACA"] * 5, + "population": [ + 1010000, + 1020000, + 1030000, + 1040000, + 1000000, + 910000, + 920000, + 930000, + 940000, + 900000, + 810000, + 820000, + 830000, + 840000, + 950000, + ], + } + ) con.execute("CREATE TABLE IF NOT EXISTS datapop AS SELECT * FROM datapop_df") @@ -191,6 +339,7 @@ def create_grouping_set_exercises(con: duckdb.DuckDBPyConnection) -> None: # POINT D'ENTRÉE # ------------------------------------------------------------ + def init_db() -> None: """Crée toutes les tables si elles n'existent pas encore.""" From 36bf3a82f0a9826f0cdee07f4142326eb36927a6 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 13:56:31 +0000 Subject: [PATCH 10/18] style(utils): ignore E501 --- utils.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/utils.py b/utils.py index de1b93a..25f4141 100644 --- a/utils.py +++ b/utils.py @@ -129,10 +129,10 @@ def check_users_solution(con, solution_df: pd.DataFrame, user_query: str) -> boo # Valeurs (order-insensitive) result_sorted = result.sort_values(by=list(result.columns)).reset_index(drop=True) - solution_sorted = solution_df.sort_values(by=list(solution_df.columns)).reset_index(drop=True) + solution_sorted = solution_df.sort_values(by=list(solution_df.columns)).reset_index(drop=True) # noqa: E501 if not result_sorted.equals(solution_sorted): - st.warning("Les colonnes et le nombre de lignes sont bons, mais certaines valeurs différent.") + st.warning("Les colonnes et le nombre de lignes sont bons, mais certaines valeurs différent.") # noqa: E501 return False st.balloons() From 6bfd279e03e5b973151514cd03f5faf867980612 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:18:35 +0000 Subject: [PATCH 11/18] fix: typo --- Exercises.yaml | 93 ++++++++++++++++++++++++++++++++++++++++++++++++++ config.py | 2 +- 2 files changed, 94 insertions(+), 1 deletion(-) create mode 100644 Exercises.yaml diff --git a/Exercises.yaml b/Exercises.yaml new file mode 100644 index 0000000..f3ead13 --- /dev/null +++ b/Exercises.yaml @@ -0,0 +1,93 @@ +# Source de vérité unique pour tous les exercices SQLingo. +# Pour ajouter un exercice : ajouter un bloc ici + le fichier .sql correspondant. +# +# Champs : +# theme : catégorie de l'exercice (utilisé pour grouper) +# name : identifiant unique de l'exercice +# tables : tables DuckDB nécessaires (pour affichage uniquement) +# answer : nom du fichier SQL solution dans Exercices/answers// + +- theme: cross_joins + name: cross_joins_1 + tables: [beverages, food_items] + answer: cross_joins_1.sql + +- theme: cross_joins + name: cross_joins_2 + tables: [sizes, trademarks] + answer: cross_joins_2.sql + +- theme: cross_joins + name: cross_joins_3 + tables: [hours, quarters] + answer: cross_joins_3.sql + +- theme: inner_joins + name: inner_joins_1 + tables: [salaries, seniorities] + answer: inner_joins_1.sql + +- theme: left_joins + name: left_joins_1 + tables: [products, order_details] + answer: left_joins_1.sql + +- theme: left_joins + name: left_joins_2 + tables: [orders, customers, products, order_details] + answer: left_joins_2.sql + +- theme: full_outer_joins + name: full_outer_joins_1 + tables: [df_store_products, df_products] + answer: full_outer_joins_1.sql + +- theme: full_outer_joins + name: full_outer_joins_2 + tables: [df_customers, df_stores, df_store_products, df_products] + answer: full_outer_joins_2.sql + +- theme: self_joins + name: self_joins_1 + tables: [employees] + answer: self_joins_1.sql + +- theme: self_joins + name: self_joins_2 + tables: [sales] + answer: self_joins_2.sql + +- theme: group_by + name: group_by_1 + tables: [ventes_immo] + answer: group_by_1.sql + +- theme: group_by + name: group_by_2 + tables: [ventes] + answer: group_by_2.sql + +- theme: case_when + name: case_when_1 + tables: [salaires] + answer: case_when_1.sql + +- theme: case_when + name: case_when_2 + tables: [discount] + answer: case_when_2.sql + +- theme: case_when + name: case_when_3 + tables: [salaires] + answer: case_when_3.sql + +- theme: grouping_set + name: grouping_set_1 + tables: [redbull] + answer: grouping_set_1.sql + +- theme: grouping_set + name: grouping_set_2 + tables: [datapop] + answer: grouping_set_2.sql \ No newline at end of file diff --git a/config.py b/config.py index 9cbf5d2..6e1dad1 100644 --- a/config.py +++ b/config.py @@ -4,7 +4,7 @@ # Paramètres chemin d'accès DB_PATH = Path("exercices_tables.duckdb") -EXERCICES_PATH = Path("Exercices.yaml") +EXERCISES_PATH = Path("Exercises.yaml") QUESTIONS_PATH = Path("Exercices/questions") ANSWERS_PATH = Path("Exercices/answers") From 7eae4464e501e4af4dc4b2ad58bb0163484279ac Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:18:54 +0000 Subject: [PATCH 12/18] chore: add pyyaml --- pyproject.toml | 1 + uv.lock | 38 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 39 insertions(+) diff --git a/pyproject.toml b/pyproject.toml index 99d00a6..4738464 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -6,5 +6,6 @@ readme = "README.md" requires-python = ">=3.13" dependencies = [ "duckdb>=1.5.2", + "pyyaml>=6.0.3", "streamlit>=1.56.0", ] diff --git a/uv.lock b/uv.lock index 57e9683..8932158 100644 --- a/uv.lock +++ b/uv.lock @@ -532,6 +532,42 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/ec/57/56b9bcc3c9c6a792fcbaf139543cee77261f3651ca9da0c93f5c1221264b/python_dateutil-2.9.0.post0-py2.py3-none-any.whl", hash = "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427", size = 229892, upload-time = "2024-03-01T18:36:18.57Z" }, ] +[[package]] +name = "pyyaml" +version = "6.0.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/05/8e/961c0007c59b8dd7729d542c61a4d537767a59645b82a0b521206e1e25c2/pyyaml-6.0.3.tar.gz", hash = "sha256:d76623373421df22fb4cf8817020cbb7ef15c725b9d5e45f17e189bfc384190f", size = 130960, upload-time = "2025-09-25T21:33:16.546Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/11/0fd08f8192109f7169db964b5707a2f1e8b745d4e239b784a5a1dd80d1db/pyyaml-6.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:8da9669d359f02c0b91ccc01cac4a67f16afec0dac22c2ad09f46bee0697eba8", size = 181669, upload-time = "2025-09-25T21:32:23.673Z" }, + { url = "https://files.pythonhosted.org/packages/b1/16/95309993f1d3748cd644e02e38b75d50cbc0d9561d21f390a76242ce073f/pyyaml-6.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:2283a07e2c21a2aa78d9c4442724ec1eb15f5e42a723b99cb3d822d48f5f7ad1", size = 173252, upload-time = "2025-09-25T21:32:25.149Z" }, + { url = "https://files.pythonhosted.org/packages/50/31/b20f376d3f810b9b2371e72ef5adb33879b25edb7a6d072cb7ca0c486398/pyyaml-6.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ee2922902c45ae8ccada2c5b501ab86c36525b883eff4255313a253a3160861c", size = 767081, upload-time = "2025-09-25T21:32:26.575Z" }, + { url = "https://files.pythonhosted.org/packages/49/1e/a55ca81e949270d5d4432fbbd19dfea5321eda7c41a849d443dc92fd1ff7/pyyaml-6.0.3-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a33284e20b78bd4a18c8c2282d549d10bc8408a2a7ff57653c0cf0b9be0afce5", size = 841159, upload-time = "2025-09-25T21:32:27.727Z" }, + { url = "https://files.pythonhosted.org/packages/74/27/e5b8f34d02d9995b80abcef563ea1f8b56d20134d8f4e5e81733b1feceb2/pyyaml-6.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0f29edc409a6392443abf94b9cf89ce99889a1dd5376d94316ae5145dfedd5d6", size = 801626, upload-time = "2025-09-25T21:32:28.878Z" }, + { url = "https://files.pythonhosted.org/packages/f9/11/ba845c23988798f40e52ba45f34849aa8a1f2d4af4b798588010792ebad6/pyyaml-6.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:f7057c9a337546edc7973c0d3ba84ddcdf0daa14533c2065749c9075001090e6", size = 753613, upload-time = "2025-09-25T21:32:30.178Z" }, + { url = "https://files.pythonhosted.org/packages/3d/e0/7966e1a7bfc0a45bf0a7fb6b98ea03fc9b8d84fa7f2229e9659680b69ee3/pyyaml-6.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:eda16858a3cab07b80edaf74336ece1f986ba330fdb8ee0d6c0d68fe82bc96be", size = 794115, upload-time = "2025-09-25T21:32:31.353Z" }, + { url = "https://files.pythonhosted.org/packages/de/94/980b50a6531b3019e45ddeada0626d45fa85cbe22300844a7983285bed3b/pyyaml-6.0.3-cp313-cp313-win32.whl", hash = "sha256:d0eae10f8159e8fdad514efdc92d74fd8d682c933a6dd088030f3834bc8e6b26", size = 137427, upload-time = "2025-09-25T21:32:32.58Z" }, + { url = "https://files.pythonhosted.org/packages/97/c9/39d5b874e8b28845e4ec2202b5da735d0199dbe5b8fb85f91398814a9a46/pyyaml-6.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:79005a0d97d5ddabfeeea4cf676af11e647e41d81c9a7722a193022accdb6b7c", size = 154090, upload-time = "2025-09-25T21:32:33.659Z" }, + { url = "https://files.pythonhosted.org/packages/73/e8/2bdf3ca2090f68bb3d75b44da7bbc71843b19c9f2b9cb9b0f4ab7a5a4329/pyyaml-6.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:5498cd1645aa724a7c71c8f378eb29ebe23da2fc0d7a08071d89469bf1d2defb", size = 140246, upload-time = "2025-09-25T21:32:34.663Z" }, + { url = "https://files.pythonhosted.org/packages/9d/8c/f4bd7f6465179953d3ac9bc44ac1a8a3e6122cf8ada906b4f96c60172d43/pyyaml-6.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:8d1fab6bb153a416f9aeb4b8763bc0f22a5586065f86f7664fc23339fc1c1fac", size = 181814, upload-time = "2025-09-25T21:32:35.712Z" }, + { url = "https://files.pythonhosted.org/packages/bd/9c/4d95bb87eb2063d20db7b60faa3840c1b18025517ae857371c4dd55a6b3a/pyyaml-6.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:34d5fcd24b8445fadc33f9cf348c1047101756fd760b4dacb5c3e99755703310", size = 173809, upload-time = "2025-09-25T21:32:36.789Z" }, + { url = "https://files.pythonhosted.org/packages/92/b5/47e807c2623074914e29dabd16cbbdd4bf5e9b2db9f8090fa64411fc5382/pyyaml-6.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:501a031947e3a9025ed4405a168e6ef5ae3126c59f90ce0cd6f2bfc477be31b7", size = 766454, upload-time = "2025-09-25T21:32:37.966Z" }, + { url = "https://files.pythonhosted.org/packages/02/9e/e5e9b168be58564121efb3de6859c452fccde0ab093d8438905899a3a483/pyyaml-6.0.3-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:b3bc83488de33889877a0f2543ade9f70c67d66d9ebb4ac959502e12de895788", size = 836355, upload-time = "2025-09-25T21:32:39.178Z" }, + { url = "https://files.pythonhosted.org/packages/88/f9/16491d7ed2a919954993e48aa941b200f38040928474c9e85ea9e64222c3/pyyaml-6.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c458b6d084f9b935061bc36216e8a69a7e293a2f1e68bf956dcd9e6cbcd143f5", size = 794175, upload-time = "2025-09-25T21:32:40.865Z" }, + { url = "https://files.pythonhosted.org/packages/dd/3f/5989debef34dc6397317802b527dbbafb2b4760878a53d4166579111411e/pyyaml-6.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:7c6610def4f163542a622a73fb39f534f8c101d690126992300bf3207eab9764", size = 755228, upload-time = "2025-09-25T21:32:42.084Z" }, + { url = "https://files.pythonhosted.org/packages/d7/ce/af88a49043cd2e265be63d083fc75b27b6ed062f5f9fd6cdc223ad62f03e/pyyaml-6.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:5190d403f121660ce8d1d2c1bb2ef1bd05b5f68533fc5c2ea899bd15f4399b35", size = 789194, upload-time = "2025-09-25T21:32:43.362Z" }, + { url = "https://files.pythonhosted.org/packages/23/20/bb6982b26a40bb43951265ba29d4c246ef0ff59c9fdcdf0ed04e0687de4d/pyyaml-6.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:4a2e8cebe2ff6ab7d1050ecd59c25d4c8bd7e6f400f5f82b96557ac0abafd0ac", size = 156429, upload-time = "2025-09-25T21:32:57.844Z" }, + { url = "https://files.pythonhosted.org/packages/f4/f4/a4541072bb9422c8a883ab55255f918fa378ecf083f5b85e87fc2b4eda1b/pyyaml-6.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:93dda82c9c22deb0a405ea4dc5f2d0cda384168e466364dec6255b293923b2f3", size = 143912, upload-time = "2025-09-25T21:32:59.247Z" }, + { url = "https://files.pythonhosted.org/packages/7c/f9/07dd09ae774e4616edf6cda684ee78f97777bdd15847253637a6f052a62f/pyyaml-6.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:02893d100e99e03eda1c8fd5c441d8c60103fd175728e23e431db1b589cf5ab3", size = 189108, upload-time = "2025-09-25T21:32:44.377Z" }, + { url = "https://files.pythonhosted.org/packages/4e/78/8d08c9fb7ce09ad8c38ad533c1191cf27f7ae1effe5bb9400a46d9437fcf/pyyaml-6.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:c1ff362665ae507275af2853520967820d9124984e0f7466736aea23d8611fba", size = 183641, upload-time = "2025-09-25T21:32:45.407Z" }, + { url = "https://files.pythonhosted.org/packages/7b/5b/3babb19104a46945cf816d047db2788bcaf8c94527a805610b0289a01c6b/pyyaml-6.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6adc77889b628398debc7b65c073bcb99c4a0237b248cacaf3fe8a557563ef6c", size = 831901, upload-time = "2025-09-25T21:32:48.83Z" }, + { url = "https://files.pythonhosted.org/packages/8b/cc/dff0684d8dc44da4d22a13f35f073d558c268780ce3c6ba1b87055bb0b87/pyyaml-6.0.3-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a80cb027f6b349846a3bf6d73b5e95e782175e52f22108cfa17876aaeff93702", size = 861132, upload-time = "2025-09-25T21:32:50.149Z" }, + { url = "https://files.pythonhosted.org/packages/b1/5e/f77dc6b9036943e285ba76b49e118d9ea929885becb0a29ba8a7c75e29fe/pyyaml-6.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:00c4bdeba853cc34e7dd471f16b4114f4162dc03e6b7afcc2128711f0eca823c", size = 839261, upload-time = "2025-09-25T21:32:51.808Z" }, + { url = "https://files.pythonhosted.org/packages/ce/88/a9db1376aa2a228197c58b37302f284b5617f56a5d959fd1763fb1675ce6/pyyaml-6.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:66e1674c3ef6f541c35191caae2d429b967b99e02040f5ba928632d9a7f0f065", size = 805272, upload-time = "2025-09-25T21:32:52.941Z" }, + { url = "https://files.pythonhosted.org/packages/da/92/1446574745d74df0c92e6aa4a7b0b3130706a4142b2d1a5869f2eaa423c6/pyyaml-6.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:16249ee61e95f858e83976573de0f5b2893b3677ba71c9dd36b9cf8be9ac6d65", size = 829923, upload-time = "2025-09-25T21:32:54.537Z" }, + { url = "https://files.pythonhosted.org/packages/f0/7a/1c7270340330e575b92f397352af856a8c06f230aa3e76f86b39d01b416a/pyyaml-6.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4ad1906908f2f5ae4e5a8ddfce73c320c2a1429ec52eafd27138b7f1cbe341c9", size = 174062, upload-time = "2025-09-25T21:32:55.767Z" }, + { url = "https://files.pythonhosted.org/packages/f1/12/de94a39c2ef588c7e6455cfbe7343d3b2dc9d6b6b2f40c4c6565744c873d/pyyaml-6.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:ebc55a14a21cb14062aa4162f906cd962b28e2e9ea38f9b4391244cd8de4ae0b", size = 149341, upload-time = "2025-09-25T21:32:56.828Z" }, +] + [[package]] name = "referencing" version = "0.37.0" @@ -650,12 +686,14 @@ version = "0.1.0" source = { virtual = "." } dependencies = [ { name = "duckdb" }, + { name = "pyyaml" }, { name = "streamlit" }, ] [package.metadata] requires-dist = [ { name = "duckdb", specifier = ">=1.5.2" }, + { name = "pyyaml", specifier = ">=6.0.3" }, { name = "streamlit", specifier = ">=1.56.0" }, ] From d77c67d55ebfd9f605c6c59355d68477c0e0dbcb Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:20:47 +0000 Subject: [PATCH 13/18] refactor: simplify app with session-based SRS, remove auth - Remove user authentication (login, signup, bcrypt) - Remove persistent memory_state, replace with st.session_state queue - Split UI into render_home / render_exercise / render_end - Add theme filter on home screen - Add progress bar and score in sidebar - Fix order-insensitive solution comparison --- app.py | 337 +++++++++++++++++++++++++++++---------------------------- 1 file changed, 169 insertions(+), 168 deletions(-) diff --git a/app.py b/app.py index 017295e..aef3624 100644 --- a/app.py +++ b/app.py @@ -1,201 +1,202 @@ -# pylint: disable=missing-module-docstring -import os -import sys -import subprocess -import logging -from datetime import date, timedelta - -import duckdb +""" +app.py - Point d'entrée de SQLingo. +""" + import streamlit as st -from functions import ( - get_questions, - get_selected_exercise, - get_selector_exercises, - get_selector_themes, +from db import get_connection, init_db +from utils import ( check_users_solution, - query_memory_df, - signup_user, - user_auth, + get_current_exercise, + init_session, + load_exercises, + load_solution, + record_result, + show_question, + show_tables, ) - # ------------------------------------------------------------ -# CONFIG PAGE +# CONFIG # ------------------------------------------------------------ + st.set_page_config( - page_title="SQL_SRS", + page_title="SQLingo", page_icon="😎", layout="wide", ) -st.markdown( - """ - - """, - unsafe_allow_html=True, -) +# ------------------------------------------------------------ +# INITIALISATION DB +# ------------------------------------------------------------ + +init_db() +con = get_connection() # ------------------------------------------------------------ -# SETUP +# ÉCRAN D'ACCUEIL # ------------------------------------------------------------ -# Création du dossier data -if "data" not in os.listdir(): - print("creating folder data") - logging.error(os.listdir()) - logging.error("creating folder data") - os.mkdir("data") - -# On lance init_db.py pour créer la BDD -if "exercises_sql_tables.duckdb" not in os.listdir("data"): - try: - subprocess.run([sys.executable, "init_db.py"], check=True) - signup_user("guest", "guest") - except subprocess.CalledProcessError as e: - print(f"Failed to execute init_db.py: {e}") + + +def render_home() -> None: + st.title("😎 SQLingo") + st.markdown("Entraîne-toi au SQL avec un système de répétition espacée.") + st.divider() + + exercises = load_exercises() + all_themes = sorted({e["theme"] for e in exercises}) + + st.subheader("Thèmes") + selected_themes = [] + cols = st.columns(3) + for i, theme in enumerate(all_themes): + with cols[i % 3]: + if st.checkbox(theme.replace("_", " ").title(), value=True, key=f"theme_{theme}"): + selected_themes.append(theme) + + st.divider() + + n_selected = sum(1 for e in exercises if e["theme"] in selected_themes) + st.caption(f"{n_selected} exercice(s) sélectionné(s)") + + if st.button("🚀 Démarrer", disabled=not selected_themes, type="primary"): + init_session(themes=selected_themes) + st.rerun() + # ------------------------------------------------------------ -# AUTHENT +# ÉCRAN EXERCICE # ------------------------------------------------------------ -user_auth() +def render_exercise() -> None: + exercise = get_current_exercise() -if "logged_in" in st.session_state and st.session_state["logged_in"]: - # -------------------- - # Affichage de l'app - # -------------------- - current_user = st.session_state["username"] - con = duckdb.connect(database="data/exercises_sql_tables.duckdb", read_only=False) - memory_df = query_memory_df(con, current_user) + if exercise is None: + st.session_state["session_started"] = False + st.session_state["session_finished"] = True + st.rerun() + return + + score = st.session_state["score"] + queue = st.session_state["queue"] + total = score["correct"] + score["incorrect"] + len(queue) + done = score["correct"] + score["incorrect"] + + # --- Sidebar --- with st.sidebar: - # st.write(f"Bienvenue, {st.session_state['username']}!") + st.markdown("## 😎 SQLingo") + st.divider() + st.metric("✅ Correct", score["correct"]) + st.metric("❌ À revoir", score["incorrect"]) + st.progress(done / total if total else 0) + st.caption(f"{done} / {total} exercices") + st.divider() + if st.button("🏠 Accueil", use_container_width=True): + st.session_state["session_started"] = False + st.rerun() - # Forcer le padding de la sidebar pour éviter l'espace blanc en haut + st.divider() st.markdown( - """ - - """, + "Made with 🍜 by " + "Fabien" + "
" + "Voir sur GitHub", unsafe_allow_html=True, ) - # Titre dans la sidebar - st.markdown( - """ -
-

😎 SQLingo 😎

-
- """, - unsafe_allow_html=True, - ) + # --- Contenu principal --- + theme_label = exercise["theme"].replace("_", " ").title() + st.markdown(f"**Thème :** {theme_label} · **Exercice :** `{exercise['name']}`") + st.divider() - # Sélection du thème - theme = get_selector_themes(memory_df) + col_question, col_tables = st.columns([3, 2]) - # Sélection de l'exercice en fonction du thème sélectionné - exercises_lst = get_selector_exercises(memory_df, theme) + with col_question: + st.subheader("Question") + show_question(exercise) - # Récupérer l'exercice - exercise, answer_str, answer, solution_df = get_selected_exercise( - con, theme, exercises_lst, current_user + st.subheader("Ta requête") + user_query = st.text_area( + label="Saisir ta requête SQL :", + height=200, + key=f"query_{exercise['name']}", + label_visibility="collapsed", ) - # TODO : Ajuster l'affichage du df en injectant du css pour fixer sa taille - # user + deconnect - st.write(f"Vous êtes connecté en tant que {current_user}") - if st.button("Déconnexion"): - st.session_state["logged_in"] = False - st.session_state["username"] = None - st.rerun() - st.markdown( - """ - Made with 🍜 by \ - Fabien\ -

Mon site \ -
GitHub - """, - unsafe_allow_html=True, - ) + if st.button("✅ Valider", type="primary", disabled=not user_query): + try: + solution_df = load_solution(con, exercise) + except FileNotFoundError: + st.error(f"Fichier solution introuvable pour `{exercise['name']}`.") + return + + correct = check_users_solution(con, solution_df, user_query) + record_result(exercise, correct) + + if correct: + st.success("Bonne réponse ! Exercice suivant →") + if st.button("Continuer →"): + st.rerun() + else: + st.error("Pas tout à fait - l'exercice reviendra un peu plus tard.") + if st.button("Continuer →"): + st.rerun() + + with col_tables: + st.subheader("Tables disponibles") + show_tables(con, exercise) + + # --- Onglet solution --- + with st.expander("Voir la solution"): + try: + solution_df = load_solution(con, exercise) + from config import ANSWERS_PATH + sql = (ANSWERS_PATH / exercise["theme"] / exercise["answer"]).read_text() + st.code(sql, language="sql") + st.dataframe(solution_df) + except FileNotFoundError: + st.warning("Fichier solution introuvable.") + + +# ------------------------------------------------------------ +# ÉCRAN DE FIN +# ------------------------------------------------------------ + +def render_end() -> None: + score = st.session_state["score"] + history = st.session_state["history"] + total = score["correct"] + score["incorrect"] + + st.title("🎉 Session terminée !") + st.divider() + + col1, col2, col3 = st.columns(3) + col1.metric("Total", total) + col2.metric("✅ Corrects", score["correct"]) + col3.metric("❌ À revoir", score["incorrect"]) + + errors = [(ex, r) for ex, r in history if r == "incorrect"] + if errors: + st.divider() + st.subheader("Exercices à retravailler") + for ex, _ in errors: + st.markdown(f"- **{ex['theme']}** · `{ex['name']}`") + + st.divider() + if st.button("🔄 Recommencer", type="primary"): + for key in ["session_started", "session_finished", "queue", "score", "history"]: + st.session_state.pop(key, None) + st.rerun() + + +# ------------------------------------------------------------ +# ROUTEUR +# ------------------------------------------------------------ - # Affichage des questions dynamiques - st.subheader("Question :") - get_questions(theme, answer_str) - - # Saisir la requête - query = st.text_area(label="Saisir votre requête SQL :", key="user_input") - - # Check de la requête - if query: - IS_SOLUTION_CORRECT = check_users_solution(con, solution_df, query) - - # si la solution est ok alors on met à jour la date "last_reviewed" - if IS_SOLUTION_CORRECT: - today = date.today() - - # Boutons pour mettre à jour la date de prochaine apparition de la question - col1, col2, col3 = st.columns(spec=3, gap="small") - with col1: - if st.button(label="Revoir dès demain"): - next_review_date = today + timedelta(days=1) - # print(f"{current_user} veut revoir la requête le {next_review_date}") - - with col2: - if st.button(label="Revoir dans 7 jours"): - next_review_date = today + timedelta(days=7) - # print(f"{current_user} veut revoir la requête le {next_review_date}") - - with col3: - if st.button(label="Revoir dans 14 jours"): - next_review_date = today + timedelta(days=14) - # print(f"{current_user} veut revoir la requête le {next_review_date}") - - # Si un bouton a été cliqué, mettre à jour la date - if "next_review_date" in locals(): - exercise_name = exercise.loc[0, "exercise_name"] - UPDATE_QUERY = """ - UPDATE memory_state - SET last_reviewed = ? - WHERE exercise_name = ? - AND user_id = ? - """ - with duckdb.connect("data/exercises_sql_tables.duckdb") as conn: - conn.execute( - UPDATE_QUERY, - ( - next_review_date.strftime("%Y-%m-%d"), - exercise_name, - current_user, - ), - ) - conn.close() - # st.rerun() - - tab1, tab2 = st.tabs(["Tables", "Solution"]) - - with tab1: - exercise_tables = exercise.loc[0, "tables"] - exercise_tables_len = len(exercise_tables) - cols = st.columns(exercise_tables_len) - for i in range(0, exercise_tables_len): - cols[i].write(exercise_tables[i]) - df_table = con.execute(f"SELECT * FROM {exercise_tables[i]}").df() - cols[i].table(df_table) - - with tab2: - st.code(answer, language="sql") - st.dataframe(solution_df) +if st.session_state.get("session_finished"): + render_end() +elif st.session_state.get("session_started"): + render_exercise() else: - st.write("Veuillez vous connecter ou vous inscrire pour continuer.") - st.write( - "Si vous ne souhaitez pas vous inscrire \ - vous pouvez utiliser l'identifiant et \ - le mot de passe : guest" - ) + render_home() From f5003565d995999541866cc40f91b259f354e6da Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:39:39 +0000 Subject: [PATCH 14/18] build: add make's commands --- Makefile | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) create mode 100644 Makefile diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..7551d0b --- /dev/null +++ b/Makefile @@ -0,0 +1,16 @@ +.PHONY: install init run start reset + +install: + uv sync + +init: + uv run db.py + +run: + uv run streamlit run app.py + +start: install init run + +reset: + rm -f data/exercises_sql_tables.duckdb + uv run db.py \ No newline at end of file From 119c1b62aaea3cd2c1b1ad7fe0ebce6267538f2d Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:39:55 +0000 Subject: [PATCH 15/18] docs: update README --- README.md | 71 +++++++++++++++++++++++++++++++++++++++++++------------ 1 file changed, 56 insertions(+), 15 deletions(-) diff --git a/README.md b/README.md index aeb17c2..4c715c0 100644 --- a/README.md +++ b/README.md @@ -1,24 +1,65 @@ +# 😎 SQLingo -# SRS - Spaced Repetition System +Entraîne-toi au SQL ! -Le **Spaced Repetition System (SRS)** est une technique d'apprentissage qui consiste à augmenter les intervalles de révision entre chaque session de révision pour un même contenu. Cette méthode est basée sur l'effet de la courbe de l'oubli, qui montre comment notre capacité à retenir des informations diminue avec le temps sans révision. +## Comment ça marche -**Utilité :** -- **Efficacité de l'apprentissage :** En révisant les informations juste avant que l'oubli ne commence, le SRS augmente la rétention à long terme des connaissances. -- **Optimisation du temps :** Les utilisateurs passent moins de temps sur des sujets qu'ils maîtrisent déjà et se concentrent davantage sur des domaines moins familiers. +Au lancement, tu sélectionnes les thèmes que tu veux travailler. SQLingo mélange les exercices et te les présente un par un. Si tu réponds correctement, l'exercice disparaît de la queue. Sinon, il revient quelques exercices plus tard. En fin de session, tu vois ton score et la liste des exercices à retravailler. -## Streamlit +## Thèmes disponibles -**Définition :** **Streamlit** est un framework open-source de Python conçu pour créer des applications webs. +- Cross joins, Inner joins, Left joins, Full outer joins, Self joins +- Group by +- Case when +- Grouping sets -**Utilité :** -- **Déploiement rapide :** Permet aux data scientists de transformer des scripts Python en applications web interactives sans nécessiter une expertise approfondie en développement web. -- **Interactivité :** Offre des widgets intégrés pour interagir avec les utilisateurs. +## Stack -## DuckDB +- **[Streamlit](https://streamlit.io/)** — interface web interactive en Python +- **[DuckDB](https://duckdb.org/)** — base de données embarquée pour exécuter les requêtes SQL +- **[uv](https://github.com/astral-sh/uv)** — gestion des dépendances -**DuckDB** est un moteur de base de données embarqué orienté colonnes, conçu pour l'analyse de données OLAP (Online Analytical Processing) sur des systèmes de gestion de base de données relationnelle. +## Choix techniques -**Utilité :** -- **Facilité d'intégration :** Peut être utilisé comme une bibliothèque au sein d'autres applications sans nécessiter de configuration de serveur séparée. -- **Performance optimisée :** Utilise l'exécution de requêtes vectorisées pour améliorer les performances de lecture et d'analyse des données. +**Pas de compte utilisateur.** La progression est gérée entièrement via `st.session_state`. Une queue d'exercices mélangée au démarrage, repositionnée selon les résultats. + +**DuckDB comme moteur SQL.** Pas de serveur à configurer, DuckDB tourne directement dans le process Python et exécute des requêtes SQL. + +**Une seule source de vérité pour les exercices.** Tous les exercices sont définis dans `exercises.yaml`. Ajouter un exercice ne nécessite pas de toucher au code Python. + +**Comparaison order-insensitive.** La vérification des requêtes trie les deux DataFrames avant comparaison — une requête correcte sans `ORDER BY` explicite n'est pas pénalisée. + +## Lancer le projet + +```bash +make start # install + init DB + lance l'app +``` + +Les commandes disponibles : + +```bash +make install # installe les dépendances (uv sync) +make init # initialise la base de données +make run # lance l'application +make reset # repart d'une DB vierge +``` + +## Structure + +``` +SQLingo/ +├── app.py # Point d'entrée, routing entre les écrans +├── db.py # Initialisation DB et connexion partagée +├── utils.py # Logique SRS, vérification des requêtes, affichage +├── config.py # Chemins et constantes globales +├── exercises.yaml # Source de vérité pour tous les exercices +└── Exercices/ + ├── answers/ # Fichiers SQL solution par thème + └── questions/ # Énoncés Markdown par thème +``` + +## Ajouter un exercice + +1. Ajouter un bloc dans `exercises.yaml` +2. Créer le fichier `Exercices/questions//.md` +3. Créer le fichier `Exercices/answers//.sql` From d0804a85f5e86e5cd719e56a44a41eb3ee0f20c9 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:40:16 +0000 Subject: [PATCH 16/18] update: use uv and ruff --- .github/workflows/check_code_quality.yaml | 33 +++++++++++++---------- 1 file changed, 19 insertions(+), 14 deletions(-) diff --git a/.github/workflows/check_code_quality.yaml b/.github/workflows/check_code_quality.yaml index c6cef25..da53b9c 100644 --- a/.github/workflows/check_code_quality.yaml +++ b/.github/workflows/check_code_quality.yaml @@ -1,17 +1,22 @@ -name: check_code_quality -on: pull_request -jobs: - black: # mettre le nom qu'on souhaite ici - runs-on: ubuntu-20.04 +name: code quality + +on: + push: + branches: [main] + pull_request: + branches: [main] + +jobs: + ruff: + runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 - - uses: actions/setup-python@v4 - with: - python-version: 3.9 - - run: | #bash multi-lignes - python -m pip install --upgrade pip - pip install black - - run: | - black --check --verbose . + - uses: actions/checkout@v4 + + - name: Install uv + uses: astral-sh/setup-uv@v5 + - name: Check formatting + run: uvx ruff format --check . + - name: Check linting + run: uvx ruff check . \ No newline at end of file From ebe1a7993166d99335b07a8199cc462a1849c8aa Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:42:20 +0000 Subject: [PATCH 17/18] style: ruff format --- app.py | 7 ++++++- utils.py | 16 +++++++++++++--- 2 files changed, 19 insertions(+), 4 deletions(-) diff --git a/app.py b/app.py index aef3624..c275e57 100644 --- a/app.py +++ b/app.py @@ -51,7 +51,9 @@ def render_home() -> None: cols = st.columns(3) for i, theme in enumerate(all_themes): with cols[i % 3]: - if st.checkbox(theme.replace("_", " ").title(), value=True, key=f"theme_{theme}"): + if st.checkbox( + theme.replace("_", " ").title(), value=True, key=f"theme_{theme}" + ): selected_themes.append(theme) st.divider() @@ -68,6 +70,7 @@ def render_home() -> None: # ÉCRAN EXERCICE # ------------------------------------------------------------ + def render_exercise() -> None: exercise = get_current_exercise() @@ -152,6 +155,7 @@ def render_exercise() -> None: try: solution_df = load_solution(con, exercise) from config import ANSWERS_PATH + sql = (ANSWERS_PATH / exercise["theme"] / exercise["answer"]).read_text() st.code(sql, language="sql") st.dataframe(solution_df) @@ -163,6 +167,7 @@ def render_exercise() -> None: # ÉCRAN DE FIN # ------------------------------------------------------------ + def render_end() -> None: score = st.session_state["score"] history = st.session_state["history"] diff --git a/utils.py b/utils.py index 25f4141..b5752f8 100644 --- a/utils.py +++ b/utils.py @@ -26,6 +26,7 @@ # CHARGEMENT DES EXERCICES # ------------------------------------------------------------ + def load_exercises() -> list[dict]: """Charge la liste des exercices depuis exercises.yaml. @@ -40,6 +41,7 @@ def load_exercises() -> list[dict]: # SESSION SRS-LIKE # ------------------------------------------------------------ + def init_session(themes: list[str] | None = None) -> None: """Initialise la session Streamlit avec une queue d'exercices mélangée. @@ -91,6 +93,7 @@ def record_result(exercise: dict, correct: bool) -> None: # VÉRIFICATION DE LA REQUÊTE UTILISATEUR # ------------------------------------------------------------ + def check_users_solution(con, solution_df: pd.DataFrame, user_query: str) -> bool: """Vérifie que la requête SQL de l'utilisateur produit le bon résultat. @@ -128,11 +131,17 @@ def check_users_solution(con, solution_df: pd.DataFrame, user_query: str) -> boo return False # Valeurs (order-insensitive) - result_sorted = result.sort_values(by=list(result.columns)).reset_index(drop=True) - solution_sorted = solution_df.sort_values(by=list(solution_df.columns)).reset_index(drop=True) # noqa: E501 + result_sorted = result.sort_values(by=list(result.columns)).reset_index( + drop=True + ) + solution_sorted = solution_df.sort_values( + by=list(solution_df.columns) + ).reset_index(drop=True) # noqa: E501 if not result_sorted.equals(solution_sorted): - st.warning("Les colonnes et le nombre de lignes sont bons, mais certaines valeurs différent.") # noqa: E501 + st.warning( + "Les colonnes et le nombre de lignes sont bons, mais certaines valeurs différent." + ) # noqa: E501 return False st.balloons() @@ -147,6 +156,7 @@ def check_users_solution(con, solution_df: pd.DataFrame, user_query: str) -> boo # AFFICHAGE # ------------------------------------------------------------ + def show_question(exercise: dict) -> None: """Affiche l'énoncé markdown de l'exercice courant. From dc542c2d0fa207cb5f9da9c8840e56262d0964b3 Mon Sep 17 00:00:00 2001 From: surybang Date: Sat, 25 Apr 2026 14:42:48 +0000 Subject: [PATCH 18/18] fix: typo name file --- Exercices.yaml | 93 -------------------------------------------------- 1 file changed, 93 deletions(-) delete mode 100644 Exercices.yaml diff --git a/Exercices.yaml b/Exercices.yaml deleted file mode 100644 index f3ead13..0000000 --- a/Exercices.yaml +++ /dev/null @@ -1,93 +0,0 @@ -# Source de vérité unique pour tous les exercices SQLingo. -# Pour ajouter un exercice : ajouter un bloc ici + le fichier .sql correspondant. -# -# Champs : -# theme : catégorie de l'exercice (utilisé pour grouper) -# name : identifiant unique de l'exercice -# tables : tables DuckDB nécessaires (pour affichage uniquement) -# answer : nom du fichier SQL solution dans Exercices/answers// - -- theme: cross_joins - name: cross_joins_1 - tables: [beverages, food_items] - answer: cross_joins_1.sql - -- theme: cross_joins - name: cross_joins_2 - tables: [sizes, trademarks] - answer: cross_joins_2.sql - -- theme: cross_joins - name: cross_joins_3 - tables: [hours, quarters] - answer: cross_joins_3.sql - -- theme: inner_joins - name: inner_joins_1 - tables: [salaries, seniorities] - answer: inner_joins_1.sql - -- theme: left_joins - name: left_joins_1 - tables: [products, order_details] - answer: left_joins_1.sql - -- theme: left_joins - name: left_joins_2 - tables: [orders, customers, products, order_details] - answer: left_joins_2.sql - -- theme: full_outer_joins - name: full_outer_joins_1 - tables: [df_store_products, df_products] - answer: full_outer_joins_1.sql - -- theme: full_outer_joins - name: full_outer_joins_2 - tables: [df_customers, df_stores, df_store_products, df_products] - answer: full_outer_joins_2.sql - -- theme: self_joins - name: self_joins_1 - tables: [employees] - answer: self_joins_1.sql - -- theme: self_joins - name: self_joins_2 - tables: [sales] - answer: self_joins_2.sql - -- theme: group_by - name: group_by_1 - tables: [ventes_immo] - answer: group_by_1.sql - -- theme: group_by - name: group_by_2 - tables: [ventes] - answer: group_by_2.sql - -- theme: case_when - name: case_when_1 - tables: [salaires] - answer: case_when_1.sql - -- theme: case_when - name: case_when_2 - tables: [discount] - answer: case_when_2.sql - -- theme: case_when - name: case_when_3 - tables: [salaires] - answer: case_when_3.sql - -- theme: grouping_set - name: grouping_set_1 - tables: [redbull] - answer: grouping_set_1.sql - -- theme: grouping_set - name: grouping_set_2 - tables: [datapop] - answer: grouping_set_2.sql \ No newline at end of file