From b2594b9daeef5ce48d769d92e64600cc9ba02ead Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Thu, 10 Oct 2024 14:13:24 +0200 Subject: [PATCH 01/16] feat: first iteration of background job to import datasets from hub --- argilla-server/pdm.lock | 330 ++++++++---------- argilla-server/pyproject.toml | 3 +- .../api/handlers/v1/datasets/datasets.py | 26 ++ .../argilla_server/api/handlers/v1/jobs.py | 52 +++ .../api/policies/v1/__init__.py | 2 + .../api/policies/v1/dataset_policy.py | 7 + .../api/policies/v1/job_policy.py | 21 ++ .../src/argilla_server/api/routes.py | 2 + .../argilla_server/api/schemas/v1/datasets.py | 6 + .../src/argilla_server/api/schemas/v1/jobs.py | 21 ++ .../src/argilla_server/contexts/hub.py | 90 +++++ .../src/argilla_server/jobs/dataset_jobs.py | 2 +- .../src/argilla_server/jobs/hub_jobs.py | 48 +++ .../unit/contexts/hub/test_hub_dataset.py | 61 ++++ 14 files changed, 493 insertions(+), 178 deletions(-) create mode 100644 argilla-server/src/argilla_server/api/handlers/v1/jobs.py create mode 100644 argilla-server/src/argilla_server/api/policies/v1/job_policy.py create mode 100644 argilla-server/src/argilla_server/api/schemas/v1/jobs.py create mode 100644 argilla-server/src/argilla_server/contexts/hub.py create mode 100644 argilla-server/src/argilla_server/jobs/hub_jobs.py create mode 100644 argilla-server/tests/unit/contexts/hub/test_hub_dataset.py diff --git a/argilla-server/pdm.lock b/argilla-server/pdm.lock index 6a6e00562c..bd32bc177c 100644 --- a/argilla-server/pdm.lock +++ b/argilla-server/pdm.lock @@ -5,7 +5,7 @@ groups = ["default", "postgresql", "test"] strategy = ["inherit_metadata"] lock_version = "4.5.0" -content_hash = "sha256:c333424e19e30dc22ae7475a8f8cec7c965c90d6d551b7efef2a724fd7354245" +content_hash = "sha256:3644c449b4ba725a772eff525d5f3a21aa2159935370f203ae14d1bb14dd0243" [[metadata.targets]] requires_python = ">=3.8,<3.11" @@ -26,7 +26,7 @@ name = "aiohttp" version = "3.9.1" requires_python = ">=3.8" summary = "Async http client/server framework (asyncio)" -groups = ["default", "test"] +groups = ["default"] dependencies = [ "aiosignal>=1.1.2", "async-timeout<5.0,>=4.0; python_version < \"3.11\"", @@ -89,7 +89,7 @@ name = "aiosignal" version = "1.3.1" requires_python = ">=3.7" summary = "aiosignal: a list of registered asynchronous callbacks" -groups = ["default", "test"] +groups = ["default"] dependencies = [ "frozenlist>=1.1.0", ] @@ -151,7 +151,7 @@ name = "async-timeout" version = "4.0.3" requires_python = ">=3.7" summary = "Timeout context manager for asyncio programs" -groups = ["default", "postgresql", "test"] +groups = ["default", "postgresql"] marker = "python_version < \"3.12.0\"" dependencies = [ "typing-extensions>=3.6.5; python_version < \"3.8\"", @@ -203,7 +203,7 @@ name = "attrs" version = "23.2.0" requires_python = ">=3.7" summary = "Classes Without Boilerplate" -groups = ["default", "test"] +groups = ["default"] dependencies = [ "importlib-metadata; python_version < \"3.8\"", ] @@ -663,30 +663,29 @@ files = [ [[package]] name = "datasets" -version = "2.16.1" +version = "3.0.1" requires_python = ">=3.8.0" summary = "HuggingFace community-driven open-source library of datasets" -groups = ["test"] +groups = ["default"] dependencies = [ "aiohttp", - "dill<0.3.8,>=0.3.0", + "dill<0.3.9,>=0.3.0", "filelock", - "fsspec[http]<=2023.10.0,>=2023.1.0", - "huggingface-hub>=0.19.4", + "fsspec[http]<=2024.6.1,>=2023.1.0", + "huggingface-hub>=0.22.0", "multiprocess", "numpy>=1.17", "packaging", "pandas", - "pyarrow-hotfix", - "pyarrow>=8.0.0", + "pyarrow>=15.0.0", "pyyaml>=5.1", - "requests>=2.19.0", - "tqdm>=4.62.1", + "requests>=2.32.2", + "tqdm>=4.66.3", "xxhash", ] files = [ - {file = "datasets-2.16.1-py3-none-any.whl", hash = "sha256:fafa300c78ff92d521473a3d47d60c2d3e0d6046212cc03ceb6caf6550737257"}, - {file = "datasets-2.16.1.tar.gz", hash = "sha256:ad3215e9b1984d1de4fda2123bc7319ccbdf1e17d0c3d5590d13debff308a080"}, + {file = "datasets-3.0.1-py3-none-any.whl", hash = "sha256:db080aab41c8cc68645117a0f172e5c6789cbc672f066de0aa5a08fc3eebc686"}, + {file = "datasets-3.0.1.tar.gz", hash = "sha256:40d63b09e76a3066c32e746d6fdc36fd3f29ed2acd49bf5b1a2100da32936511"}, ] [[package]] @@ -702,13 +701,13 @@ files = [ [[package]] name = "dill" -version = "0.3.7" -requires_python = ">=3.7" +version = "0.3.8" +requires_python = ">=3.8" summary = "serialize all of Python" -groups = ["test"] +groups = ["default"] files = [ - {file = "dill-0.3.7-py3-none-any.whl", hash = "sha256:76b122c08ef4ce2eedcd4d1abd8e641114bfc6c2867f49f3c41facf65bf19f5e"}, - {file = "dill-0.3.7.tar.gz", hash = "sha256:cc1c8b182eb3013e24bd475ff2e9295af86c1a38eb1aff128dac8962a9ce3c03"}, + {file = "dill-0.3.8-py3-none-any.whl", hash = "sha256:c36ca9ffb54365bdd2f8eb3eff7d2a21237f8452b57ace88b1ac615b7e815bd7"}, + {file = "dill-0.3.8.tar.gz", hash = "sha256:3ebe3c479ad625c4553aca177444d89b486b1d84982eeacded644afc0cf797ca"}, ] [[package]] @@ -834,7 +833,7 @@ name = "filelock" version = "3.13.1" requires_python = ">=3.8" summary = "A platform independent file lock." -groups = ["default", "test"] +groups = ["default"] files = [ {file = "filelock-3.13.1-py3-none-any.whl", hash = "sha256:57dbda9b35157b05fb3e58ee91448612eb674172fab98ee235ccb0b5bee19a1c"}, {file = "filelock-3.13.1.tar.gz", hash = "sha256:521f5f56c50f8426f5e03ad3b281b490a87ef15bc6c526f168290f0c7148d44e"}, @@ -845,7 +844,7 @@ name = "frozenlist" version = "1.4.1" requires_python = ">=3.8" summary = "A list-like structure which implements collections.abc.MutableSequence" -groups = ["default", "test"] +groups = ["default"] files = [ {file = "frozenlist-1.4.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:f9aa1878d1083b276b0196f2dfbe00c9b7e752475ed3b682025ff20c1c1f51ac"}, {file = "frozenlist-1.4.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:29acab3f66f0f24674b7dc4736477bcd4bc3ad4b896f5f45379a67bce8b96868"}, @@ -898,30 +897,29 @@ files = [ [[package]] name = "fsspec" -version = "2023.10.0" +version = "2024.6.1" requires_python = ">=3.8" summary = "File-system specification" -groups = ["default", "test"] +groups = ["default"] files = [ - {file = "fsspec-2023.10.0-py3-none-any.whl", hash = "sha256:346a8f024efeb749d2a5fca7ba8854474b1ff9af7c3faaf636a4548781136529"}, - {file = "fsspec-2023.10.0.tar.gz", hash = "sha256:330c66757591df346ad3091a53bd907e15348c2ba17d63fd54f5c39c4457d2a5"}, + {file = "fsspec-2024.6.1-py3-none-any.whl", hash = "sha256:3cb443f8bcd2efb31295a5b9fdb02aee81d8452c80d28f97a6d0959e6cee101e"}, + {file = "fsspec-2024.6.1.tar.gz", hash = "sha256:fad7d7e209dd4c1208e3bbfda706620e0da5142bebbd9c384afb95b07e798e49"}, ] [[package]] name = "fsspec" -version = "2023.10.0" +version = "2024.6.1" extras = ["http"] requires_python = ">=3.8" summary = "File-system specification" -groups = ["test"] +groups = ["default"] dependencies = [ "aiohttp!=4.0.0a0,!=4.0.0a1", - "fsspec==2023.10.0", - "requests", + "fsspec==2024.6.1", ] files = [ - {file = "fsspec-2023.10.0-py3-none-any.whl", hash = "sha256:346a8f024efeb749d2a5fca7ba8854474b1ff9af7c3faaf636a4548781136529"}, - {file = "fsspec-2023.10.0.tar.gz", hash = "sha256:330c66757591df346ad3091a53bd907e15348c2ba17d63fd54f5c39c4457d2a5"}, + {file = "fsspec-2024.6.1-py3-none-any.whl", hash = "sha256:3cb443f8bcd2efb31295a5b9fdb02aee81d8452c80d28f97a6d0959e6cee101e"}, + {file = "fsspec-2024.6.1.tar.gz", hash = "sha256:fad7d7e209dd4c1208e3bbfda706620e0da5142bebbd9c384afb95b07e798e49"}, ] [[package]] @@ -1043,10 +1041,10 @@ files = [ [[package]] name = "huggingface-hub" -version = "0.20.2" +version = "0.25.2" requires_python = ">=3.8.0" summary = "Client library to download and publish models, datasets and other repos on the huggingface.co hub" -groups = ["default", "test"] +groups = ["default"] dependencies = [ "filelock", "fsspec>=2023.5.0", @@ -1057,8 +1055,8 @@ dependencies = [ "typing-extensions>=3.7.4.3", ] files = [ - {file = "huggingface_hub-0.20.2-py3-none-any.whl", hash = "sha256:53752eda2239d30a470c307a61cf9adcf136bc77b0a734338c7d04941af560d8"}, - {file = "huggingface_hub-0.20.2.tar.gz", hash = "sha256:215c5fceff631030c7a3d19ba7b588921c908b3f21eef31d160ebc245b200ff6"}, + {file = "huggingface_hub-0.25.2-py3-none-any.whl", hash = "sha256:1897caf88ce7f97fe0110603d8f66ac264e3ba6accdf30cd66cc0fed5282ad25"}, + {file = "huggingface_hub-0.25.2.tar.gz", hash = "sha256:a1014ea111a5f40ccd23f7f7ba8ac46e20fa3b658ced1f86a00c75c06ec6423c"}, ] [[package]] @@ -1244,7 +1242,7 @@ name = "multidict" version = "6.0.4" requires_python = ">=3.7" summary = "multidict implementation" -groups = ["default", "test"] +groups = ["default"] files = [ {file = "multidict-6.0.4-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:0b1a97283e0c85772d613878028fec909f003993e1007eafa715b24b377cb9b8"}, {file = "multidict-6.0.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:eeb6dcc05e911516ae3d1f207d4b0520d07f54484c49dfc294d6e7d63b734171"}, @@ -1296,28 +1294,24 @@ files = [ [[package]] name = "multiprocess" -version = "0.70.15" -requires_python = ">=3.7" +version = "0.70.16" +requires_python = ">=3.8" summary = "better multiprocessing and multithreading in Python" -groups = ["test"] +groups = ["default"] dependencies = [ - "dill>=0.3.7", + "dill>=0.3.8", ] files = [ - {file = "multiprocess-0.70.15-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:aa36c7ed16f508091438687fe9baa393a7a8e206731d321e443745e743a0d4e5"}, - {file = "multiprocess-0.70.15-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:20e024018c46d0d1602024c613007ac948f9754659e3853b0aa705e83f6931d8"}, - {file = "multiprocess-0.70.15-pp37-pypy37_pp73-manylinux_2_24_i686.whl", hash = "sha256:e576062981c91f0fe8a463c3d52506e598dfc51320a8dd8d78b987dfca91c5db"}, - {file = "multiprocess-0.70.15-pp37-pypy37_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:e73f497e6696a0f5433ada2b3d599ae733b87a6e8b008e387c62ac9127add177"}, - {file = "multiprocess-0.70.15-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:73db2e7b32dcc7f9b0f075c2ffa45c90b6729d3f1805f27e88534c8d321a1be5"}, - {file = "multiprocess-0.70.15-pp38-pypy38_pp73-manylinux_2_24_i686.whl", hash = "sha256:4271647bd8a49c28ecd6eb56a7fdbd3c212c45529ad5303b40b3c65fc6928e5f"}, - {file = "multiprocess-0.70.15-pp38-pypy38_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:cf981fb998d6ec3208cb14f0cf2e9e80216e834f5d51fd09ebc937c32b960902"}, - {file = "multiprocess-0.70.15-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:18f9f2c7063346d1617bd1684fdcae8d33380ae96b99427260f562e1a1228b67"}, - {file = "multiprocess-0.70.15-pp39-pypy39_pp73-manylinux_2_24_i686.whl", hash = "sha256:0eac53214d664c49a34695e5824872db4006b1a465edd7459a251809c3773370"}, - {file = "multiprocess-0.70.15-pp39-pypy39_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:1a51dd34096db47fb21fa2b839e615b051d51b97af9a67afbcdaa67186b44883"}, - {file = "multiprocess-0.70.15-py310-none-any.whl", hash = "sha256:7dd58e33235e83cf09d625e55cffd7b0f0eede7ee9223cdd666a87624f60c21a"}, - {file = "multiprocess-0.70.15-py38-none-any.whl", hash = "sha256:bee9afba476c91f9ebee7beeee0601face9eff67d822e893f9a893725fbd6316"}, - {file = "multiprocess-0.70.15-py39-none-any.whl", hash = "sha256:3e0953f5d52b4c76f1c973eaf8214554d146f2be5decb48e928e55c7a2d19338"}, - {file = "multiprocess-0.70.15.tar.gz", hash = "sha256:f20eed3036c0ef477b07a4177cf7c1ba520d9a2677870a4f47fe026f0cd6787e"}, + {file = "multiprocess-0.70.16-pp310-pypy310_pp73-macosx_10_13_x86_64.whl", hash = "sha256:476887be10e2f59ff183c006af746cb6f1fd0eadcfd4ef49e605cbe2659920ee"}, + {file = "multiprocess-0.70.16-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:d951bed82c8f73929ac82c61f01a7b5ce8f3e5ef40f5b52553b4f547ce2b08ec"}, + {file = "multiprocess-0.70.16-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:37b55f71c07e2d741374998c043b9520b626a8dddc8b3129222ca4f1a06ef67a"}, + {file = "multiprocess-0.70.16-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:ba8c31889abf4511c7308a8c52bb4a30b9d590e7f58523302ba00237702ca054"}, + {file = "multiprocess-0.70.16-pp39-pypy39_pp73-macosx_10_13_x86_64.whl", hash = "sha256:0dfd078c306e08d46d7a8d06fb120313d87aa43af60d66da43ffff40b44d2f41"}, + {file = "multiprocess-0.70.16-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:e7b9d0f307cd9bd50851afaac0dba2cb6c44449efff697df7c7645f7d3f2be3a"}, + {file = "multiprocess-0.70.16-py310-none-any.whl", hash = "sha256:c4a9944c67bd49f823687463660a2d6daae94c289adff97e0f9d696ba6371d02"}, + {file = "multiprocess-0.70.16-py38-none-any.whl", hash = "sha256:a71d82033454891091a226dfc319d0cfa8019a4e888ef9ca910372a446de4435"}, + {file = "multiprocess-0.70.16-py39-none-any.whl", hash = "sha256:a0bafd3ae1b732eac64be2e72038231c1ba97724b60b09400d68f229fcc2fbf3"}, + {file = "multiprocess-0.70.16.tar.gz", hash = "sha256:161af703d4652a0e1410be6abccecde4a7ddffd19341be0a7011b94aeb171ac1"}, ] [[package]] @@ -1419,7 +1413,7 @@ name = "pandas" version = "2.0.3" requires_python = ">=3.8" summary = "Powerful data structures for data analysis, time series, and statistics" -groups = ["test"] +groups = ["default"] dependencies = [ "numpy>=1.20.3; python_version < \"3.10\"", "numpy>=1.21.0; python_version >= \"3.10\"", @@ -1623,47 +1617,36 @@ files = [ [[package]] name = "pyarrow" -version = "14.0.2" +version = "17.0.0" requires_python = ">=3.8" summary = "Python library for Apache Arrow" -groups = ["test"] +groups = ["default"] dependencies = [ "numpy>=1.16.6", ] files = [ - {file = "pyarrow-14.0.2-cp310-cp310-macosx_10_14_x86_64.whl", hash = "sha256:ba9fe808596c5dbd08b3aeffe901e5f81095baaa28e7d5118e01354c64f22807"}, - {file = "pyarrow-14.0.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:22a768987a16bb46220cef490c56c671993fbee8fd0475febac0b3e16b00a10e"}, - {file = "pyarrow-14.0.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2dbba05e98f247f17e64303eb876f4a80fcd32f73c7e9ad975a83834d81f3fda"}, - {file = "pyarrow-14.0.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a898d134d00b1eca04998e9d286e19653f9d0fcb99587310cd10270907452a6b"}, - {file = "pyarrow-14.0.2-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:87e879323f256cb04267bb365add7208f302df942eb943c93a9dfeb8f44840b1"}, - {file = "pyarrow-14.0.2-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:76fc257559404ea5f1306ea9a3ff0541bf996ff3f7b9209fc517b5e83811fa8e"}, - {file = "pyarrow-14.0.2-cp310-cp310-win_amd64.whl", hash = "sha256:b0c4a18e00f3a32398a7f31da47fefcd7a927545b396e1f15d0c85c2f2c778cd"}, - {file = "pyarrow-14.0.2-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:e354fba8490de258be7687f341bc04aba181fc8aa1f71e4584f9890d9cb2dec2"}, - {file = "pyarrow-14.0.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:20e003a23a13da963f43e2b432483fdd8c38dc8882cd145f09f21792e1cf22a1"}, - {file = "pyarrow-14.0.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fc0de7575e841f1595ac07e5bc631084fd06ca8b03c0f2ecece733d23cd5102a"}, - {file = "pyarrow-14.0.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:66e986dc859712acb0bd45601229021f3ffcdfc49044b64c6d071aaf4fa49e98"}, - {file = "pyarrow-14.0.2-cp38-cp38-manylinux_2_28_aarch64.whl", hash = "sha256:f7d029f20ef56673a9730766023459ece397a05001f4e4d13805111d7c2108c0"}, - {file = "pyarrow-14.0.2-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:209bac546942b0d8edc8debda248364f7f668e4aad4741bae58e67d40e5fcf75"}, - {file = "pyarrow-14.0.2-cp38-cp38-win_amd64.whl", hash = "sha256:1e6987c5274fb87d66bb36816afb6f65707546b3c45c44c28e3c4133c010a881"}, - {file = "pyarrow-14.0.2-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:a01d0052d2a294a5f56cc1862933014e696aa08cc7b620e8c0cce5a5d362e976"}, - {file = "pyarrow-14.0.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:a51fee3a7db4d37f8cda3ea96f32530620d43b0489d169b285d774da48ca9785"}, - {file = "pyarrow-14.0.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:64df2bf1ef2ef14cee531e2dfe03dd924017650ffaa6f9513d7a1bb291e59c15"}, - {file = "pyarrow-14.0.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3c0fa3bfdb0305ffe09810f9d3e2e50a2787e3a07063001dcd7adae0cee3601a"}, - {file = "pyarrow-14.0.2-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:c65bf4fd06584f058420238bc47a316e80dda01ec0dfb3044594128a6c2db794"}, - {file = "pyarrow-14.0.2-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:63ac901baec9369d6aae1cbe6cca11178fb018a8d45068aaf5bb54f94804a866"}, - {file = "pyarrow-14.0.2-cp39-cp39-win_amd64.whl", hash = "sha256:75ee0efe7a87a687ae303d63037d08a48ef9ea0127064df18267252cfe2e9541"}, - {file = "pyarrow-14.0.2.tar.gz", hash = "sha256:36cef6ba12b499d864d1def3e990f97949e0b79400d08b7cf74504ffbd3eb025"}, -] - -[[package]] -name = "pyarrow-hotfix" -version = "0.6" -requires_python = ">=3.5" -summary = "" -groups = ["test"] -files = [ - {file = "pyarrow_hotfix-0.6-py3-none-any.whl", hash = "sha256:dcc9ae2d220dff0083be6a9aa8e0cdee5182ad358d4931fce825c545e5c89178"}, - {file = "pyarrow_hotfix-0.6.tar.gz", hash = "sha256:79d3e030f7ff890d408a100ac16d6f00b14d44a502d7897cd9fc3e3a534e9945"}, + {file = "pyarrow-17.0.0-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:a5c8b238d47e48812ee577ee20c9a2779e6a5904f1708ae240f53ecbee7c9f07"}, + {file = "pyarrow-17.0.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:db023dc4c6cae1015de9e198d41250688383c3f9af8f565370ab2b4cb5f62655"}, + {file = "pyarrow-17.0.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da1e060b3876faa11cee287839f9cc7cdc00649f475714b8680a05fd9071d545"}, + {file = "pyarrow-17.0.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:75c06d4624c0ad6674364bb46ef38c3132768139ddec1c56582dbac54f2663e2"}, + {file = "pyarrow-17.0.0-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:fa3c246cc58cb5a4a5cb407a18f193354ea47dd0648194e6265bd24177982fe8"}, + {file = "pyarrow-17.0.0-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:f7ae2de664e0b158d1607699a16a488de3d008ba99b3a7aa5de1cbc13574d047"}, + {file = "pyarrow-17.0.0-cp310-cp310-win_amd64.whl", hash = "sha256:5984f416552eea15fd9cee03da53542bf4cddaef5afecefb9aa8d1010c335087"}, + {file = "pyarrow-17.0.0-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:af5ff82a04b2171415f1410cff7ebb79861afc5dae50be73ce06d6e870615204"}, + {file = "pyarrow-17.0.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:edca18eaca89cd6382dfbcff3dd2d87633433043650c07375d095cd3517561d8"}, + {file = "pyarrow-17.0.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7c7916bff914ac5d4a8fe25b7a25e432ff921e72f6f2b7547d1e325c1ad9d155"}, + {file = "pyarrow-17.0.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f553ca691b9e94b202ff741bdd40f6ccb70cdd5fbf65c187af132f1317de6145"}, + {file = "pyarrow-17.0.0-cp38-cp38-manylinux_2_28_aarch64.whl", hash = "sha256:0cdb0e627c86c373205a2f94a510ac4376fdc523f8bb36beab2e7f204416163c"}, + {file = "pyarrow-17.0.0-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:d7d192305d9d8bc9082d10f361fc70a73590a4c65cf31c3e6926cd72b76bc35c"}, + {file = "pyarrow-17.0.0-cp38-cp38-win_amd64.whl", hash = "sha256:02dae06ce212d8b3244dd3e7d12d9c4d3046945a5933d28026598e9dbbda1fca"}, + {file = "pyarrow-17.0.0-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:13d7a460b412f31e4c0efa1148e1d29bdf18ad1411eb6757d38f8fbdcc8645fb"}, + {file = "pyarrow-17.0.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:9b564a51fbccfab5a04a80453e5ac6c9954a9c5ef2890d1bcf63741909c3f8df"}, + {file = "pyarrow-17.0.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:32503827abbc5aadedfa235f5ece8c4f8f8b0a3cf01066bc8d29de7539532687"}, + {file = "pyarrow-17.0.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a155acc7f154b9ffcc85497509bcd0d43efb80d6f733b0dc3bb14e281f131c8b"}, + {file = "pyarrow-17.0.0-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:dec8d129254d0188a49f8a1fc99e0560dc1b85f60af729f47de4046015f9b0a5"}, + {file = "pyarrow-17.0.0-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:a48ddf5c3c6a6c505904545c25a4ae13646ae1f8ba703c4df4a1bfe4f4006bda"}, + {file = "pyarrow-17.0.0-cp39-cp39-win_amd64.whl", hash = "sha256:42bf93249a083aca230ba7e2786c5f673507fa97bbd9725a1e2754715151a204"}, + {file = "pyarrow-17.0.0.tar.gz", hash = "sha256:4beca9521ed2c0921c1023e68d097d0299b62c362639ea315572a58f3f50fd28"}, ] [[package]] @@ -1923,12 +1906,12 @@ files = [ [[package]] name = "pytz" -version = "2023.3.post1" +version = "2024.2" summary = "World timezone definitions, modern and historical" -groups = ["test"] +groups = ["default"] files = [ - {file = "pytz-2023.3.post1-py2.py3-none-any.whl", hash = "sha256:ce42d816b81b68506614c11e8937d3aa9e41007ceb50bfdcb0749b921bf646c7"}, - {file = "pytz-2023.3.post1.tar.gz", hash = "sha256:7b4fddbeb94a1eba4b557da24f19fdf9db575192544270a9101d8509f9f43d7b"}, + {file = "pytz-2024.2-py2.py3-none-any.whl", hash = "sha256:31c7c1817eb7fae7ca4b8c7ee50c72f93aa2dd863de768e1ef4245d426aa0725"}, + {file = "pytz-2024.2.tar.gz", hash = "sha256:2aa355083c50a0f93fa581709deac0c9ad65cca8a9e9beac660adcbd493c798a"}, ] [[package]] @@ -1936,7 +1919,7 @@ name = "pyyaml" version = "6.0.1" requires_python = ">=3.6" summary = "YAML parser and emitter for Python" -groups = ["default", "test"] +groups = ["default"] files = [ {file = "PyYAML-6.0.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d858aa552c999bc8a8d57426ed01e40bef403cd8ccdd0fc5f6f04a00414cac2a"}, {file = "PyYAML-6.0.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:fd66fc5d0da6d9815ba2cebeb4205f95818ff4b79c3ebe268e75d961704af52f"}, @@ -1982,8 +1965,8 @@ files = [ [[package]] name = "requests" -version = "2.31.0" -requires_python = ">=3.7" +version = "2.32.3" +requires_python = ">=3.8" summary = "Python HTTP for Humans." groups = ["default", "test"] dependencies = [ @@ -1993,8 +1976,8 @@ dependencies = [ "urllib3<3,>=1.21.1", ] files = [ - {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, - {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, + {file = "requests-2.32.3-py3-none-any.whl", hash = "sha256:70761cfe03c773ceb22aa2f671b4757976145175cdfca038c02654d061d6dcc6"}, + {file = "requests-2.32.3.tar.gz", hash = "sha256:55365417734eb18255590a9ff9eb97e9e1da868d4ccd6402399eaf68af20a760"}, ] [[package]] @@ -2418,7 +2401,7 @@ files = [ [[package]] name = "tqdm" -version = "4.66.1" +version = "4.66.5" requires_python = ">=3.7" summary = "Fast, Extensible Progress Meter" groups = ["default", "test"] @@ -2426,8 +2409,8 @@ dependencies = [ "colorama; platform_system == \"Windows\"", ] files = [ - {file = "tqdm-4.66.1-py3-none-any.whl", hash = "sha256:d302b3c5b53d47bce91fea46679d9c3c6508cf6332229aa1e7d8653723793386"}, - {file = "tqdm-4.66.1.tar.gz", hash = "sha256:d88e651f9db8d8551a62556d3cff9e3034274ca5d66e93197cf2490e2dcb69c7"}, + {file = "tqdm-4.66.5-py3-none-any.whl", hash = "sha256:90279a3770753eafc9194a0364852159802111925aa30eb3f9d85b0e805ac7cd"}, + {file = "tqdm-4.66.5.tar.gz", hash = "sha256:e1020aef2e5096702d8a025ac7d16b1577279c9d63f8375b63083e9a5f0fcbad"}, ] [[package]] @@ -2458,13 +2441,13 @@ files = [ [[package]] name = "tzdata" -version = "2023.4" +version = "2024.2" requires_python = ">=2" summary = "Provider of IANA time zone data" -groups = ["test"] +groups = ["default"] files = [ - {file = "tzdata-2023.4-py2.py3-none-any.whl", hash = "sha256:aa3ace4329eeacda5b7beb7ea08ece826c28d761cda36e747cfbf97996d39bf3"}, - {file = "tzdata-2023.4.tar.gz", hash = "sha256:dd54c94f294765522c77399649b4fefd95522479a664a0cec87f41bebc6148c9"}, + {file = "tzdata-2024.2-py2.py3-none-any.whl", hash = "sha256:a48093786cdcde33cad18c2555e8532f34422074448fbc874186f0abd79565cd"}, + {file = "tzdata-2024.2.tar.gz", hash = "sha256:7d85cc416e9382e69095b7bdf4afd9e3880418a2413feec7069d533d6b4e31cc"}, ] [[package]] @@ -2682,76 +2665,71 @@ files = [ [[package]] name = "xxhash" -version = "3.4.1" +version = "3.5.0" requires_python = ">=3.7" summary = "Python binding for xxHash" -groups = ["test"] +groups = ["default"] files = [ - {file = "xxhash-3.4.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:91dbfa55346ad3e18e738742236554531a621042e419b70ad8f3c1d9c7a16e7f"}, - {file = "xxhash-3.4.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:665a65c2a48a72068fcc4d21721510df5f51f1142541c890491afc80451636d2"}, - {file = "xxhash-3.4.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bb11628470a6004dc71a09fe90c2f459ff03d611376c1debeec2d648f44cb693"}, - {file = "xxhash-3.4.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5bef2a7dc7b4f4beb45a1edbba9b9194c60a43a89598a87f1a0226d183764189"}, - {file = "xxhash-3.4.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9c0f7b2d547d72c7eda7aa817acf8791f0146b12b9eba1d4432c531fb0352228"}, - {file = "xxhash-3.4.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:00f2fdef6b41c9db3d2fc0e7f94cb3db86693e5c45d6de09625caad9a469635b"}, - {file = "xxhash-3.4.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:23cfd9ca09acaf07a43e5a695143d9a21bf00f5b49b15c07d5388cadf1f9ce11"}, - {file = "xxhash-3.4.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:6a9ff50a3cf88355ca4731682c168049af1ca222d1d2925ef7119c1a78e95b3b"}, - {file = "xxhash-3.4.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:f1d7c69a1e9ca5faa75546fdd267f214f63f52f12692f9b3a2f6467c9e67d5e7"}, - {file = "xxhash-3.4.1-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:672b273040d5d5a6864a36287f3514efcd1d4b1b6a7480f294c4b1d1ee1b8de0"}, - {file = "xxhash-3.4.1-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:4178f78d70e88f1c4a89ff1ffe9f43147185930bb962ee3979dba15f2b1cc799"}, - {file = "xxhash-3.4.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:9804b9eb254d4b8cc83ab5a2002128f7d631dd427aa873c8727dba7f1f0d1c2b"}, - {file = "xxhash-3.4.1-cp310-cp310-win32.whl", hash = "sha256:c09c49473212d9c87261d22c74370457cfff5db2ddfc7fd1e35c80c31a8c14ce"}, - {file = "xxhash-3.4.1-cp310-cp310-win_amd64.whl", hash = "sha256:ebbb1616435b4a194ce3466d7247df23499475c7ed4eb2681a1fa42ff766aff6"}, - {file = "xxhash-3.4.1-cp310-cp310-win_arm64.whl", hash = "sha256:25dc66be3db54f8a2d136f695b00cfe88018e59ccff0f3b8f545869f376a8a46"}, - {file = "xxhash-3.4.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:961d948b7b1c1b6c08484bbce3d489cdf153e4122c3dfb07c2039621243d8795"}, - {file = "xxhash-3.4.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:719a378930504ab159f7b8e20fa2aa1896cde050011af838af7e7e3518dd82de"}, - {file = "xxhash-3.4.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:74fb5cb9406ccd7c4dd917f16630d2e5e8cbbb02fc2fca4e559b2a47a64f4940"}, - {file = "xxhash-3.4.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5dab508ac39e0ab988039bc7f962c6ad021acd81fd29145962b068df4148c476"}, - {file = "xxhash-3.4.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8c59f3e46e7daf4c589e8e853d700ef6607afa037bfad32c390175da28127e8c"}, - {file = "xxhash-3.4.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8cc07256eff0795e0f642df74ad096f8c5d23fe66bc138b83970b50fc7f7f6c5"}, - {file = "xxhash-3.4.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e9f749999ed80f3955a4af0eb18bb43993f04939350b07b8dd2f44edc98ffee9"}, - {file = "xxhash-3.4.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:7688d7c02149a90a3d46d55b341ab7ad1b4a3f767be2357e211b4e893efbaaf6"}, - {file = "xxhash-3.4.1-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a8b4977963926f60b0d4f830941c864bed16aa151206c01ad5c531636da5708e"}, - {file = "xxhash-3.4.1-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:8106d88da330f6535a58a8195aa463ef5281a9aa23b04af1848ff715c4398fb4"}, - {file = "xxhash-3.4.1-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:4c76a77dbd169450b61c06fd2d5d436189fc8ab7c1571d39265d4822da16df22"}, - {file = "xxhash-3.4.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:11f11357c86d83e53719c592021fd524efa9cf024dc7cb1dfb57bbbd0d8713f2"}, - {file = "xxhash-3.4.1-cp38-cp38-win32.whl", hash = "sha256:0c786a6cd74e8765c6809892a0d45886e7c3dc54de4985b4a5eb8b630f3b8e3b"}, - {file = "xxhash-3.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:aabf37fb8fa27430d50507deeab2ee7b1bcce89910dd10657c38e71fee835594"}, - {file = "xxhash-3.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6127813abc1477f3a83529b6bbcfeddc23162cece76fa69aee8f6a8a97720562"}, - {file = "xxhash-3.4.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ef2e194262f5db16075caea7b3f7f49392242c688412f386d3c7b07c7733a70a"}, - {file = "xxhash-3.4.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:71be94265b6c6590f0018bbf73759d21a41c6bda20409782d8117e76cd0dfa8b"}, - {file = "xxhash-3.4.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:10e0a619cdd1c0980e25eb04e30fe96cf8f4324758fa497080af9c21a6de573f"}, - {file = "xxhash-3.4.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fa122124d2e3bd36581dd78c0efa5f429f5220313479fb1072858188bc2d5ff1"}, - {file = "xxhash-3.4.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e17032f5a4fea0a074717fe33477cb5ee723a5f428de7563e75af64bfc1b1e10"}, - {file = "xxhash-3.4.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ca7783b20e3e4f3f52f093538895863f21d18598f9a48211ad757680c3bd006f"}, - {file = "xxhash-3.4.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d77d09a1113899fad5f354a1eb4f0a9afcf58cefff51082c8ad643ff890e30cf"}, - {file = "xxhash-3.4.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:21287bcdd299fdc3328cc0fbbdeaa46838a1c05391264e51ddb38a3f5b09611f"}, - {file = "xxhash-3.4.1-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:dfd7a6cc483e20b4ad90224aeb589e64ec0f31e5610ab9957ff4314270b2bf31"}, - {file = "xxhash-3.4.1-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:543c7fcbc02bbb4840ea9915134e14dc3dc15cbd5a30873a7a5bf66039db97ec"}, - {file = "xxhash-3.4.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:fe0a98d990e433013f41827b62be9ab43e3cf18e08b1483fcc343bda0d691182"}, - {file = "xxhash-3.4.1-cp39-cp39-win32.whl", hash = "sha256:b9097af00ebf429cc7c0e7d2fdf28384e4e2e91008130ccda8d5ae653db71e54"}, - {file = "xxhash-3.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:d699b921af0dcde50ab18be76c0d832f803034d80470703700cb7df0fbec2832"}, - {file = "xxhash-3.4.1-cp39-cp39-win_arm64.whl", hash = "sha256:2be491723405e15cc099ade1280133ccfbf6322d2ef568494fb7d07d280e7eee"}, - {file = "xxhash-3.4.1-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:431625fad7ab5649368c4849d2b49a83dc711b1f20e1f7f04955aab86cd307bc"}, - {file = "xxhash-3.4.1-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fc6dbd5fc3c9886a9e041848508b7fb65fd82f94cc793253990f81617b61fe49"}, - {file = "xxhash-3.4.1-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f3ff8dbd0ec97aec842476cb8ccc3e17dd288cd6ce3c8ef38bff83d6eb927817"}, - {file = "xxhash-3.4.1-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ef73a53fe90558a4096e3256752268a8bdc0322f4692ed928b6cd7ce06ad4fe3"}, - {file = "xxhash-3.4.1-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:450401f42bbd274b519d3d8dcf3c57166913381a3d2664d6609004685039f9d3"}, - {file = "xxhash-3.4.1-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:a162840cf4de8a7cd8720ff3b4417fbc10001eefdd2d21541a8226bb5556e3bb"}, - {file = "xxhash-3.4.1-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b736a2a2728ba45017cb67785e03125a79d246462dfa892d023b827007412c52"}, - {file = "xxhash-3.4.1-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1d0ae4c2e7698adef58710d6e7a32ff518b66b98854b1c68e70eee504ad061d8"}, - {file = "xxhash-3.4.1-pp37-pypy37_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d6322c4291c3ff174dcd104fae41500e75dad12be6f3085d119c2c8a80956c51"}, - {file = "xxhash-3.4.1-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:dd59ed668801c3fae282f8f4edadf6dc7784db6d18139b584b6d9677ddde1b6b"}, - {file = "xxhash-3.4.1-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:92693c487e39523a80474b0394645b393f0ae781d8db3474ccdcead0559ccf45"}, - {file = "xxhash-3.4.1-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4603a0f642a1e8d7f3ba5c4c25509aca6a9c1cc16f85091004a7028607ead663"}, - {file = "xxhash-3.4.1-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6fa45e8cbfbadb40a920fe9ca40c34b393e0b067082d94006f7f64e70c7490a6"}, - {file = "xxhash-3.4.1-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:595b252943b3552de491ff51e5bb79660f84f033977f88f6ca1605846637b7c6"}, - {file = "xxhash-3.4.1-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:562d8b8f783c6af969806aaacf95b6c7b776929ae26c0cd941d54644ea7ef51e"}, - {file = "xxhash-3.4.1-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:41ddeae47cf2828335d8d991f2d2b03b0bdc89289dc64349d712ff8ce59d0647"}, - {file = "xxhash-3.4.1-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c44d584afdf3c4dbb3277e32321d1a7b01d6071c1992524b6543025fb8f4206f"}, - {file = "xxhash-3.4.1-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fd7bddb3a5b86213cc3f2c61500c16945a1b80ecd572f3078ddbbe68f9dabdfb"}, - {file = "xxhash-3.4.1-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9ecb6c987b62437c2f99c01e97caf8d25660bf541fe79a481d05732e5236719c"}, - {file = "xxhash-3.4.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:696b4e18b7023527d5c50ed0626ac0520edac45a50ec7cf3fc265cd08b1f4c03"}, - {file = "xxhash-3.4.1.tar.gz", hash = "sha256:0379d6cf1ff987cd421609a264ce025e74f346e3e145dd106c0cc2e3ec3f99a9"}, + {file = "xxhash-3.5.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:ece616532c499ee9afbb83078b1b952beffef121d989841f7f4b3dc5ac0fd212"}, + {file = "xxhash-3.5.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3171f693dbc2cef6477054a665dc255d996646b4023fe56cb4db80e26f4cc520"}, + {file = "xxhash-3.5.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7c5d3e570ef46adaf93fc81b44aca6002b5a4d8ca11bd0580c07eac537f36680"}, + {file = "xxhash-3.5.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7cb29a034301e2982df8b1fe6328a84f4b676106a13e9135a0d7e0c3e9f806da"}, + {file = "xxhash-3.5.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5d0d307d27099bb0cbeea7260eb39ed4fdb99c5542e21e94bb6fd29e49c57a23"}, + {file = "xxhash-3.5.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0342aafd421795d740e514bc9858ebddfc705a75a8c5046ac56d85fe97bf196"}, + {file = "xxhash-3.5.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3dbbd9892c5ebffeca1ed620cf0ade13eb55a0d8c84e0751a6653adc6ac40d0c"}, + {file = "xxhash-3.5.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:4cc2d67fdb4d057730c75a64c5923abfa17775ae234a71b0200346bfb0a7f482"}, + {file = "xxhash-3.5.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:ec28adb204b759306a3d64358a5e5c07d7b1dd0ccbce04aa76cb9377b7b70296"}, + {file = "xxhash-3.5.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:1328f6d8cca2b86acb14104e381225a3d7b42c92c4b86ceae814e5c400dbb415"}, + {file = "xxhash-3.5.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:8d47ebd9f5d9607fd039c1fbf4994e3b071ea23eff42f4ecef246ab2b7334198"}, + {file = "xxhash-3.5.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:b96d559e0fcddd3343c510a0fe2b127fbff16bf346dd76280b82292567523442"}, + {file = "xxhash-3.5.0-cp310-cp310-win32.whl", hash = "sha256:61c722ed8d49ac9bc26c7071eeaa1f6ff24053d553146d5df031802deffd03da"}, + {file = "xxhash-3.5.0-cp310-cp310-win_amd64.whl", hash = "sha256:9bed5144c6923cc902cd14bb8963f2d5e034def4486ab0bbe1f58f03f042f9a9"}, + {file = "xxhash-3.5.0-cp310-cp310-win_arm64.whl", hash = "sha256:893074d651cf25c1cc14e3bea4fceefd67f2921b1bb8e40fcfeba56820de80c6"}, + {file = "xxhash-3.5.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:74752ecaa544657d88b1d1c94ae68031e364a4d47005a90288f3bab3da3c970f"}, + {file = "xxhash-3.5.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:dee1316133c9b463aa81aca676bc506d3f80d8f65aeb0bba2b78d0b30c51d7bd"}, + {file = "xxhash-3.5.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:602d339548d35a8579c6b013339fb34aee2df9b4e105f985443d2860e4d7ffaa"}, + {file = "xxhash-3.5.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:695735deeddfb35da1677dbc16a083445360e37ff46d8ac5c6fcd64917ff9ade"}, + {file = "xxhash-3.5.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1030a39ba01b0c519b1a82f80e8802630d16ab95dc3f2b2386a0b5c8ed5cbb10"}, + {file = "xxhash-3.5.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a5bc08f33c4966f4eb6590d6ff3ceae76151ad744576b5fc6c4ba8edd459fdec"}, + {file = "xxhash-3.5.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:160e0c19ee500482ddfb5d5570a0415f565d8ae2b3fd69c5dcfce8a58107b1c3"}, + {file = "xxhash-3.5.0-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:f1abffa122452481a61c3551ab3c89d72238e279e517705b8b03847b1d93d738"}, + {file = "xxhash-3.5.0-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:d5e9db7ef3ecbfc0b4733579cea45713a76852b002cf605420b12ef3ef1ec148"}, + {file = "xxhash-3.5.0-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:23241ff6423378a731d84864bf923a41649dc67b144debd1077f02e6249a0d54"}, + {file = "xxhash-3.5.0-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:82b833d5563fefd6fceafb1aed2f3f3ebe19f84760fdd289f8b926731c2e6e91"}, + {file = "xxhash-3.5.0-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:0a80ad0ffd78bef9509eee27b4a29e56f5414b87fb01a888353e3d5bda7038bd"}, + {file = "xxhash-3.5.0-cp38-cp38-win32.whl", hash = "sha256:50ac2184ffb1b999e11e27c7e3e70cc1139047e7ebc1aa95ed12f4269abe98d4"}, + {file = "xxhash-3.5.0-cp38-cp38-win_amd64.whl", hash = "sha256:392f52ebbb932db566973693de48f15ce787cabd15cf6334e855ed22ea0be5b3"}, + {file = "xxhash-3.5.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bfc8cdd7f33d57f0468b0614ae634cc38ab9202c6957a60e31d285a71ebe0301"}, + {file = "xxhash-3.5.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e0c48b6300cd0b0106bf49169c3e0536408dfbeb1ccb53180068a18b03c662ab"}, + {file = "xxhash-3.5.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fe1a92cfbaa0a1253e339ccec42dbe6db262615e52df591b68726ab10338003f"}, + {file = "xxhash-3.5.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:33513d6cc3ed3b559134fb307aae9bdd94d7e7c02907b37896a6c45ff9ce51bd"}, + {file = "xxhash-3.5.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:eefc37f6138f522e771ac6db71a6d4838ec7933939676f3753eafd7d3f4c40bc"}, + {file = "xxhash-3.5.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a606c8070ada8aa2a88e181773fa1ef17ba65ce5dd168b9d08038e2a61b33754"}, + {file = "xxhash-3.5.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:42eca420c8fa072cc1dd62597635d140e78e384a79bb4944f825fbef8bfeeef6"}, + {file = "xxhash-3.5.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:604253b2143e13218ff1ef0b59ce67f18b8bd1c4205d2ffda22b09b426386898"}, + {file = "xxhash-3.5.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:6e93a5ad22f434d7876665444a97e713a8f60b5b1a3521e8df11b98309bff833"}, + {file = "xxhash-3.5.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:7a46e1d6d2817ba8024de44c4fd79913a90e5f7265434cef97026215b7d30df6"}, + {file = "xxhash-3.5.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:30eb2efe6503c379b7ab99c81ba4a779748e3830241f032ab46bd182bf5873af"}, + {file = "xxhash-3.5.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:c8aa771ff2c13dd9cda8166d685d7333d389fae30a4d2bb39d63ab5775de8606"}, + {file = "xxhash-3.5.0-cp39-cp39-win32.whl", hash = "sha256:5ed9ebc46f24cf91034544b26b131241b699edbfc99ec5e7f8f3d02d6eb7fba4"}, + {file = "xxhash-3.5.0-cp39-cp39-win_amd64.whl", hash = "sha256:220f3f896c6b8d0316f63f16c077d52c412619e475f9372333474ee15133a558"}, + {file = "xxhash-3.5.0-cp39-cp39-win_arm64.whl", hash = "sha256:a7b1d8315d9b5e9f89eb2933b73afae6ec9597a258d52190944437158b49d38e"}, + {file = "xxhash-3.5.0-pp310-pypy310_pp73-macosx_10_15_x86_64.whl", hash = "sha256:2014c5b3ff15e64feecb6b713af12093f75b7926049e26a580e94dcad3c73d8c"}, + {file = "xxhash-3.5.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fab81ef75003eda96239a23eda4e4543cedc22e34c373edcaf744e721a163986"}, + {file = "xxhash-3.5.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4e2febf914ace002132aa09169cc572e0d8959d0f305f93d5828c4836f9bc5a6"}, + {file = "xxhash-3.5.0-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5d3a10609c51da2a1c0ea0293fc3968ca0a18bd73838455b5bca3069d7f8e32b"}, + {file = "xxhash-3.5.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:5a74f23335b9689b66eb6dbe2a931a88fcd7a4c2cc4b1cb0edba8ce381c7a1da"}, + {file = "xxhash-3.5.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:297595fe6138d4da2c8ce9e72a04d73e58725bb60f3a19048bc96ab2ff31c692"}, + {file = "xxhash-3.5.0-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cc1276d369452040cbb943300dc8abeedab14245ea44056a2943183822513a18"}, + {file = "xxhash-3.5.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2061188a1ba352fc699c82bff722f4baacb4b4b8b2f0c745d2001e56d0dfb514"}, + {file = "xxhash-3.5.0-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:38c384c434021e4f62b8d9ba0bc9467e14d394893077e2c66d826243025e1f81"}, + {file = "xxhash-3.5.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:e6a4dd644d72ab316b580a1c120b375890e4c52ec392d4aef3c63361ec4d77d1"}, + {file = "xxhash-3.5.0-pp39-pypy39_pp73-macosx_10_15_x86_64.whl", hash = "sha256:531af8845aaadcadf951b7e0c1345c6b9c68a990eeb74ff9acd8501a0ad6a1c9"}, + {file = "xxhash-3.5.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7ce379bcaa9fcc00f19affa7773084dd09f5b59947b3fb47a1ceb0179f91aaa1"}, + {file = "xxhash-3.5.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fd1b2281d01723f076df3c8188f43f2472248a6b63118b036e641243656b1b0f"}, + {file = "xxhash-3.5.0-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9c770750cc80e8694492244bca7251385188bc5597b6a39d98a9f30e8da984e0"}, + {file = "xxhash-3.5.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:b150b8467852e1bd844387459aa6fbe11d7f38b56e901f9f3b3e6aba0d660240"}, + {file = "xxhash-3.5.0.tar.gz", hash = "sha256:84f2caddf951c9cbf8dc2e22a89d4ccf5d86391ac6418fe81e3c67d0cf60b45f"}, ] [[package]] @@ -2759,7 +2737,7 @@ name = "yarl" version = "1.9.4" requires_python = ">=3.7" summary = "Yet another URL library" -groups = ["default", "test"] +groups = ["default"] dependencies = [ "idna>=2.0", "multidict>=4.0", diff --git a/argilla-server/pyproject.toml b/argilla-server/pyproject.toml index 2df84995bd..21e6a56192 100644 --- a/argilla-server/pyproject.toml +++ b/argilla-server/pyproject.toml @@ -59,8 +59,10 @@ dependencies = [ "typer >= 0.6.0, < 0.10.0", # spaCy only supports typer<0.10.0 "packaging>=23.2", "psycopg2-binary>=2.9.9", + "datasets>=3.0.1", # For Telemetry "huggingface_hub>=0.13,<1", + ] [project.optional-dependencies] @@ -100,7 +102,6 @@ test = [ "factory-boy~=3.2.1", "httpx>=0.26.0", # Required by tests/unit/utils/test_dependency.py but we should take a look a probably removed them - "datasets > 1.17.0,!= 2.3.2", "spacy>=3.5.0,<3.7.0", "pytest-randomly>=3.15.0", ] diff --git a/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py b/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py index e0d1ec3765..825408e329 100644 --- a/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py +++ b/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py @@ -29,6 +29,7 @@ DatasetProgress, Datasets, DatasetUpdate, + HubDataset, UsersProgress, ) from argilla_server.api.schemas.v1.fields import Field, FieldCreate, Fields @@ -38,9 +39,11 @@ MetadataPropertyCreate, ) from argilla_server.api.schemas.v1.vector_settings import VectorSettings, VectorSettingsCreate, VectorsSettings +from argilla_server.api.schemas.v1.jobs import Job as JobSchema from argilla_server.contexts import datasets from argilla_server.database import get_async_db from argilla_server.enums import DatasetStatus +from argilla_server.jobs import hub_jobs from argilla_server.models import Dataset, User from argilla_server.search_engine import ( SearchEngine, @@ -301,3 +304,26 @@ async def update_dataset( await authorize(current_user, DatasetPolicy.update(dataset)) return await datasets.update_dataset(db, dataset, dataset_update.dict(exclude_unset=True)) + + +# TODO: Maybe change /import to /import-from-hub? +@router.post("/datasets/{dataset_id}/import", status_code=status.HTTP_202_ACCEPTED, response_model=JobSchema) +async def import_dataset_from_hub( + *, + db: AsyncSession = Depends(get_async_db), + dataset_id: UUID, + hub_dataset: HubDataset, + current_user: User = Security(auth.get_current_user), +): + dataset = await Dataset.get_or_raise(db, dataset_id) + + await authorize(current_user, DatasetPolicy.import_from_hub(dataset)) + + job = hub_jobs.import_dataset_from_hub_job.delay( + name=hub_dataset.name, + subset=hub_dataset.subset, + split=hub_dataset.split, + dataset_id=dataset.id, + ) + + return JobSchema(id=job.id, status=job.get_status()) diff --git a/argilla-server/src/argilla_server/api/handlers/v1/jobs.py b/argilla-server/src/argilla_server/api/handlers/v1/jobs.py new file mode 100644 index 0000000000..dbded9497c --- /dev/null +++ b/argilla-server/src/argilla_server/api/handlers/v1/jobs.py @@ -0,0 +1,52 @@ +# Copyright 2021-present, the Recognai S.L. team. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from fastapi import APIRouter, Depends, HTTPException, Security, status +from sqlalchemy.ext.asyncio import AsyncSession + +from rq.job import Job +from rq.exceptions import NoSuchJobError + +from argilla_server.database import get_async_db +from argilla_server.jobs.queues import REDIS_CONNECTION +from argilla_server.models import User +from argilla_server.api.policies.v1 import JobPolicy, authorize +from argilla_server.api.schemas.v1.jobs import Job as JobSchema +from argilla_server.security import auth + +router = APIRouter(tags=["jobs"]) + + +def _get_job(job_id: str) -> Job: + try: + return Job.fetch(job_id, connection=REDIS_CONNECTION) + except NoSuchJobError: + raise HTTPException( + status_code=status.HTTP_404_NOT_FOUND, + detail=f"Job with id `{job_id}` not found", + ) + + +@router.get("/jobs/{job_id}", response_model=JobSchema) +async def get_job( + *, + db: AsyncSession = Depends(get_async_db), + job_id: str, + current_user: User = Security(auth.get_current_user), +): + job = _get_job(job_id) + + await authorize(current_user, JobPolicy.get) + + return JobSchema(id=job.id, status=job.get_status(refresh=True)) diff --git a/argilla-server/src/argilla_server/api/policies/v1/__init__.py b/argilla-server/src/argilla_server/api/policies/v1/__init__.py index 706b196d28..dccbad5b3d 100644 --- a/argilla-server/src/argilla_server/api/policies/v1/__init__.py +++ b/argilla-server/src/argilla_server/api/policies/v1/__init__.py @@ -24,6 +24,7 @@ from argilla_server.api.policies.v1.vector_settings_policy import VectorSettingsPolicy from argilla_server.api.policies.v1.workspace_policy import WorkspacePolicy from argilla_server.api.policies.v1.workspace_user_policy import WorkspaceUserPolicy +from argilla_server.api.policies.v1.job_policy import JobPolicy __all__ = [ "DatasetPolicy", @@ -37,6 +38,7 @@ "VectorSettingsPolicy", "WorkspacePolicy", "WorkspaceUserPolicy", + "JobPolicy", "authorize", "is_authorized", ] diff --git a/argilla-server/src/argilla_server/api/policies/v1/dataset_policy.py b/argilla-server/src/argilla_server/api/policies/v1/dataset_policy.py index 5b71f5c030..755ea0b01b 100644 --- a/argilla-server/src/argilla_server/api/policies/v1/dataset_policy.py +++ b/argilla-server/src/argilla_server/api/policies/v1/dataset_policy.py @@ -141,3 +141,10 @@ async def is_allowed(actor: User) -> bool: return actor.is_owner or (actor.is_admin and await actor.is_member(dataset.workspace_id)) return is_allowed + + @classmethod + def import_from_hub(cls, dataset: Dataset) -> PolicyAction: + async def is_allowed(actor: User) -> bool: + return actor.is_owner or (actor.is_admin and await actor.is_member(dataset.workspace_id)) + + return is_allowed diff --git a/argilla-server/src/argilla_server/api/policies/v1/job_policy.py b/argilla-server/src/argilla_server/api/policies/v1/job_policy.py new file mode 100644 index 0000000000..7cab380225 --- /dev/null +++ b/argilla-server/src/argilla_server/api/policies/v1/job_policy.py @@ -0,0 +1,21 @@ +# Copyright 2021-present, the Recognai S.L. team. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from argilla_server.models import User + + +class JobPolicy: + @classmethod + async def get(cls, actor: User) -> bool: + return actor.is_owner or actor.is_admin diff --git a/argilla-server/src/argilla_server/api/routes.py b/argilla-server/src/argilla_server/api/routes.py index 456b918c3f..8cd7244777 100644 --- a/argilla-server/src/argilla_server/api/routes.py +++ b/argilla-server/src/argilla_server/api/routes.py @@ -62,6 +62,7 @@ from argilla_server.api.handlers.v1 import ( workspaces as workspaces_v1, ) +from argilla_server.api.handlers.v1 import jobs as jobs_v1 from argilla_server.errors.base_errors import __ALL__ from argilla_server.errors.error_handler import APIErrorHandler @@ -92,6 +93,7 @@ def create_api_v1(): users_v1.router, vectors_settings_v1.router, workspaces_v1.router, + jobs_v1.router, oauth2_v1.router, settings_v1.router, ]: diff --git a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py index 2becb6c1f2..67669696a0 100644 --- a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py +++ b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py @@ -142,3 +142,9 @@ class DatasetUpdate(UpdateSchema): distribution: Optional[DatasetDistributionUpdate] __non_nullable_fields__ = {"name", "allow_extra_metadata", "distribution"} + + +class HubDataset(BaseModel): + name: str + subset: str + split: str diff --git a/argilla-server/src/argilla_server/api/schemas/v1/jobs.py b/argilla-server/src/argilla_server/api/schemas/v1/jobs.py new file mode 100644 index 0000000000..3f3f916516 --- /dev/null +++ b/argilla-server/src/argilla_server/api/schemas/v1/jobs.py @@ -0,0 +1,21 @@ +# Copyright 2021-present, the Recognai S.L. team. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from rq.job import JobStatus +from pydantic import BaseModel + + +class Job(BaseModel): + id: str + status: JobStatus diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py new file mode 100644 index 0000000000..b3d7d5ce53 --- /dev/null +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -0,0 +1,90 @@ +# Copyright 2021-present, the Recognai S.L. team. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from typing_extensions import Self + +from datasets import load_dataset +from sqlalchemy.ext.asyncio import AsyncSession + +from argilla_server.models.database import Dataset +from argilla_server.search_engine import SearchEngine +from argilla_server.bulk.records_bulk import CreateRecordsBulk +from argilla_server.api.schemas.v1.records import RecordCreate as RecordCreateSchema +from argilla_server.api.schemas.v1.records_bulk import RecordsBulkCreate as RecordsBulkCreateSchema + +BATCH_SIZE = 100 + + +class HubDataset: + # TODO: (Ben feedback) rename `name` to `repository_id` or `repo_id` + # TODO: (Ben feedback) check subset and split and see if we should support None + def __init__(self, name: str, subset: str, split: str): + self.dataset = load_dataset(path=name, name=subset, split=split) + self.iterable_dataset = self.dataset.to_iterable_dataset() + + @property + def num_rows(self) -> int: + return self.dataset.num_rows + + def take(self, n: int) -> Self: + self.iterable_dataset = self.iterable_dataset.take(n) + + return self + + # TODO: We can change things so we get the database and search engine here instead of receiving them as parameters + async def import_to(self, db: AsyncSession, search_engine: SearchEngine, dataset: Dataset) -> None: + if not dataset.is_ready: + raise Exception("it's not possible to import records to a non published dataset") + + batched_dataset = self.iterable_dataset.batch(batch_size=BATCH_SIZE) + for batch in batched_dataset: + await self._import_batch_to(db, search_engine, batch, dataset) + + async def _import_batch_to( + self, db: AsyncSession, search_engine: SearchEngine, batch: dict, dataset: Dataset + ) -> None: + batch_size = len(next(iter(batch.values()))) + + items = [] + for i in range(batch_size): + # NOTE: if there is a value with key "id" in the batch, we will use it as external_id + external_id = None + if "id" in batch: + external_id = batch["id"][i] + + fields = {} + for field in dataset.fields: + # TODO: Should we cast to string or change the schema to use not strict string? + value = batch[field.name][i] + if field.is_text: + value = str(value) + + fields[field.name] = value + + metadata = {} + for metadata_property in dataset.metadata_properties: + metadata[metadata_property.name] = batch[metadata_property.name][i] + + items.append( + RecordCreateSchema( + fields=fields, + metadata=metadata, + external_id=external_id, + responses=None, + suggestions=None, + vectors=None, + ), + ) + + await CreateRecordsBulk(db, search_engine).create_records_bulk(dataset, RecordsBulkCreateSchema(items=items)) diff --git a/argilla-server/src/argilla_server/jobs/dataset_jobs.py b/argilla-server/src/argilla_server/jobs/dataset_jobs.py index 2389a315e8..7edb0f131d 100644 --- a/argilla-server/src/argilla_server/jobs/dataset_jobs.py +++ b/argilla-server/src/argilla_server/jobs/dataset_jobs.py @@ -31,7 +31,7 @@ @job(DEFAULT_QUEUE, timeout=JOB_TIMEOUT_DISABLED, retry=Retry(max=3)) -async def update_dataset_records_status_job(dataset_id: UUID): +async def update_dataset_records_status_job(dataset_id: UUID) -> None: """This Job updates the status of all the records in the dataset when the distribution strategy changes.""" record_ids = [] diff --git a/argilla-server/src/argilla_server/jobs/hub_jobs.py b/argilla-server/src/argilla_server/jobs/hub_jobs.py new file mode 100644 index 0000000000..70d9402c57 --- /dev/null +++ b/argilla-server/src/argilla_server/jobs/hub_jobs.py @@ -0,0 +1,48 @@ +# Copyright 2021-present, the Recognai S.L. team. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from uuid import UUID + +from rq import Retry +from rq.decorators import job +from sqlalchemy.orm import selectinload + +from argilla_server.models import Dataset +from argilla_server.settings import settings +from argilla_server.contexts.hub import HubDataset +from argilla_server.database import AsyncSessionLocal +from argilla_server.search_engine.base import SearchEngine +from argilla_server.jobs.queues import DEFAULT_QUEUE + +# TODO: Move this to be defined on jobs queues as a shared constant +JOB_TIMEOUT_DISABLED = -1 + + +# TODO: Once we merge webhooks we should change the queue to use a different one (default queue is deleted there) +@job(DEFAULT_QUEUE, timeout=JOB_TIMEOUT_DISABLED, retry=Retry(max=3)) +async def import_dataset_from_hub_job(name: str, subset: str, split: str, dataset_id: UUID) -> None: + hub_dataset = HubDataset(name, subset, split) + + async with AsyncSessionLocal() as db: + async with SearchEngine.get_by_name(settings.search_engine) as search_engine: + dataset = await Dataset.get_or_raise( + db, + dataset_id, + options=[ + selectinload(Dataset.fields), + selectinload(Dataset.metadata_properties), + ], + ) + + await hub_dataset.import_to(db, search_engine, dataset) diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py new file mode 100644 index 0000000000..74399ba427 --- /dev/null +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -0,0 +1,61 @@ +# Copyright 2021-present, the Recognai S.L. team. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import pytest + +from sqlalchemy import select, func +from sqlalchemy.ext.asyncio import AsyncSession + +from argilla_server.api.schemas.v1.metadata_properties import IntegerMetadataProperty +from argilla_server.enums import DatasetStatus +from argilla_server.models import Record +from argilla_server.contexts.hub import HubDataset +from argilla_server.search_engine import SearchEngine + +from tests.factories import DatasetFactory, TextFieldFactory, IntegerMetadataPropertyFactory + + +@pytest.mark.asyncio +class TestHubDataset: + async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: SearchEngine): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) + await TextFieldFactory.create(name="review", required=True, dataset=dataset) + await TextFieldFactory.create(name="date", dataset=dataset) + await TextFieldFactory.create(name="star", dataset=dataset) + + await IntegerMetadataPropertyFactory.create(name="version_id", dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) + + record = (await db.execute(select(Record))).scalar_one() + assert record.external_id == "7bd227d9-afc9-11e6-aba1-c4b301cdf627" + assert record.fields["package_name"] == "com.mantz_it.rfanalyzer" + assert ( + record.fields["review"] + == "Great app! The new version now works on my Bravia Android TV which is great as it's right by my rooftop aerial cable. The scan feature would be useful...any ETA on when this will be available? Also the option to import a list of bookmarks e.g. from a simple properties file would be useful." + ) + assert record.fields["date"] == "October 12 2016" + assert record.fields["star"] == "4" + + async def test_hub_dataset_num_rows(self): + hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + + assert hub_dataset.num_rows == 5 From 6108ffe031fc8cf28fc9abff3b2b27a4e3de32c5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 11 Oct 2024 11:43:15 +0200 Subject: [PATCH 02/16] feat: improve import_dataset_from_hub_job to get dataset before instantiate HubDataset class --- .../src/argilla_server/jobs/hub_jobs.py | 22 +++++++++---------- 1 file changed, 10 insertions(+), 12 deletions(-) diff --git a/argilla-server/src/argilla_server/jobs/hub_jobs.py b/argilla-server/src/argilla_server/jobs/hub_jobs.py index 70d9402c57..b527d8852a 100644 --- a/argilla-server/src/argilla_server/jobs/hub_jobs.py +++ b/argilla-server/src/argilla_server/jobs/hub_jobs.py @@ -32,17 +32,15 @@ # TODO: Once we merge webhooks we should change the queue to use a different one (default queue is deleted there) @job(DEFAULT_QUEUE, timeout=JOB_TIMEOUT_DISABLED, retry=Retry(max=3)) async def import_dataset_from_hub_job(name: str, subset: str, split: str, dataset_id: UUID) -> None: - hub_dataset = HubDataset(name, subset, split) - async with AsyncSessionLocal() as db: - async with SearchEngine.get_by_name(settings.search_engine) as search_engine: - dataset = await Dataset.get_or_raise( - db, - dataset_id, - options=[ - selectinload(Dataset.fields), - selectinload(Dataset.metadata_properties), - ], - ) + dataset = await Dataset.get_or_raise( + db, + dataset_id, + options=[ + selectinload(Dataset.fields), + selectinload(Dataset.metadata_properties), + ], + ) - await hub_dataset.import_to(db, search_engine, dataset) + async with SearchEngine.get_by_name(settings.search_engine) as search_engine: + await HubDataset(name, subset, split).import_to(db, search_engine, dataset) From 3a875ee5e6ce287134f139f7df76ff07833ccde8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 11 Oct 2024 11:45:05 +0200 Subject: [PATCH 03/16] feat: improve HubDataset batch processing --- argilla-server/pyproject.toml | 1 - .../src/argilla_server/contexts/hub.py | 67 +++++++++++-------- 2 files changed, 38 insertions(+), 30 deletions(-) diff --git a/argilla-server/pyproject.toml b/argilla-server/pyproject.toml index 21e6a56192..ddd7e2a818 100644 --- a/argilla-server/pyproject.toml +++ b/argilla-server/pyproject.toml @@ -62,7 +62,6 @@ dependencies = [ "datasets>=3.0.1", # For Telemetry "huggingface_hub>=0.13,<1", - ] [project.optional-dependencies] diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index b3d7d5ce53..c1e3229a3f 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -12,6 +12,7 @@ # See the License for the specific language governing permissions and # limitations under the License. +from typing import Union from typing_extensions import Self from datasets import load_dataset @@ -42,7 +43,6 @@ def take(self, n: int) -> Self: return self - # TODO: We can change things so we get the database and search engine here instead of receiving them as parameters async def import_to(self, db: AsyncSession, search_engine: SearchEngine, dataset: Dataset) -> None: if not dataset.is_ready: raise Exception("it's not possible to import records to a non published dataset") @@ -58,33 +58,42 @@ async def _import_batch_to( items = [] for i in range(batch_size): - # NOTE: if there is a value with key "id" in the batch, we will use it as external_id - external_id = None - if "id" in batch: - external_id = batch["id"][i] - - fields = {} - for field in dataset.fields: - # TODO: Should we cast to string or change the schema to use not strict string? - value = batch[field.name][i] - if field.is_text: - value = str(value) - - fields[field.name] = value - - metadata = {} - for metadata_property in dataset.metadata_properties: - metadata[metadata_property.name] = batch[metadata_property.name][i] - - items.append( - RecordCreateSchema( - fields=fields, - metadata=metadata, - external_id=external_id, - responses=None, - suggestions=None, - vectors=None, - ), - ) + items.append(self._batch_row_to_record_schema(batch, i, dataset)) await CreateRecordsBulk(db, search_engine).create_records_bulk(dataset, RecordsBulkCreateSchema(items=items)) + + def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) -> RecordCreateSchema: + return RecordCreateSchema( + fields=self._batch_row_fields(batch, index, dataset), + metadata=self._batch_row_metadata(batch, index, dataset), + external_id=self._batch_row_external_id(batch, index), + responses=None, + suggestions=None, + vectors=None, + ) + + # NOTE: if there is a value with key "id" in the batch, we will use it as external_id + def _batch_row_external_id(self, batch: dict, index: int) -> Union[str, None]: + if not "id" in batch: + return None + + return batch["id"][index] + + def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: + fields = {} + for field in dataset.fields: + # TODO: Should we cast to string or change the schema to use not strict string? + value = batch[field.name][index] + if field.is_text: + value = str(value) + + fields[field.name] = value + + return fields + + def _batch_row_metadata(self, batch: dict, index: int, dataset: Dataset) -> dict: + metadata = {} + for metadata_property in dataset.metadata_properties: + metadata[metadata_property.name] = batch[metadata_property.name][index] + + return metadata From b10a92e0e0d42016ccda240fb2c79d627070d422 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 11 Oct 2024 14:56:08 +0200 Subject: [PATCH 04/16] feat: use UpsertRecordsBulk of CreateRecordsBulk for importing datasets from hub --- .../api/handlers/v1/datasets/records_bulk.py | 4 ++-- .../src/argilla_server/contexts/hub.py | 13 +++++++------ .../tests/unit/contexts/hub/test_hub_dataset.py | 16 ++++++++++++++++ 3 files changed, 25 insertions(+), 8 deletions(-) diff --git a/argilla-server/src/argilla_server/api/handlers/v1/datasets/records_bulk.py b/argilla-server/src/argilla_server/api/handlers/v1/datasets/records_bulk.py index 69cc536a0f..4244c3a735 100644 --- a/argilla-server/src/argilla_server/api/handlers/v1/datasets/records_bulk.py +++ b/argilla-server/src/argilla_server/api/handlers/v1/datasets/records_bulk.py @@ -67,7 +67,7 @@ async def create_dataset_records_bulk( async def upsert_dataset_records_bulk( *, dataset_id: UUID, - records_bulk_create: RecordsBulkUpsert, + records_bulk_upsert: RecordsBulkUpsert, db: AsyncSession = Depends(get_async_db), search_engine: SearchEngine = Depends(get_search_engine), current_user: User = Security(auth.get_current_user), @@ -86,7 +86,7 @@ async def upsert_dataset_records_bulk( await authorize(current_user, DatasetPolicy.upsert_records(dataset)) - records_bulk = await UpsertRecordsBulk(db, search_engine).upsert_records_bulk(dataset, records_bulk_create) + records_bulk = await UpsertRecordsBulk(db, search_engine).upsert_records_bulk(dataset, records_bulk_upsert) updated = len(records_bulk.updated_item_ids) created = len(records_bulk.items) - updated diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index c1e3229a3f..6e1739cdb3 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -20,9 +20,9 @@ from argilla_server.models.database import Dataset from argilla_server.search_engine import SearchEngine -from argilla_server.bulk.records_bulk import CreateRecordsBulk -from argilla_server.api.schemas.v1.records import RecordCreate as RecordCreateSchema -from argilla_server.api.schemas.v1.records_bulk import RecordsBulkCreate as RecordsBulkCreateSchema +from argilla_server.bulk.records_bulk import UpsertRecordsBulk +from argilla_server.api.schemas.v1.records import RecordUpsert as RecordUpsertSchema +from argilla_server.api.schemas.v1.records_bulk import RecordsBulkUpsert as RecordsBulkUpsertSchema BATCH_SIZE = 100 @@ -60,10 +60,11 @@ async def _import_batch_to( for i in range(batch_size): items.append(self._batch_row_to_record_schema(batch, i, dataset)) - await CreateRecordsBulk(db, search_engine).create_records_bulk(dataset, RecordsBulkCreateSchema(items=items)) + await UpsertRecordsBulk(db, search_engine).upsert_records_bulk(dataset, RecordsBulkUpsertSchema(items=items)) - def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) -> RecordCreateSchema: - return RecordCreateSchema( + def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) -> RecordUpsertSchema: + return RecordUpsertSchema( + id=None, fields=self._batch_row_fields(batch, index, dataset), metadata=self._batch_row_metadata(batch, index, dataset), external_id=self._batch_row_external_id(batch, index), diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index 74399ba427..b8643e3259 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -55,6 +55,22 @@ async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: assert record.fields["date"] == "October 12 2016" assert record.fields["star"] == "4" + async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_search_engine: SearchEngine): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + async def test_hub_dataset_num_rows(self): hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") From 2b47522e9972d05d7b8488d0debcf3e2ea9d3595 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 11 Oct 2024 16:39:33 +0200 Subject: [PATCH 05/16] feat: transform dataset importing value columns with PIL images to data URLs --- argilla-server/pdm.lock | 58 ++++++++++++++++++- argilla-server/pyproject.toml | 2 + .../src/argilla_server/contexts/hub.py | 19 +++++- .../unit/contexts/hub/test_hub_dataset.py | 21 ++++++- 4 files changed, 97 insertions(+), 3 deletions(-) diff --git a/argilla-server/pdm.lock b/argilla-server/pdm.lock index bd32bc177c..ea7a5579dd 100644 --- a/argilla-server/pdm.lock +++ b/argilla-server/pdm.lock @@ -5,7 +5,7 @@ groups = ["default", "postgresql", "test"] strategy = ["inherit_metadata"] lock_version = "4.5.0" -content_hash = "sha256:3644c449b4ba725a772eff525d5f3a21aa2159935370f203ae14d1bb14dd0243" +content_hash = "sha256:48b12a94ef28b58816d58c6aeb07735c1c834e2b2edf9988649a3641ed3a08ab" [[metadata.targets]] requires_python = ">=3.8,<3.11" @@ -1497,6 +1497,62 @@ files = [ {file = "pathy-0.11.0.tar.gz", hash = "sha256:bb3d0e6b0b8bf76ef4f63c7191e96e0af2ed65c8fdb5fa17488f9c879e63706d"}, ] +[[package]] +name = "pillow" +version = "10.4.0" +requires_python = ">=3.8" +summary = "Python Imaging Library (Fork)" +groups = ["default"] +files = [ + {file = "pillow-10.4.0-cp310-cp310-macosx_10_10_x86_64.whl", hash = "sha256:4d9667937cfa347525b319ae34375c37b9ee6b525440f3ef48542fcf66f2731e"}, + {file = "pillow-10.4.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:543f3dc61c18dafb755773efc89aae60d06b6596a63914107f75459cf984164d"}, + {file = "pillow-10.4.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7928ecbf1ece13956b95d9cbcfc77137652b02763ba384d9ab508099a2eca856"}, + {file = "pillow-10.4.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e4d49b85c4348ea0b31ea63bc75a9f3857869174e2bf17e7aba02945cd218e6f"}, + {file = "pillow-10.4.0-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:6c762a5b0997f5659a5ef2266abc1d8851ad7749ad9a6a5506eb23d314e4f46b"}, + {file = "pillow-10.4.0-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:a985e028fc183bf12a77a8bbf36318db4238a3ded7fa9df1b9a133f1cb79f8fc"}, + {file = "pillow-10.4.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:812f7342b0eee081eaec84d91423d1b4650bb9828eb53d8511bcef8ce5aecf1e"}, + {file = "pillow-10.4.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:ac1452d2fbe4978c2eec89fb5a23b8387aba707ac72810d9490118817d9c0b46"}, + {file = "pillow-10.4.0-cp310-cp310-win32.whl", hash = "sha256:bcd5e41a859bf2e84fdc42f4edb7d9aba0a13d29a2abadccafad99de3feff984"}, + {file = "pillow-10.4.0-cp310-cp310-win_amd64.whl", hash = "sha256:ecd85a8d3e79cd7158dec1c9e5808e821feea088e2f69a974db5edf84dc53141"}, + {file = "pillow-10.4.0-cp310-cp310-win_arm64.whl", hash = "sha256:ff337c552345e95702c5fde3158acb0625111017d0e5f24bf3acdb9cc16b90d1"}, + {file = "pillow-10.4.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:8d4d5063501b6dd4024b8ac2f04962d661222d120381272deea52e3fc52d3736"}, + {file = "pillow-10.4.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7c1ee6f42250df403c5f103cbd2768a28fe1a0ea1f0f03fe151c8741e1469c8b"}, + {file = "pillow-10.4.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b15e02e9bb4c21e39876698abf233c8c579127986f8207200bc8a8f6bb27acf2"}, + {file = "pillow-10.4.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7a8d4bade9952ea9a77d0c3e49cbd8b2890a399422258a77f357b9cc9be8d680"}, + {file = "pillow-10.4.0-cp38-cp38-manylinux_2_28_aarch64.whl", hash = "sha256:43efea75eb06b95d1631cb784aa40156177bf9dd5b4b03ff38979e048258bc6b"}, + {file = "pillow-10.4.0-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:950be4d8ba92aca4b2bb0741285a46bfae3ca699ef913ec8416c1b78eadd64cd"}, + {file = "pillow-10.4.0-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:d7480af14364494365e89d6fddc510a13e5a2c3584cb19ef65415ca57252fb84"}, + {file = "pillow-10.4.0-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:73664fe514b34c8f02452ffb73b7a92c6774e39a647087f83d67f010eb9a0cf0"}, + {file = "pillow-10.4.0-cp38-cp38-win32.whl", hash = "sha256:e88d5e6ad0d026fba7bdab8c3f225a69f063f116462c49892b0149e21b6c0a0e"}, + {file = "pillow-10.4.0-cp38-cp38-win_amd64.whl", hash = "sha256:5161eef006d335e46895297f642341111945e2c1c899eb406882a6c61a4357ab"}, + {file = "pillow-10.4.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:0ae24a547e8b711ccaaf99c9ae3cd975470e1a30caa80a6aaee9a2f19c05701d"}, + {file = "pillow-10.4.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:298478fe4f77a4408895605f3482b6cc6222c018b2ce565c2b6b9c354ac3229b"}, + {file = "pillow-10.4.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:134ace6dc392116566980ee7436477d844520a26a4b1bd4053f6f47d096997fd"}, + {file = "pillow-10.4.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:930044bb7679ab003b14023138b50181899da3f25de50e9dbee23b61b4de2126"}, + {file = "pillow-10.4.0-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:c76e5786951e72ed3686e122d14c5d7012f16c8303a674d18cdcd6d89557fc5b"}, + {file = "pillow-10.4.0-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:b2724fdb354a868ddf9a880cb84d102da914e99119211ef7ecbdc613b8c96b3c"}, + {file = "pillow-10.4.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:dbc6ae66518ab3c5847659e9988c3b60dc94ffb48ef9168656e0019a93dbf8a1"}, + {file = "pillow-10.4.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:06b2f7898047ae93fad74467ec3d28fe84f7831370e3c258afa533f81ef7f3df"}, + {file = "pillow-10.4.0-cp39-cp39-win32.whl", hash = "sha256:7970285ab628a3779aecc35823296a7869f889b8329c16ad5a71e4901a3dc4ef"}, + {file = "pillow-10.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:961a7293b2457b405967af9c77dcaa43cc1a8cd50d23c532e62d48ab6cdd56f5"}, + {file = "pillow-10.4.0-cp39-cp39-win_arm64.whl", hash = "sha256:32cda9e3d601a52baccb2856b8ea1fc213c90b340c542dcef77140dfa3278a9e"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-macosx_10_15_x86_64.whl", hash = "sha256:5b4815f2e65b30f5fbae9dfffa8636d992d49705723fe86a3661806e069352d4"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:8f0aef4ef59694b12cadee839e2ba6afeab89c0f39a3adc02ed51d109117b8da"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9f4727572e2918acaa9077c919cbbeb73bd2b3ebcfe033b72f858fc9fbef0026"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ff25afb18123cea58a591ea0244b92eb1e61a1fd497bf6d6384f09bc3262ec3e"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:dc3e2db6ba09ffd7d02ae9141cfa0ae23393ee7687248d46a7507b75d610f4f5"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:02a2be69f9c9b8c1e97cf2713e789d4e398c751ecfd9967c18d0ce304efbf885"}, + {file = "pillow-10.4.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:0755ffd4a0c6f267cccbae2e9903d95477ca2f77c4fcf3a3a09570001856c8a5"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-macosx_10_15_x86_64.whl", hash = "sha256:a02364621fe369e06200d4a16558e056fe2805d3468350df3aef21e00d26214b"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:1b5dea9831a90e9d0721ec417a80d4cbd7022093ac38a568db2dd78363b00908"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9b885f89040bb8c4a1573566bbb2f44f5c505ef6e74cec7ab9068c900047f04b"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:87dd88ded2e6d74d31e1e0a99a726a6765cda32d00ba72dc37f0651f306daaa8"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:2db98790afc70118bd0255c2eeb465e9767ecf1f3c25f9a1abb8ffc8cfd1fe0a"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:f7baece4ce06bade126fb84b8af1c33439a76d8a6fd818970215e0560ca28c27"}, + {file = "pillow-10.4.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:cfdd747216947628af7b259d274771d84db2268ca062dd5faf373639d00113a3"}, + {file = "pillow-10.4.0.tar.gz", hash = "sha256:166c1cd4d24309b30d61f79f4a9114b7b2313d7450912277855ff5dfd7cd4a06"}, +] + [[package]] name = "pluggy" version = "1.3.0" diff --git a/argilla-server/pyproject.toml b/argilla-server/pyproject.toml index ddd7e2a818..637e0e48fe 100644 --- a/argilla-server/pyproject.toml +++ b/argilla-server/pyproject.toml @@ -59,7 +59,9 @@ dependencies = [ "typer >= 0.6.0, < 0.10.0", # spaCy only supports typer<0.10.0 "packaging>=23.2", "psycopg2-binary>=2.9.9", + # For HF dataset import "datasets>=3.0.1", + "pillow>=10.4.0", # For Telemetry "huggingface_hub>=0.13,<1", ] diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index 6e1739cdb3..a6d0283b21 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -12,11 +12,15 @@ # See the License for the specific language governing permissions and # limitations under the License. +import io +import base64 + from typing import Union from typing_extensions import Self from datasets import load_dataset from sqlalchemy.ext.asyncio import AsyncSession +from PIL import Image from argilla_server.models.database import Dataset from argilla_server.search_engine import SearchEngine @@ -83,11 +87,14 @@ def _batch_row_external_id(self, batch: dict, index: int) -> Union[str, None]: def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: fields = {} for field in dataset.fields: - # TODO: Should we cast to string or change the schema to use not strict string? value = batch[field.name][index] + if field.is_text: value = str(value) + if field.is_image and isinstance(value, Image.Image): + value = pil_image_to_data_url(value) + fields[field.name] = value return fields @@ -98,3 +105,13 @@ def _batch_row_metadata(self, batch: dict, index: int, dataset: Dataset) -> dict metadata[metadata_property.name] = batch[metadata_property.name][index] return metadata + + +def pil_image_to_data_url(image: Image.Image): + buffer = io.BytesIO() + + image.save(buffer, format=image.format) + + base64_image = base64.b64encode(buffer.getvalue()).decode("utf-8") + + return f"data:{image.get_format_mimetype()};base64,{base64_image}" diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index b8643e3259..0e681109d6 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -23,7 +23,7 @@ from argilla_server.contexts.hub import HubDataset from argilla_server.search_engine import SearchEngine -from tests.factories import DatasetFactory, TextFieldFactory, IntegerMetadataPropertyFactory +from tests.factories import DatasetFactory, ImageFieldFactory, TextFieldFactory, IntegerMetadataPropertyFactory @pytest.mark.asyncio @@ -71,6 +71,25 @@ async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_se await hub_dataset.import_to(db, mock_search_engine, dataset) assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + async def test_hub_dataset_import_image_fields(self, db: AsyncSession, mock_search_engine: SearchEngine): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await ImageFieldFactory.create(name="image", required=True, dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset(name="lmms-lab/llava-critic-113k", subset="pairwise", split="train") + + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) + + record = (await db.execute(select(Record))).scalar_one() + assert record.external_id == "vlfeedback_1" + assert ( + record.fields["image"][:100] + == "" + ) + async def test_hub_dataset_num_rows(self): hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") From 7f93e0b2ad3998a5ea6f5ee23c07318ecc11c1e5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Mon, 14 Oct 2024 12:12:09 +0200 Subject: [PATCH 06/16] feat: add support to map suggestions importing datasets from hub --- .../src/argilla_server/contexts/hub.py | 35 +++++++++-- .../src/argilla_server/jobs/hub_jobs.py | 1 + .../src/argilla_server/models/database.py | 12 ++++ .../unit/contexts/hub/test_hub_dataset.py | 59 ++++++++++++++++--- 4 files changed, 94 insertions(+), 13 deletions(-) diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index a6d0283b21..774a974d04 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -18,22 +18,21 @@ from typing import Union from typing_extensions import Self +from PIL import Image from datasets import load_dataset from sqlalchemy.ext.asyncio import AsyncSession -from PIL import Image from argilla_server.models.database import Dataset from argilla_server.search_engine import SearchEngine from argilla_server.bulk.records_bulk import UpsertRecordsBulk from argilla_server.api.schemas.v1.records import RecordUpsert as RecordUpsertSchema from argilla_server.api.schemas.v1.records_bulk import RecordsBulkUpsert as RecordsBulkUpsertSchema +from argilla_server.api.schemas.v1.suggestions import SuggestionCreate BATCH_SIZE = 100 class HubDataset: - # TODO: (Ben feedback) rename `name` to `repository_id` or `repo_id` - # TODO: (Ben feedback) check subset and split and see if we should support None def __init__(self, name: str, subset: str, split: str): self.dataset = load_dataset(path=name, name=subset, split=split) self.iterable_dataset = self.dataset.to_iterable_dataset() @@ -69,11 +68,11 @@ async def _import_batch_to( def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) -> RecordUpsertSchema: return RecordUpsertSchema( id=None, + external_id=self._batch_row_external_id(batch, index), fields=self._batch_row_fields(batch, index, dataset), metadata=self._batch_row_metadata(batch, index, dataset), - external_id=self._batch_row_external_id(batch, index), + suggestions=self._batch_row_suggestions(batch, index, dataset), responses=None, - suggestions=None, vectors=None, ) @@ -106,6 +105,32 @@ def _batch_row_metadata(self, batch: dict, index: int, dataset: Dataset) -> dict return metadata + def _batch_row_suggestions(self, batch: dict, index: int, dataset: Dataset) -> list: + suggestions = [] + for question in dataset.questions: + if not question.name in batch: + continue + + value = batch[question.name][index] + + if question.is_text or question.is_label_selection: + value = str(value) + + if question.is_rating: + value = int(value) + + suggestions.append( + SuggestionCreate( + question_id=question.id, + value=value, + type=None, + agent=None, + score=None, + ), + ) + + return suggestions + def pil_image_to_data_url(image: Image.Image): buffer = io.BytesIO() diff --git a/argilla-server/src/argilla_server/jobs/hub_jobs.py b/argilla-server/src/argilla_server/jobs/hub_jobs.py index b527d8852a..fcbbbafc90 100644 --- a/argilla-server/src/argilla_server/jobs/hub_jobs.py +++ b/argilla-server/src/argilla_server/jobs/hub_jobs.py @@ -38,6 +38,7 @@ async def import_dataset_from_hub_job(name: str, subset: str, split: str, datase dataset_id, options=[ selectinload(Dataset.fields), + selectinload(Dataset.questions), selectinload(Dataset.metadata_properties), ], ) diff --git a/argilla-server/src/argilla_server/models/database.py b/argilla-server/src/argilla_server/models/database.py index 2d61e3836e..186097e060 100644 --- a/argilla-server/src/argilla_server/models/database.py +++ b/argilla-server/src/argilla_server/models/database.py @@ -285,6 +285,18 @@ class Question(DatabaseModel): def parsed_settings(self) -> QuestionSettings: return parse_obj_as(QuestionSettings, self.settings) + @property + def is_text(self) -> bool: + return self.settings.get("type") == QuestionType.text + + @property + def is_label_selection(self) -> bool: + return self.settings.get("type") == QuestionType.label_selection + + @property + def is_rating(self) -> bool: + return self.settings.get("type") == QuestionType.rating + @property def type(self) -> QuestionType: return QuestionType(self.settings["type"]) diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index 0e681109d6..362b31df71 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -18,12 +18,18 @@ from sqlalchemy.ext.asyncio import AsyncSession from argilla_server.api.schemas.v1.metadata_properties import IntegerMetadataProperty -from argilla_server.enums import DatasetStatus +from argilla_server.enums import DatasetStatus, QuestionType from argilla_server.models import Record from argilla_server.contexts.hub import HubDataset from argilla_server.search_engine import SearchEngine -from tests.factories import DatasetFactory, ImageFieldFactory, TextFieldFactory, IntegerMetadataPropertyFactory +from tests.factories import ( + DatasetFactory, + ImageFieldFactory, + RatingQuestionFactory, + TextFieldFactory, + IntegerMetadataPropertyFactory, +) @pytest.mark.asyncio @@ -39,6 +45,7 @@ async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: await IntegerMetadataPropertyFactory.create(name="version_id", dataset=dataset) await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") @@ -55,28 +62,47 @@ async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: assert record.fields["date"] == "October 12 2016" assert record.fields["star"] == "4" - async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_search_engine: SearchEngine): + async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mock_search_engine: SearchEngine): dataset = await DatasetFactory.create(status=DatasetStatus.ready) await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) + await TextFieldFactory.create(name="review", required=True, dataset=dataset) + + question = await RatingQuestionFactory.create( + name="star", + required=True, + dataset=dataset, + settings={ + "type": QuestionType.rating, + "options": [ + {"value": 1}, + {"value": 2}, + {"value": 3}, + {"value": 4}, + {"value": 5}, + ], + }, + ) await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") - await hub_dataset.import_to(db, mock_search_engine, dataset) - assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) - await hub_dataset.import_to(db, mock_search_engine, dataset) - assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + record = (await db.execute(select(Record))).scalar_one() + assert record.suggestions[0].value == 4 + assert record.suggestions[0].question_id == question.id - async def test_hub_dataset_import_image_fields(self, db: AsyncSession, mock_search_engine: SearchEngine): + async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, mock_search_engine: SearchEngine): dataset = await DatasetFactory.create(status=DatasetStatus.ready) await ImageFieldFactory.create(name="image", required=True, dataset=dataset) await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties hub_dataset = HubDataset(name="lmms-lab/llava-critic-113k", subset="pairwise", split="train") @@ -90,6 +116,23 @@ async def test_hub_dataset_import_image_fields(self, db: AsyncSession, mock_sear == "" ) + async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_search_engine: SearchEngine): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + async def test_hub_dataset_num_rows(self): hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") From a38fddaa6ac69e8d380ab53ac1f334132e449d1a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Mon, 14 Oct 2024 15:32:03 +0200 Subject: [PATCH 07/16] feat: add support for hub dataset mapping --- .../api/handlers/v1/datasets/datasets.py | 1 + .../argilla_server/api/schemas/v1/datasets.py | 13 ++++ .../src/argilla_server/contexts/hub.py | 34 ++++++--- .../src/argilla_server/jobs/hub_jobs.py | 7 +- .../src/argilla_server/models/database.py | 2 +- .../unit/contexts/hub/test_hub_dataset.py | 76 +++++++++++++++++-- 6 files changed, 111 insertions(+), 22 deletions(-) diff --git a/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py b/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py index 825408e329..f0d234f4b8 100644 --- a/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py +++ b/argilla-server/src/argilla_server/api/handlers/v1/datasets/datasets.py @@ -324,6 +324,7 @@ async def import_dataset_from_hub( subset=hub_dataset.subset, split=hub_dataset.split, dataset_id=dataset.id, + mapping=hub_dataset.mapping.dict(), ) return JobSchema(id=job.id, status=job.get_status()) diff --git a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py index 67669696a0..f7ce04dbe3 100644 --- a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py +++ b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py @@ -144,7 +144,20 @@ class DatasetUpdate(UpdateSchema): __non_nullable_fields__ = {"name", "allow_extra_metadata", "distribution"} +class HubDatasetMappingItem(BaseModel): + source: str = Field(..., description="The name of the column in the Hub Dataset") + target: str = Field(..., description="The name of the target resource in the Argilla Dataset") + + +class HubDatasetMapping(BaseModel): + fields: List[HubDatasetMappingItem] = Field(..., min_items=1) + metadata: List[HubDatasetMappingItem] + suggestions: List[HubDatasetMappingItem] + external_id: Optional[str] + + class HubDataset(BaseModel): name: str subset: str split: str + mapping: HubDatasetMapping diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index 774a974d04..8638771b6c 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -22,9 +22,11 @@ from datasets import load_dataset from sqlalchemy.ext.asyncio import AsyncSession + from argilla_server.models.database import Dataset from argilla_server.search_engine import SearchEngine from argilla_server.bulk.records_bulk import UpsertRecordsBulk +from argilla_server.api.schemas.v1.datasets import HubDatasetMapping from argilla_server.api.schemas.v1.records import RecordUpsert as RecordUpsertSchema from argilla_server.api.schemas.v1.records_bulk import RecordsBulkUpsert as RecordsBulkUpsertSchema from argilla_server.api.schemas.v1.suggestions import SuggestionCreate @@ -33,8 +35,9 @@ class HubDataset: - def __init__(self, name: str, subset: str, split: str): + def __init__(self, name: str, subset: str, split: str, mapping: HubDatasetMapping): self.dataset = load_dataset(path=name, name=subset, split=split) + self.mapping = mapping self.iterable_dataset = self.dataset.to_iterable_dataset() @property @@ -76,17 +79,19 @@ def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) vectors=None, ) - # NOTE: if there is a value with key "id" in the batch, we will use it as external_id def _batch_row_external_id(self, batch: dict, index: int) -> Union[str, None]: - if not "id" in batch: + if not self.mapping.external_id: return None - return batch["id"][index] + return batch[self.mapping.external_id][index] def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: fields = {} - for field in dataset.fields: - value = batch[field.name][index] + for mapping_field in self.mapping.fields: + value = batch[mapping_field.source][index] + field = dataset.field_by_name(mapping_field.target) + if not field: + continue if field.is_text: value = str(value) @@ -100,19 +105,24 @@ def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: def _batch_row_metadata(self, batch: dict, index: int, dataset: Dataset) -> dict: metadata = {} - for metadata_property in dataset.metadata_properties: - metadata[metadata_property.name] = batch[metadata_property.name][index] + for mapping_metadata in self.mapping.metadata: + value = batch[mapping_metadata.source][index] + metadata_property = dataset.metadata_property_by_name(mapping_metadata.target) + if not metadata_property: + continue + + metadata[metadata_property.name] = value return metadata def _batch_row_suggestions(self, batch: dict, index: int, dataset: Dataset) -> list: suggestions = [] - for question in dataset.questions: - if not question.name in batch: + for mapping_suggestion in self.mapping.suggestions: + value = batch[mapping_suggestion.source][index] + question = dataset.question_by_name(mapping_suggestion.target) + if not question: continue - value = batch[question.name][index] - if question.is_text or question.is_label_selection: value = str(value) diff --git a/argilla-server/src/argilla_server/jobs/hub_jobs.py b/argilla-server/src/argilla_server/jobs/hub_jobs.py index fcbbbafc90..60c915f524 100644 --- a/argilla-server/src/argilla_server/jobs/hub_jobs.py +++ b/argilla-server/src/argilla_server/jobs/hub_jobs.py @@ -23,6 +23,7 @@ from argilla_server.contexts.hub import HubDataset from argilla_server.database import AsyncSessionLocal from argilla_server.search_engine.base import SearchEngine +from argilla_server.api.schemas.v1.datasets import HubDatasetMapping from argilla_server.jobs.queues import DEFAULT_QUEUE # TODO: Move this to be defined on jobs queues as a shared constant @@ -31,7 +32,7 @@ # TODO: Once we merge webhooks we should change the queue to use a different one (default queue is deleted there) @job(DEFAULT_QUEUE, timeout=JOB_TIMEOUT_DISABLED, retry=Retry(max=3)) -async def import_dataset_from_hub_job(name: str, subset: str, split: str, dataset_id: UUID) -> None: +async def import_dataset_from_hub_job(name: str, subset: str, split: str, dataset_id: UUID, mapping: dict) -> None: async with AsyncSessionLocal() as db: dataset = await Dataset.get_or_raise( db, @@ -44,4 +45,6 @@ async def import_dataset_from_hub_job(name: str, subset: str, split: str, datase ) async with SearchEngine.get_by_name(settings.search_engine) as search_engine: - await HubDataset(name, subset, split).import_to(db, search_engine, dataset) + parsed_mapping = HubDatasetMapping.parse_obj(mapping) + + await HubDataset(name, subset, split, parsed_mapping).import_to(db, search_engine, dataset) diff --git a/argilla-server/src/argilla_server/models/database.py b/argilla-server/src/argilla_server/models/database.py index 186097e060..07d831ed9f 100644 --- a/argilla-server/src/argilla_server/models/database.py +++ b/argilla-server/src/argilla_server/models/database.py @@ -424,7 +424,7 @@ def question_by_id(self, question_id: UUID) -> Union[Question, None]: if question.id == question_id: return question - def question_by_name(self, name: str) -> Union["Question", None]: + def question_by_name(self, name: str) -> Union[Question, None]: for question in self.questions: if question.name == name: return question diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index 362b31df71..d922f4f875 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -17,6 +17,7 @@ from sqlalchemy import select, func from sqlalchemy.ext.asyncio import AsyncSession +from argilla_server.api.schemas.v1.datasets import HubDatasetMapping, HubDatasetMappingItem from argilla_server.api.schemas.v1.metadata_properties import IntegerMetadataProperty from argilla_server.enums import DatasetStatus, QuestionType from argilla_server.models import Record @@ -48,7 +49,24 @@ async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties - hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + hub_dataset = HubDataset( + name="lhoestq/demo1", + subset="default", + split="train", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="package_name", target="package_name"), + HubDatasetMappingItem(source="review", target="review"), + HubDatasetMappingItem(source="date", target="date"), + HubDatasetMappingItem(source="star", target="star"), + ], + metadata=[ + HubDatasetMappingItem(source="version_id", target="version_id"), + ], + suggestions=[], + external_id="id", + ), + ) await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) @@ -61,6 +79,7 @@ async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: ) assert record.fields["date"] == "October 12 2016" assert record.fields["star"] == "4" + assert record.metadata_ == {"version_id": 1487} async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mock_search_engine: SearchEngine): dataset = await DatasetFactory.create(status=DatasetStatus.ready) @@ -88,7 +107,20 @@ async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mo await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties - hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + hub_dataset = HubDataset( + name="lhoestq/demo1", + subset="default", + split="train", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="package_name", target="package_name"), + HubDatasetMappingItem(source="review", target="review"), + ], + metadata=[], + suggestions=[HubDatasetMappingItem(source="star", target="star")], + external_id=None, + ), + ) await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) @@ -99,20 +131,30 @@ async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mo async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, mock_search_engine: SearchEngine): dataset = await DatasetFactory.create(status=DatasetStatus.ready) - await ImageFieldFactory.create(name="image", required=True, dataset=dataset) + await ImageFieldFactory.create(name="image-to-review", required=True, dataset=dataset) await dataset.awaitable_attrs.fields await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties - hub_dataset = HubDataset(name="lmms-lab/llava-critic-113k", subset="pairwise", split="train") + hub_dataset = HubDataset( + name="lmms-lab/llava-critic-113k", + subset="pairwise", + split="train", + mapping=HubDatasetMapping( + fields=[HubDatasetMappingItem(source="image", target="image-to-review")], + metadata=[], + suggestions=[], + external_id="id", + ), + ) await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) record = (await db.execute(select(Record))).scalar_one() assert record.external_id == "vlfeedback_1" assert ( - record.fields["image"][:100] + record.fields["image-to-review"][:100] == "" ) @@ -125,7 +167,17 @@ async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_se await dataset.awaitable_attrs.questions await dataset.awaitable_attrs.metadata_properties - hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + hub_dataset = HubDataset( + name="lhoestq/demo1", + subset="default", + split="train", + mapping=HubDatasetMapping( + fields=[HubDatasetMappingItem(source="package_name", target="package_name")], + metadata=[], + suggestions=[], + external_id="id", + ), + ) await hub_dataset.import_to(db, mock_search_engine, dataset) assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 @@ -134,6 +186,16 @@ async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_se assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 async def test_hub_dataset_num_rows(self): - hub_dataset = HubDataset(name="lhoestq/demo1", subset="default", split="train") + hub_dataset = HubDataset( + name="lhoestq/demo1", + subset="default", + split="train", + mapping=HubDatasetMapping( + fields=[HubDatasetMappingItem(source="package_name", target="package_name")], + metadata=[], + suggestions=[], + external_id=None, + ), + ) assert hub_dataset.num_rows == 5 From 15d694f0c5809404f8ee635ac81b0808721b45e2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Mon, 14 Oct 2024 17:02:24 +0200 Subject: [PATCH 08/16] feat: set metadata and suggestions as optional for HubDatasetMapping --- .../argilla_server/api/schemas/v1/datasets.py | 6 ++--- .../unit/contexts/hub/test_hub_dataset.py | 23 ++++++++++--------- 2 files changed, 15 insertions(+), 14 deletions(-) diff --git a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py index f7ce04dbe3..4da742afb3 100644 --- a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py +++ b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py @@ -151,9 +151,9 @@ class HubDatasetMappingItem(BaseModel): class HubDatasetMapping(BaseModel): fields: List[HubDatasetMappingItem] = Field(..., min_items=1) - metadata: List[HubDatasetMappingItem] - suggestions: List[HubDatasetMappingItem] - external_id: Optional[str] + metadata: Optional[List[HubDatasetMappingItem]] = [] + suggestions: Optional[List[HubDatasetMappingItem]] = [] + external_id: Optional[str] = None class HubDataset(BaseModel): diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index d922f4f875..f6908dd909 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -63,7 +63,6 @@ async def test_hub_dataset_import_to(self, db: AsyncSession, mock_search_engine: metadata=[ HubDatasetMappingItem(source="version_id", target="version_id"), ], - suggestions=[], external_id="id", ), ) @@ -116,9 +115,9 @@ async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mo HubDatasetMappingItem(source="package_name", target="package_name"), HubDatasetMappingItem(source="review", target="review"), ], - metadata=[], - suggestions=[HubDatasetMappingItem(source="star", target="star")], - external_id=None, + suggestions=[ + HubDatasetMappingItem(source="star", target="star"), + ], ), ) @@ -142,9 +141,9 @@ async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, m subset="pairwise", split="train", mapping=HubDatasetMapping( - fields=[HubDatasetMappingItem(source="image", target="image-to-review")], - metadata=[], - suggestions=[], + fields=[ + HubDatasetMappingItem(source="image", target="image-to-review"), + ], external_id="id", ), ) @@ -172,9 +171,9 @@ async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_se subset="default", split="train", mapping=HubDatasetMapping( - fields=[HubDatasetMappingItem(source="package_name", target="package_name")], - metadata=[], - suggestions=[], + fields=[ + HubDatasetMappingItem(source="package_name", target="package_name"), + ], external_id="id", ), ) @@ -191,7 +190,9 @@ async def test_hub_dataset_num_rows(self): subset="default", split="train", mapping=HubDatasetMapping( - fields=[HubDatasetMappingItem(source="package_name", target="package_name")], + fields=[ + HubDatasetMappingItem(source="package_name", target="package_name"), + ], metadata=[], suggestions=[], external_id=None, From c3752e8574d316d995bd34f9a125994c404de6fb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Mon, 14 Oct 2024 18:01:45 +0200 Subject: [PATCH 09/16] feat: when no external_id is mapped row_idx is used --- .../src/argilla_server/contexts/hub.py | 16 ++++++- .../unit/contexts/hub/test_hub_dataset.py | 44 ++++++++++++++++++- 2 files changed, 57 insertions(+), 3 deletions(-) diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index 8638771b6c..eb4f253c66 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -32,6 +32,7 @@ from argilla_server.api.schemas.v1.suggestions import SuggestionCreate BATCH_SIZE = 100 +RESET_ROW_IDX = -1 class HubDataset: @@ -39,6 +40,7 @@ def __init__(self, name: str, subset: str, split: str, mapping: HubDatasetMappin self.dataset = load_dataset(path=name, name=subset, split=split) self.mapping = mapping self.iterable_dataset = self.dataset.to_iterable_dataset() + self.row_idx = RESET_ROW_IDX @property def num_rows(self) -> int: @@ -53,10 +55,20 @@ async def import_to(self, db: AsyncSession, search_engine: SearchEngine, dataset if not dataset.is_ready: raise Exception("it's not possible to import records to a non published dataset") + self._reset_row_idx() + batched_dataset = self.iterable_dataset.batch(batch_size=BATCH_SIZE) for batch in batched_dataset: await self._import_batch_to(db, search_engine, batch, dataset) + def _reset_row_idx(self) -> None: + self.row_idx = RESET_ROW_IDX + + def _next_row_idx(self) -> int: + self.row_idx += 1 + + return self.row_idx + async def _import_batch_to( self, db: AsyncSession, search_engine: SearchEngine, batch: dict, dataset: Dataset ) -> None: @@ -79,9 +91,9 @@ def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) vectors=None, ) - def _batch_row_external_id(self, batch: dict, index: int) -> Union[str, None]: + def _batch_row_external_id(self, batch: dict, index: int) -> str: if not self.mapping.external_id: - return None + return str(self._next_row_idx()) return batch[self.mapping.external_id][index] diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index f6908dd909..a53cf0ebf5 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -157,7 +157,9 @@ async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, m == "" ) - async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_search_engine: SearchEngine): + async def test_hub_dataset_import_to_idempotency_with_external_id( + self, db: AsyncSession, mock_search_engine: SearchEngine + ): dataset = await DatasetFactory.create(status=DatasetStatus.ready) await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) @@ -184,6 +186,46 @@ async def test_hub_dataset_import_to_idempotency(self, db: AsyncSession, mock_se await hub_dataset.import_to(db, mock_search_engine, dataset) assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + records = (await db.execute(select(Record))).scalars().all() + assert [record.external_id for record in records] == [ + "7bd227d9-afc9-11e6-aba1-c4b301cdf627", + "7bd22905-afc9-11e6-a5dc-c4b301cdf627", + "7bd2299c-afc9-11e6-85d6-c4b301cdf627", + "7bd22a26-afc9-11e6-9309-c4b301cdf627", + "7bd22aba-afc9-11e6-8293-c4b301cdf627", + ] + + async def test_hub_dataset_import_to_idempotency_without_external_id( + self, db: AsyncSession, mock_search_engine: SearchEngine + ): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset( + name="lhoestq/demo1", + subset="default", + split="train", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="package_name", target="package_name"), + ], + ), + ) + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 5 + + records = (await db.execute(select(Record))).scalars().all() + assert [record.external_id for record in records] == ["0", "1", "2", "3", "4"] + async def test_hub_dataset_num_rows(self): hub_dataset = HubDataset( name="lhoestq/demo1", From 469ebc4aecec6e17ece4b858a49caba2b7c31257 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Tue, 15 Oct 2024 11:11:35 +0200 Subject: [PATCH 10/16] feat: use streaming when loading the dataset --- .../src/argilla_server/contexts/hub.py | 11 +++-------- .../tests/unit/contexts/hub/test_hub_dataset.py | 17 ----------------- 2 files changed, 3 insertions(+), 25 deletions(-) diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index eb4f253c66..6993775377 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -37,17 +37,12 @@ class HubDataset: def __init__(self, name: str, subset: str, split: str, mapping: HubDatasetMapping): - self.dataset = load_dataset(path=name, name=subset, split=split) + self.dataset = load_dataset(path=name, name=subset, split=split, streaming=True) self.mapping = mapping - self.iterable_dataset = self.dataset.to_iterable_dataset() self.row_idx = RESET_ROW_IDX - @property - def num_rows(self) -> int: - return self.dataset.num_rows - def take(self, n: int) -> Self: - self.iterable_dataset = self.iterable_dataset.take(n) + self.dataset = self.dataset.take(n) return self @@ -57,7 +52,7 @@ async def import_to(self, db: AsyncSession, search_engine: SearchEngine, dataset self._reset_row_idx() - batched_dataset = self.iterable_dataset.batch(batch_size=BATCH_SIZE) + batched_dataset = self.dataset.batch(batch_size=BATCH_SIZE) for batch in batched_dataset: await self._import_batch_to(db, search_engine, batch, dataset) diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index a53cf0ebf5..bf0322d5b8 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -225,20 +225,3 @@ async def test_hub_dataset_import_to_idempotency_without_external_id( records = (await db.execute(select(Record))).scalars().all() assert [record.external_id for record in records] == ["0", "1", "2", "3", "4"] - - async def test_hub_dataset_num_rows(self): - hub_dataset = HubDataset( - name="lhoestq/demo1", - subset="default", - split="train", - mapping=HubDatasetMapping( - fields=[ - HubDatasetMappingItem(source="package_name", target="package_name"), - ], - metadata=[], - suggestions=[], - external_id=None, - ), - ) - - assert hub_dataset.num_rows == 5 From b665019be14cbb559fcae6bca9c6e8dd6758cc88 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Tue, 15 Oct 2024 12:38:10 +0200 Subject: [PATCH 11/16] feat: refactor UpsertRecordsBulk to validate records individually --- .../src/argilla_server/bulk/records_bulk.py | 30 ++++++++----- .../src/argilla_server/validators/records.py | 37 ---------------- .../unit/api/handlers/v1/test_datasets.py | 3 +- .../unit/validators/test_records_bulk.py | 43 +------------------ 4 files changed, 23 insertions(+), 90 deletions(-) diff --git a/argilla-server/src/argilla_server/bulk/records_bulk.py b/argilla-server/src/argilla_server/bulk/records_bulk.py index 79b59c9796..6caec289be 100644 --- a/argilla-server/src/argilla_server/bulk/records_bulk.py +++ b/argilla-server/src/argilla_server/bulk/records_bulk.py @@ -21,7 +21,7 @@ from sqlalchemy.orm import selectinload from fastapi.encoders import jsonable_encoder -from argilla_server.api.schemas.v1.records import RecordCreate, RecordUpsert +from argilla_server.api.schemas.v1.records import RecordCreate, RecordUpdate, RecordUpsert from argilla_server.api.schemas.v1.records_bulk import ( RecordsBulk, RecordsBulkCreate, @@ -39,7 +39,7 @@ from argilla_server.errors.future import UnprocessableEntityError from argilla_server.models import Dataset, Record, Response, Suggestion, Vector, VectorSettings from argilla_server.search_engine import SearchEngine -from argilla_server.validators.records import RecordsBulkCreateValidator, RecordsBulkUpsertValidator +from argilla_server.validators.records import RecordsBulkCreateValidator, RecordCreateValidator, RecordUpdateValidator from argilla_server.validators.responses import ResponseCreateValidator from argilla_server.validators.suggestions import SuggestionCreateValidator from argilla_server.validators.vectors import VectorValidator @@ -184,26 +184,36 @@ def _metadata_is_set(self, record_create: RecordCreate) -> bool: class UpsertRecordsBulk(CreateRecordsBulk): async def upsert_records_bulk(self, dataset: Dataset, bulk_upsert: RecordsBulkUpsert) -> RecordsBulkWithUpdateInfo: found_records = await self._fetch_existing_dataset_records(dataset, bulk_upsert.items) - # found_records is passed to the validator to avoid querying the database again, but ideally, it should be - # computed inside the validator - RecordsBulkUpsertValidator.validate(bulk_upsert, dataset, found_records) records = [] async with self._db.begin_nested(): - for record_upsert in bulk_upsert.items: + for idx, record_upsert in enumerate(bulk_upsert.items): record = found_records.get(record_upsert.id) or found_records.get(record_upsert.external_id) if not record: + try: + RecordCreateValidator.validate(RecordCreate.parse_obj(record_upsert), dataset) + except (UnprocessableEntityError, ValueError) as ex: + raise UnprocessableEntityError(f"record at position {idx} is not valid because {ex}") from ex + record = Record( fields=jsonable_encoder(record_upsert.fields), metadata_=record_upsert.metadata, external_id=record_upsert.external_id, dataset_id=dataset.id, ) - elif self._metadata_is_set(record_upsert): - record.metadata_ = record_upsert.metadata - record.updated_at = datetime.utcnow() - records.append(record) + records.append(record) + else: + try: + RecordUpdateValidator.validate(RecordUpdate.parse_obj(record_upsert), dataset) + except (UnprocessableEntityError, ValueError) as ex: + raise UnprocessableEntityError(f"record at position {idx} is not valid because {ex}") from ex + + if self._metadata_is_set(record_upsert): + record.metadata_ = record_upsert.metadata + record.updated_at = datetime.utcnow() + + records.append(record) self._db.add_all(records) await self._db.flush(records) diff --git a/argilla-server/src/argilla_server/validators/records.py b/argilla-server/src/argilla_server/validators/records.py index 345b4386b0..43aa462cc6 100644 --- a/argilla-server/src/argilla_server/validators/records.py +++ b/argilla-server/src/argilla_server/validators/records.py @@ -226,40 +226,3 @@ def _validate_all_bulk_records(dataset: Dataset, records_create: List[RecordCrea RecordCreateValidator.validate(record_create, dataset) except UnprocessableEntityError as ex: raise UnprocessableEntityError(f"record at position {idx} is not valid because {ex}") from ex - - -class RecordsBulkUpsertValidator: - @classmethod - def validate( - cls, - records_upsert: RecordsBulkUpsert, - dataset: Dataset, - existing_records_by_external_id_or_record_id: Union[Dict[Union[str, UUID], Record], None] = None, - ) -> None: - cls._validate_dataset_is_ready(dataset) - cls._validate_all_bulk_records(dataset, records_upsert.items, existing_records_by_external_id_or_record_id) - - @staticmethod - def _validate_dataset_is_ready(dataset: Dataset) -> None: - if not dataset.is_ready: - raise UnprocessableEntityError("records cannot be created or updated for a non published dataset") - - @staticmethod - def _validate_all_bulk_records( - dataset: Dataset, - records_upsert: List[RecordUpsert], - existing_records_by_external_id_or_record_id: Union[Dict[Union[str, UUID], Record], None] = None, - ): - existing_records_by_external_id_or_record_id = existing_records_by_external_id_or_record_id or {} - for idx, record_upsert in enumerate(records_upsert): - try: - record = existing_records_by_external_id_or_record_id.get( - record_upsert.id - ) or existing_records_by_external_id_or_record_id.get(record_upsert.external_id) - - if record: - RecordUpdateValidator.validate(RecordUpdate.parse_obj(record_upsert), dataset) - else: - RecordCreateValidator.validate(RecordCreate.parse_obj(record_upsert), dataset) - except (UnprocessableEntityError, ValueError) as ex: - raise UnprocessableEntityError(f"record at position {idx} is not valid because {ex}") from ex diff --git a/argilla-server/tests/unit/api/handlers/v1/test_datasets.py b/argilla-server/tests/unit/api/handlers/v1/test_datasets.py index d362a4666d..550e640ca7 100644 --- a/argilla-server/tests/unit/api/handlers/v1/test_datasets.py +++ b/argilla-server/tests/unit/api/handlers/v1/test_datasets.py @@ -3119,10 +3119,11 @@ async def test_update_dataset_records_with_nonexistent_vector_settings_name( assert response.status_code == 422 + @pytest.mark.skip(reason="It's failing because we are not checking for duplicated ids by now") async def test_update_dataset_records_with_duplicate_records_ids( self, async_client: "AsyncClient", owner_auth_header: dict ): - dataset = await DatasetFactory.create() + dataset = await DatasetFactory.create(status=DatasetStatus.ready) record = await RecordFactory.create(dataset=dataset) response = await async_client.put( diff --git a/argilla-server/tests/unit/validators/test_records_bulk.py b/argilla-server/tests/unit/validators/test_records_bulk.py index 8b00ccc2dc..219ca59e2a 100644 --- a/argilla-server/tests/unit/validators/test_records_bulk.py +++ b/argilla-server/tests/unit/validators/test_records_bulk.py @@ -17,7 +17,7 @@ from argilla_server.api.schemas.v1.records_bulk import RecordsBulkCreate, RecordsBulkUpsert from argilla_server.errors.future import UnprocessableEntityError from argilla_server.models import Dataset -from argilla_server.validators.records import RecordsBulkCreateValidator, RecordsBulkUpsertValidator +from argilla_server.validators.records import RecordsBulkCreateValidator from sqlalchemy.ext.asyncio import AsyncSession from tests.factories import DatasetFactory, RecordFactory, TextFieldFactory @@ -95,44 +95,3 @@ async def test_records_bulk_create_validator_with_record_errors(self, db: AsyncS match="record at position 1 is not valid because", ): await RecordsBulkCreateValidator.validate(db, records_create, dataset) - - async def test_records_bulk_upsert_validator(self, db: AsyncSession): - dataset = await self.configure_dataset() - - records_upsert = RecordsBulkUpsert( - items=[ - RecordUpsert(fields={"text": "hello world"}, metadata={"source": "test"}), - ] - ) - - RecordsBulkUpsertValidator.validate(records_upsert, dataset) - - async def test_records_bulk_upsert_validator_with_draft_dataset(self, db: AsyncSession): - dataset = await DatasetFactory.create(status="draft") - - with pytest.raises( - UnprocessableEntityError, match="records cannot be created or updated for a non published dataset" - ): - records_upsert = RecordsBulkUpsert( - items=[ - RecordUpsert(fields={"text": "hello world"}, metadata={"source": "test"}), - ] - ) - - RecordsBulkUpsertValidator.validate(records_upsert, dataset) - - async def test_records_bulk_upsert_validator_with_record_error(self, db: AsyncSession): - dataset = await self.configure_dataset() - records_upsert = RecordsBulkUpsert( - items=[ - RecordUpsert(fields={"text": "hello world"}, metadata={"source": "test"}), - RecordUpsert(fields={"text": "hello world"}, metadata={"source": "test"}), - RecordUpsert(fields={"wrong-field": "hello world"}), - ] - ) - - with pytest.raises( - UnprocessableEntityError, - match="record at position 2 is not valid because", - ): - RecordsBulkUpsertValidator.validate(records_upsert, dataset) From a2fbc10464b35d3303a30eaccc0039e035f6cfad Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Tue, 15 Oct 2024 13:06:05 +0200 Subject: [PATCH 12/16] feat: ignore invalid records when importing datasets from hub --- .../src/argilla_server/bulk/records_bulk.py | 20 +++++++++-- .../src/argilla_server/contexts/hub.py | 8 +++-- .../unit/contexts/hub/test_hub_dataset.py | 36 +++++++++++++++++++ 3 files changed, 59 insertions(+), 5 deletions(-) diff --git a/argilla-server/src/argilla_server/bulk/records_bulk.py b/argilla-server/src/argilla_server/bulk/records_bulk.py index 6caec289be..aa5162699d 100644 --- a/argilla-server/src/argilla_server/bulk/records_bulk.py +++ b/argilla-server/src/argilla_server/bulk/records_bulk.py @@ -182,7 +182,9 @@ def _metadata_is_set(self, record_create: RecordCreate) -> bool: class UpsertRecordsBulk(CreateRecordsBulk): - async def upsert_records_bulk(self, dataset: Dataset, bulk_upsert: RecordsBulkUpsert) -> RecordsBulkWithUpdateInfo: + async def upsert_records_bulk( + self, dataset: Dataset, bulk_upsert: RecordsBulkUpsert, raise_on_error: bool = True + ) -> RecordsBulkWithUpdateInfo: found_records = await self._fetch_existing_dataset_records(dataset, bulk_upsert.items) records = [] @@ -193,7 +195,13 @@ async def upsert_records_bulk(self, dataset: Dataset, bulk_upsert: RecordsBulkUp try: RecordCreateValidator.validate(RecordCreate.parse_obj(record_upsert), dataset) except (UnprocessableEntityError, ValueError) as ex: - raise UnprocessableEntityError(f"record at position {idx} is not valid because {ex}") from ex + if raise_on_error: + raise UnprocessableEntityError( + f"record at position {idx} is not valid because {ex}" + ) from ex + else: + # NOTE: Ignore the errors in this record and continue with the next one + continue record = Record( fields=jsonable_encoder(record_upsert.fields), @@ -207,7 +215,13 @@ async def upsert_records_bulk(self, dataset: Dataset, bulk_upsert: RecordsBulkUp try: RecordUpdateValidator.validate(RecordUpdate.parse_obj(record_upsert), dataset) except (UnprocessableEntityError, ValueError) as ex: - raise UnprocessableEntityError(f"record at position {idx} is not valid because {ex}") from ex + if raise_on_error: + raise UnprocessableEntityError( + f"record at position {idx} is not valid because {ex}" + ) from ex + else: + # NOTE: Ignore the errors in this record and continue with the next one + continue if self._metadata_is_set(record_upsert): record.metadata_ = record_upsert.metadata diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index 6993775377..197c4b5dac 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -73,7 +73,11 @@ async def _import_batch_to( for i in range(batch_size): items.append(self._batch_row_to_record_schema(batch, i, dataset)) - await UpsertRecordsBulk(db, search_engine).upsert_records_bulk(dataset, RecordsBulkUpsertSchema(items=items)) + await UpsertRecordsBulk(db, search_engine).upsert_records_bulk( + dataset, + RecordsBulkUpsertSchema(items=items), + raise_on_error=False, + ) def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) -> RecordUpsertSchema: return RecordUpsertSchema( @@ -100,7 +104,7 @@ def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: if not field: continue - if field.is_text: + if field.is_text and value is not None: value = str(value) if field.is_image and isinstance(value, Image.Image): diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index bf0322d5b8..ea77ca23b2 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -157,6 +157,42 @@ async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, m == "" ) + async def test_hub_dataset_import_to_with_invalid_rows(self, db: AsyncSession, mock_search_engine: SearchEngine): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="letter", required=True, dataset=dataset) + await TextFieldFactory.create(name="count", dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset( + name="argilla-internal-testing/argilla-invalid-rows", + subset="default", + split="train", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="letter", target="letter"), + HubDatasetMappingItem(source="count", target="count"), + ], + external_id="id", + ), + ) + + await hub_dataset.import_to(db, mock_search_engine, dataset) + assert (await db.execute(select(func.count(Record.id)))).scalar_one() == 4 + + records = (await db.execute(select(Record))).scalars().all() + assert records[0].external_id == "1.0" + assert records[0].fields == {"letter": "A", "count": "100.0"} + assert records[1].external_id == "2.0" + assert records[1].fields == {"letter": "B", "count": "200.0"} + assert records[2].external_id == "4.0" + assert records[2].fields == {"letter": "D", "count": None} + assert records[3].external_id == "5.0" + assert records[3].fields == {"letter": "E", "count": "500.0"} + async def test_hub_dataset_import_to_idempotency_with_external_id( self, db: AsyncSession, mock_search_engine: SearchEngine ): From 3d1f04f0c23f4ee8cdf8e15eeaf0de6ddcf70398 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Thu, 17 Oct 2024 12:24:04 +0200 Subject: [PATCH 13/16] feat: add a fixed number of rows to take importing dataset from Hub (#5597) # Description I have added a number of row to take importing a dataset from the hub. Specifically 500K rows. This can help us to avoid importing really big datasets with millions of rows into Argilla. Refs https://github.com/argilla-io/roadmap/issues/21 **Type of change** - New feature (non-breaking change which adds functionality) **How Has This Been Tested** - [ ] Manually tested importing a dataset with a big number of records (> 500K) **Checklist** - I added relevant documentation - I followed the style guidelines of this project - I did a self-review of my code - I made corresponding changes to the documentation - I confirm My changes generate no new warnings - I have added tests that prove my fix is effective or that my feature works - I have added relevant notes to the CHANGELOG.md file (See https://keepachangelog.com/) --------- Co-authored-by: Paco Aranda --- argilla-server/src/argilla_server/jobs/hub_jobs.py | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/argilla-server/src/argilla_server/jobs/hub_jobs.py b/argilla-server/src/argilla_server/jobs/hub_jobs.py index 60c915f524..0315435b24 100644 --- a/argilla-server/src/argilla_server/jobs/hub_jobs.py +++ b/argilla-server/src/argilla_server/jobs/hub_jobs.py @@ -29,6 +29,8 @@ # TODO: Move this to be defined on jobs queues as a shared constant JOB_TIMEOUT_DISABLED = -1 +HUB_DATASET_TAKE_ROWS = 10_000 + # TODO: Once we merge webhooks we should change the queue to use a different one (default queue is deleted there) @job(DEFAULT_QUEUE, timeout=JOB_TIMEOUT_DISABLED, retry=Retry(max=3)) @@ -47,4 +49,8 @@ async def import_dataset_from_hub_job(name: str, subset: str, split: str, datase async with SearchEngine.get_by_name(settings.search_engine) as search_engine: parsed_mapping = HubDatasetMapping.parse_obj(mapping) - await HubDataset(name, subset, split, parsed_mapping).import_to(db, search_engine, dataset) + await ( + HubDataset(name, subset, split, parsed_mapping) + .take(HUB_DATASET_TAKE_ROWS) + .import_to(db, search_engine, dataset) + ) From 08ebe28226ac99d288d91284b8fd0bc1f322cb00 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 18 Oct 2024 12:55:08 +0200 Subject: [PATCH 14/16] feat: add support for class labels and casting rows (#5601) # Description This PR adds the following features: * Add support to dataset `ClassLabel` features casting them using `int2str` so we store string values on the Argilla imported dataset. * Now the casting is done at the row level and discarding keys that are not part of the mapping sources. Once the values arrive to the different record create values these are already casted. * We are casting `ClassLabel` features to string and `Image` features to data-url strings. Refs https://github.com/argilla-io/roadmap/issues/21 **Type of change** - New feature (non-breaking change which adds functionality) **How Has This Been Tested** - [x] Adding additional tests import real datasets from HF. **Checklist** - I added relevant documentation - I followed the style guidelines of this project - I did a self-review of my code - I made corresponding changes to the documentation - I confirm My changes generate no new warnings - I have added tests that prove my fix is effective or that my feature works - I have added relevant notes to the CHANGELOG.md file (See https://keepachangelog.com/) --- .../argilla_server/api/schemas/v1/datasets.py | 9 +++ .../src/argilla_server/contexts/hub.py | 58 +++++++++----- .../unit/contexts/hub/test_hub_dataset.py | 76 ++++++++++++++++++- 3 files changed, 123 insertions(+), 20 deletions(-) diff --git a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py index d2b28046ce..60272dc331 100644 --- a/argilla-server/src/argilla_server/api/schemas/v1/datasets.py +++ b/argilla-server/src/argilla_server/api/schemas/v1/datasets.py @@ -168,6 +168,15 @@ class HubDatasetMapping(BaseModel): suggestions: Optional[List[HubDatasetMappingItem]] = [] external_id: Optional[str] = None + @property + def sources(self) -> List[str]: + fields_sources = [field.source for field in self.fields] + metadata_sources = [metadata.source for metadata in self.metadata] + suggestions_sources = [suggestion.source for suggestion in self.suggestions] + external_id_source = [self.external_id] if self.external_id else [] + + return list(set(fields_sources + metadata_sources + suggestions_sources + external_id_source)) + class HubDataset(BaseModel): name: str diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index 197c4b5dac..e9e64fba91 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -22,7 +22,6 @@ from datasets import load_dataset from sqlalchemy.ext.asyncio import AsyncSession - from argilla_server.models.database import Dataset from argilla_server.search_engine import SearchEngine from argilla_server.bulk.records_bulk import UpsertRecordsBulk @@ -34,13 +33,21 @@ BATCH_SIZE = 100 RESET_ROW_IDX = -1 +FEATURE_TYPE_IMAGE = "Image" +FEATURE_TYPE_CLASS_LABEL = "ClassLabel" + class HubDataset: def __init__(self, name: str, subset: str, split: str, mapping: HubDatasetMapping): self.dataset = load_dataset(path=name, name=subset, split=split, streaming=True) self.mapping = mapping + self.mapping_feature_names = mapping.sources self.row_idx = RESET_ROW_IDX + @property + def features(self) -> dict: + return self.dataset.features + def take(self, n: int) -> Self: self.dataset = self.dataset.take(n) @@ -71,7 +78,7 @@ async def _import_batch_to( items = [] for i in range(batch_size): - items.append(self._batch_row_to_record_schema(batch, i, dataset)) + items.append(self._row_to_record_schema(self._batch_index_to_row(batch, i), dataset)) await UpsertRecordsBulk(db, search_engine).upsert_records_bulk( dataset, @@ -79,27 +86,45 @@ async def _import_batch_to( raise_on_error=False, ) - def _batch_row_to_record_schema(self, batch: dict, index: int, dataset: Dataset) -> RecordUpsertSchema: + def _batch_index_to_row(self, batch: dict, index: int) -> dict: + row = {} + for feature_name, values in batch.items(): + if not feature_name in self.mapping_feature_names: + continue + + value = values[index] + feature = self.features[feature_name] + + if feature._type == FEATURE_TYPE_CLASS_LABEL: + row[feature_name] = feature.int2str(value) + elif feature._type == FEATURE_TYPE_IMAGE and isinstance(value, Image.Image): + row[feature_name] = pil_image_to_data_url(value) + else: + row[feature_name] = value + + return row + + def _row_to_record_schema(self, row: dict, dataset: Dataset) -> RecordUpsertSchema: return RecordUpsertSchema( id=None, - external_id=self._batch_row_external_id(batch, index), - fields=self._batch_row_fields(batch, index, dataset), - metadata=self._batch_row_metadata(batch, index, dataset), - suggestions=self._batch_row_suggestions(batch, index, dataset), + external_id=self._row_external_id(row), + fields=self._row_fields(row, dataset), + metadata=self._row_metadata(row, dataset), + suggestions=self._row_suggestions(row, dataset), responses=None, vectors=None, ) - def _batch_row_external_id(self, batch: dict, index: int) -> str: + def _row_external_id(self, row: dict) -> str: if not self.mapping.external_id: return str(self._next_row_idx()) - return batch[self.mapping.external_id][index] + return row[self.mapping.external_id] - def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: + def _row_fields(self, row: dict, dataset: Dataset) -> dict: fields = {} for mapping_field in self.mapping.fields: - value = batch[mapping_field.source][index] + value = row[mapping_field.source] field = dataset.field_by_name(mapping_field.target) if not field: continue @@ -107,17 +132,14 @@ def _batch_row_fields(self, batch: dict, index: int, dataset: Dataset) -> dict: if field.is_text and value is not None: value = str(value) - if field.is_image and isinstance(value, Image.Image): - value = pil_image_to_data_url(value) - fields[field.name] = value return fields - def _batch_row_metadata(self, batch: dict, index: int, dataset: Dataset) -> dict: + def _row_metadata(self, row: dict, dataset: Dataset) -> dict: metadata = {} for mapping_metadata in self.mapping.metadata: - value = batch[mapping_metadata.source][index] + value = row[mapping_metadata.source] metadata_property = dataset.metadata_property_by_name(mapping_metadata.target) if not metadata_property: continue @@ -126,10 +148,10 @@ def _batch_row_metadata(self, batch: dict, index: int, dataset: Dataset) -> dict return metadata - def _batch_row_suggestions(self, batch: dict, index: int, dataset: Dataset) -> list: + def _row_suggestions(self, row: dict, dataset: Dataset) -> list: suggestions = [] for mapping_suggestion in self.mapping.suggestions: - value = batch[mapping_suggestion.source][index] + value = row[mapping_suggestion.source] question = dataset.question_by_name(mapping_suggestion.target) if not question: continue diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index ea77ca23b2..c3b4e930c7 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -27,7 +27,7 @@ from tests.factories import ( DatasetFactory, ImageFieldFactory, - RatingQuestionFactory, + QuestionFactory, TextFieldFactory, IntegerMetadataPropertyFactory, ) @@ -86,7 +86,7 @@ async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mo await TextFieldFactory.create(name="package_name", required=True, dataset=dataset) await TextFieldFactory.create(name="review", required=True, dataset=dataset) - question = await RatingQuestionFactory.create( + question = await QuestionFactory.create( name="star", required=True, dataset=dataset, @@ -127,6 +127,78 @@ async def test_hub_dataset_import_to_with_suggestions(self, db: AsyncSession, mo assert record.suggestions[0].value == 4 assert record.suggestions[0].question_id == question.id + async def test_hub_dataset_import_to_with_class_label_suggestions( + self, db: AsyncSession, mock_search_engine: SearchEngine + ): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="text", required=True, dataset=dataset) + + question = await QuestionFactory.create( + name="label", + settings={ + "type": QuestionType.label_selection, + "options": [ + {"value": "neg", "text": "Negative"}, + {"value": "pos", "text": "Positive"}, + ], + }, + dataset=dataset, + ) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset( + name="stanfordnlp/imdb", + subset="plain_text", + split="train", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="text", target="text"), + ], + suggestions=[ + HubDatasetMappingItem(source="label", target="label"), + ], + ), + ) + + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) + + record = (await db.execute(select(Record))).scalar_one() + assert record.suggestions[0].value == "neg" + assert record.suggestions[0].question_id == question.id + + async def test_hub_dataset_import_to_with_class_label_fields( + self, db: AsyncSession, mock_search_engine: SearchEngine + ): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="text", required=True, dataset=dataset) + await TextFieldFactory.create(name="label", required=True, dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset( + name="stanfordnlp/imdb", + subset="plain_text", + split="train", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="text", target="text"), + HubDatasetMappingItem(source="label", target="label"), + ], + ), + ) + + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) + + record = (await db.execute(select(Record))).scalar_one() + assert record.fields["label"] == "neg" + async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, mock_search_engine: SearchEngine): dataset = await DatasetFactory.create(status=DatasetStatus.ready) From 312551a343361e232ab0fa2270ace59940c6d57f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 18 Oct 2024 12:56:45 +0200 Subject: [PATCH 15/16] feat: improve `HubDataset` image processing support (#5606) # Description This PR adds the following changes related to how `HubDataset` import functionality process rows with images: * If the image has not format we transform the image to `png`. * We convert images to `RGB` color space to avoid problems with other unsupported color spaces. Refs https://github.com/argilla-io/roadmap/issues/21 **Type of change** - New feature (non-breaking change which adds functionality) **How Has This Been Tested** - [x] Manually testing `microsoft/cats_vs_dogs` dataset. **Checklist** - I added relevant documentation - I followed the style guidelines of this project - I did a self-review of my code - I made corresponding changes to the documentation - I confirm My changes generate no new warnings - I have added tests that prove my fix is effective or that my feature works - I have added relevant notes to the CHANGELOG.md file (See https://keepachangelog.com/) --- argilla-server/src/argilla_server/contexts/hub.py | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index e9e64fba91..31faf5cf82 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -36,6 +36,9 @@ FEATURE_TYPE_IMAGE = "Image" FEATURE_TYPE_CLASS_LABEL = "ClassLabel" +DATA_URL_DEFAULT_IMAGE_FORMAT = "png" +DATA_URL_DEFAULT_IMAGE_MIMETYPE = "image/png" + class HubDataset: def __init__(self, name: str, subset: str, split: str, mapping: HubDatasetMapping): @@ -178,8 +181,11 @@ def _row_suggestions(self, row: dict, dataset: Dataset) -> list: def pil_image_to_data_url(image: Image.Image): buffer = io.BytesIO() - image.save(buffer, format=image.format) + image_format = image.format or DATA_URL_DEFAULT_IMAGE_FORMAT + image_mimetype = image.get_format_mimetype() if image.format else DATA_URL_DEFAULT_IMAGE_MIMETYPE + + image.convert("RGB").save(buffer, format=image_format) base64_image = base64.b64encode(buffer.getvalue()).decode("utf-8") - return f"data:{image.get_format_mimetype()};base64,{base64_image}" + return f"data:{image_mimetype};base64,{base64_image}" From f06426701f98ca097fa4490b93988352ed056e4c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Francisco=20Calvo?= Date: Fri, 18 Oct 2024 13:03:19 +0200 Subject: [PATCH 16/16] feat: add support to `-1` no label values for `ClassLabel` dataset features (#5607) # Description This PR adds the following changes: * Add support to `ClassLabel` features using `-1` (no label) values and casting these values to be `None`. * Now when importing `fields`, `metadata`, or `suggestions` and the value is `None` they will be ignored. Refs https://github.com/argilla-io/roadmap/issues/21 **Type of change** - New feature (non-breaking change which adds functionality) **How Has This Been Tested** - [x] Adding new tests and modifying existing ones. **Checklist** - I added relevant documentation - I followed the style guidelines of this project - I did a self-review of my code - I made corresponding changes to the documentation - I confirm My changes generate no new warnings - I have added tests that prove my fix is effective or that my feature works - I have added relevant notes to the CHANGELOG.md file (See https://keepachangelog.com/) --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> --- .../src/argilla_server/contexts/hub.py | 13 +++- .../unit/contexts/hub/test_hub_dataset.py | 73 ++++++++++++++++++- 2 files changed, 81 insertions(+), 5 deletions(-) diff --git a/argilla-server/src/argilla_server/contexts/hub.py b/argilla-server/src/argilla_server/contexts/hub.py index 31faf5cf82..27bd1b8f5f 100644 --- a/argilla-server/src/argilla_server/contexts/hub.py +++ b/argilla-server/src/argilla_server/contexts/hub.py @@ -36,6 +36,8 @@ FEATURE_TYPE_IMAGE = "Image" FEATURE_TYPE_CLASS_LABEL = "ClassLabel" +FEATURE_CLASS_LABEL_NO_LABEL = -1 + DATA_URL_DEFAULT_IMAGE_FORMAT = "png" DATA_URL_DEFAULT_IMAGE_MIMETYPE = "image/png" @@ -99,7 +101,10 @@ def _batch_index_to_row(self, batch: dict, index: int) -> dict: feature = self.features[feature_name] if feature._type == FEATURE_TYPE_CLASS_LABEL: - row[feature_name] = feature.int2str(value) + if value == FEATURE_CLASS_LABEL_NO_LABEL: + row[feature_name] = None + else: + row[feature_name] = feature.int2str(value) elif feature._type == FEATURE_TYPE_IMAGE and isinstance(value, Image.Image): row[feature_name] = pil_image_to_data_url(value) else: @@ -129,7 +134,7 @@ def _row_fields(self, row: dict, dataset: Dataset) -> dict: for mapping_field in self.mapping.fields: value = row[mapping_field.source] field = dataset.field_by_name(mapping_field.target) - if not field: + if value is None or not field: continue if field.is_text and value is not None: @@ -144,7 +149,7 @@ def _row_metadata(self, row: dict, dataset: Dataset) -> dict: for mapping_metadata in self.mapping.metadata: value = row[mapping_metadata.source] metadata_property = dataset.metadata_property_by_name(mapping_metadata.target) - if not metadata_property: + if value is None or not metadata_property: continue metadata[metadata_property.name] = value @@ -156,7 +161,7 @@ def _row_suggestions(self, row: dict, dataset: Dataset) -> list: for mapping_suggestion in self.mapping.suggestions: value = row[mapping_suggestion.source] question = dataset.question_by_name(mapping_suggestion.target) - if not question: + if value is None or not question: continue if question.is_text or question.is_label_selection: diff --git a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py index c3b4e930c7..4adfceeefd 100644 --- a/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py +++ b/argilla-server/tests/unit/contexts/hub/test_hub_dataset.py @@ -199,6 +199,77 @@ async def test_hub_dataset_import_to_with_class_label_fields( record = (await db.execute(select(Record))).scalar_one() assert record.fields["label"] == "neg" + async def test_hub_dataset_import_to_with_class_label_suggestions_using_no_label( + self, db: AsyncSession, mock_search_engine: SearchEngine + ): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="text", required=True, dataset=dataset) + + question = await QuestionFactory.create( + name="label", + settings={ + "type": QuestionType.label_selection, + "options": [ + {"value": "neg", "text": "Negative"}, + {"value": "pos", "text": "Positive"}, + ], + }, + dataset=dataset, + ) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset( + name="stanfordnlp/imdb", + subset="plain_text", + split="unsupervised", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="text", target="text"), + ], + suggestions=[ + HubDatasetMappingItem(source="label", target="label"), + ], + ), + ) + + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) + + record = (await db.execute(select(Record))).scalar_one() + assert record.suggestions == [] + + async def test_hub_dataset_import_to_with_class_label_fields_using_no_label( + self, db: AsyncSession, mock_search_engine: SearchEngine + ): + dataset = await DatasetFactory.create(status=DatasetStatus.ready) + + await TextFieldFactory.create(name="text", required=True, dataset=dataset) + await TextFieldFactory.create(name="label", dataset=dataset) + + await dataset.awaitable_attrs.fields + await dataset.awaitable_attrs.questions + await dataset.awaitable_attrs.metadata_properties + + hub_dataset = HubDataset( + name="stanfordnlp/imdb", + subset="plain_text", + split="unsupervised", + mapping=HubDatasetMapping( + fields=[ + HubDatasetMappingItem(source="text", target="text"), + HubDatasetMappingItem(source="label", target="label"), + ], + ), + ) + + await hub_dataset.take(1).import_to(db, mock_search_engine, dataset) + + record = (await db.execute(select(Record))).scalar_one() + assert "label" not in record.fields + async def test_hub_dataset_import_to_with_image_fields(self, db: AsyncSession, mock_search_engine: SearchEngine): dataset = await DatasetFactory.create(status=DatasetStatus.ready) @@ -261,7 +332,7 @@ async def test_hub_dataset_import_to_with_invalid_rows(self, db: AsyncSession, m assert records[1].external_id == "2.0" assert records[1].fields == {"letter": "B", "count": "200.0"} assert records[2].external_id == "4.0" - assert records[2].fields == {"letter": "D", "count": None} + assert records[2].fields == {"letter": "D"} assert records[3].external_id == "5.0" assert records[3].fields == {"letter": "E", "count": "500.0"}