From 1881c4df164cad93ff00755327a1efbbb479a7be Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 03:08:47 +0000 Subject: [PATCH 01/16] refactor: change CI from Travis to GitHub Actions --- .github/workflows/ci.yml | 28 ++++++++++++++++++++++++++++ .gitignore | 2 ++ .travis.yml | 10 ---------- README.md | 2 +- 4 files changed, 31 insertions(+), 11 deletions(-) create mode 100644 .github/workflows/ci.yml delete mode 100644 .travis.yml diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..69180b1 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,28 @@ +name: CI + +on: + push: + pull_request: + +jobs: + test: + runs-on: ubuntu-latest + + strategy: + matrix: + python-version: [3.11, 3.12, 3.13, 3.14] + + steps: + - name: Check out code + uses: actions/checkout@v5 + + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v6 + with: + python-version: ${{ matrix.python-version }} + + - name: Install dependencies + run: pip install -r requirements.txt + + - name: Run tests + run: PYTHONPATH=. python -m unittest tests \ No newline at end of file diff --git a/.gitignore b/.gitignore index dc612bb..930e751 100644 --- a/.gitignore +++ b/.gitignore @@ -91,3 +91,5 @@ tmp temp node_modules bower_components + +.devcontainer \ No newline at end of file diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 597b9b4..0000000 --- a/.travis.yml +++ /dev/null @@ -1,10 +0,0 @@ -language: python -python: - - "2.7" - - "3.5" - - "3.6" - - "3.7" -install: - - pip install -r requirements.txt -script: - - PYTHONPATH=. python -m unittest tests diff --git a/README.md b/README.md index 6da98c5..affb38f 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ pybacklog Backlog API v2 Client Library for Python -[![Build Status](https://travis-ci.org/netmarkjp/pybacklog.svg?branch=master)](https://travis-ci.org/netmarkjp/pybacklog) +[![CI](https://github.com/netmarkjp/pybacklog/actions/workflows/ci.yml/badge.svg)](https://github.com/netmarkjp/pybacklog/actions/workflows/ci.yml) # Requirements From 1004a865f948406667800741fe87cbbe4b15b981 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 05:47:38 +0000 Subject: [PATCH 02/16] refactor: change toolset to pyproject.toml, uv and ruff --- .github/workflows/ci.yml | 12 +- pyproject.toml | 42 +++++ requirements.txt | 1 - requirements_dev.txt | 4 - setup.py | 19 -- uv.lock | 362 +++++++++++++++++++++++++++++++++++++++ 6 files changed, 413 insertions(+), 27 deletions(-) create mode 100644 pyproject.toml delete mode 100644 requirements.txt delete mode 100644 requirements_dev.txt delete mode 100755 setup.py create mode 100644 uv.lock diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 69180b1..b32e8d4 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -21,8 +21,14 @@ jobs: with: python-version: ${{ matrix.python-version }} - - name: Install dependencies - run: pip install -r requirements.txt + - name: Install uv + run: pip install uv + + - name: Sync dependencies + run: uv sync --group dev + + - name: Run ruff + run: uv run ruff . - name: Run tests - run: PYTHONPATH=. python -m unittest tests \ No newline at end of file + run: uv run python3 -m unittest tests \ No newline at end of file diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..1c5ad55 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,42 @@ +[build-system] +requires = ["setuptools", "wheel"] +build-backend = "setuptools.build_meta" + +[project] +name = "pybacklog" +version = "0.1.8" +description = "Backlog API v2 Client" +readme = "README.md" +license = { file = "LICENSE" } +authors = [ + { name = "Toshiaki Baba", email = "toshiaki@netmark.jp" } +] +urls = { "Homepage" = "https://github.com/netmarkjp/pybacklog" } +classifiers = [ + "Programming Language :: Python :: 3.11", + "Programming Language :: Python :: 3.12", + "Programming Language :: Python :: 3.13", + "Programming Language :: Python :: 3.14", + "Topic :: Utilities" +] +requires-python = ">=3.11, <3.15" + +dependencies = [ + "requests >=2.12.4,<3.0" +] + +[dependency-groups] +dev = [ + "ipython>=9.7.0", + "ruff>=0.14.6", +] + +[tool.uv] +# Additional uv-specific configurations can be added here. + +[tool.ruff] +target-version = "py311" +line-length = 88 +select = ["E", "F", "W"] +ignore = [] # Add specific error codes to ignore if needed +exclude = [".venv", "build", "dist"] diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index d2a612a..0000000 --- a/requirements.txt +++ /dev/null @@ -1 +0,0 @@ -requests >=2.12.4,<3.0 diff --git a/requirements_dev.txt b/requirements_dev.txt deleted file mode 100644 index 88aea91..0000000 --- a/requirements_dev.txt +++ /dev/null @@ -1,4 +0,0 @@ -autopep8 -flake8 -ipython -pyflakes diff --git a/setup.py b/setup.py deleted file mode 100755 index 826ad2e..0000000 --- a/setup.py +++ /dev/null @@ -1,19 +0,0 @@ -from setuptools import setup - -setup( - name="pybacklog", - version="0.1.8", - description="Backlog API v2 Client", - author="Toshiaki Baba", - author_email="toshiaki@netmark.jp", - url="https://github.com/netmarkjp/pybacklog", - classifiers=[ - "Programming Language :: Python :: 2.7", - "Programming Language :: Python :: 3.5", - "Programming Language :: Python :: 3.6", - "Programming Language :: Python :: 3.7", - "Topic :: Utilities", - ], - packages=["pybacklog"], - install_requires=["requests"], -) diff --git a/uv.lock b/uv.lock new file mode 100644 index 0000000..9c6a2cd --- /dev/null +++ b/uv.lock @@ -0,0 +1,362 @@ +version = 1 +revision = 3 +requires-python = ">=3.11, <3.15" + +[[package]] +name = "asttokens" +version = "3.0.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/be/a5/8e3f9b6771b0b408517c82d97aed8f2036509bc247d46114925e32fe33f0/asttokens-3.0.1.tar.gz", hash = "sha256:71a4ee5de0bde6a31d64f6b13f2293ac190344478f081c3d1bccfcf5eacb0cb7", size = 62308, upload-time = "2025-11-15T16:43:48.578Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d2/39/e7eaf1799466a4aef85b6a4fe7bd175ad2b1c6345066aa33f1f58d4b18d0/asttokens-3.0.1-py3-none-any.whl", hash = "sha256:15a3ebc0f43c2d0a50eeafea25e19046c68398e487b9f1f5b517f7c0f40f976a", size = 27047, upload-time = "2025-11-15T16:43:16.109Z" }, +] + +[[package]] +name = "certifi" +version = "2025.11.12" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/a2/8c/58f469717fa48465e4a50c014a0400602d3c437d7c0c468e17ada824da3a/certifi-2025.11.12.tar.gz", hash = "sha256:d8ab5478f2ecd78af242878415affce761ca6bc54a22a27e026d7c25357c3316", size = 160538, upload-time = "2025-11-12T02:54:51.517Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/70/7d/9bc192684cea499815ff478dfcdc13835ddf401365057044fb721ec6bddb/certifi-2025.11.12-py3-none-any.whl", hash = "sha256:97de8790030bbd5c2d96b7ec782fc2f7820ef8dba6db909ccf95449f2d062d4b", size = 159438, upload-time = "2025-11-12T02:54:49.735Z" }, +] + +[[package]] +name = "charset-normalizer" +version = "3.4.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/13/69/33ddede1939fdd074bce5434295f38fae7136463422fe4fd3e0e89b98062/charset_normalizer-3.4.4.tar.gz", hash = "sha256:94537985111c35f28720e43603b8e7b43a6ecfb2ce1d3058bbe955b73404e21a", size = 129418, upload-time = "2025-10-14T04:42:32.879Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ed/27/c6491ff4954e58a10f69ad90aca8a1b6fe9c5d3c6f380907af3c37435b59/charset_normalizer-3.4.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:6e1fcf0720908f200cd21aa4e6750a48ff6ce4afe7ff5a79a90d5ed8a08296f8", size = 206988, upload-time = "2025-10-14T04:40:33.79Z" }, + { url = "https://files.pythonhosted.org/packages/94/59/2e87300fe67ab820b5428580a53cad894272dbb97f38a7a814a2a1ac1011/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5f819d5fe9234f9f82d75bdfa9aef3a3d72c4d24a6e57aeaebba32a704553aa0", size = 147324, upload-time = "2025-10-14T04:40:34.961Z" }, + { url = "https://files.pythonhosted.org/packages/07/fb/0cf61dc84b2b088391830f6274cb57c82e4da8bbc2efeac8c025edb88772/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:a59cb51917aa591b1c4e6a43c132f0cdc3c76dbad6155df4e28ee626cc77a0a3", size = 142742, upload-time = "2025-10-14T04:40:36.105Z" }, + { url = "https://files.pythonhosted.org/packages/62/8b/171935adf2312cd745d290ed93cf16cf0dfe320863ab7cbeeae1dcd6535f/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:8ef3c867360f88ac904fd3f5e1f902f13307af9052646963ee08ff4f131adafc", size = 160863, upload-time = "2025-10-14T04:40:37.188Z" }, + { url = "https://files.pythonhosted.org/packages/09/73/ad875b192bda14f2173bfc1bc9a55e009808484a4b256748d931b6948442/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d9e45d7faa48ee908174d8fe84854479ef838fc6a705c9315372eacbc2f02897", size = 157837, upload-time = "2025-10-14T04:40:38.435Z" }, + { url = "https://files.pythonhosted.org/packages/6d/fc/de9cce525b2c5b94b47c70a4b4fb19f871b24995c728e957ee68ab1671ea/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:840c25fb618a231545cbab0564a799f101b63b9901f2569faecd6b222ac72381", size = 151550, upload-time = "2025-10-14T04:40:40.053Z" }, + { url = "https://files.pythonhosted.org/packages/55/c2/43edd615fdfba8c6f2dfbd459b25a6b3b551f24ea21981e23fb768503ce1/charset_normalizer-3.4.4-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ca5862d5b3928c4940729dacc329aa9102900382fea192fc5e52eb69d6093815", size = 149162, upload-time = "2025-10-14T04:40:41.163Z" }, + { url = "https://files.pythonhosted.org/packages/03/86/bde4ad8b4d0e9429a4e82c1e8f5c659993a9a863ad62c7df05cf7b678d75/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d9c7f57c3d666a53421049053eaacdd14bbd0a528e2186fcb2e672effd053bb0", size = 150019, upload-time = "2025-10-14T04:40:42.276Z" }, + { url = "https://files.pythonhosted.org/packages/1f/86/a151eb2af293a7e7bac3a739b81072585ce36ccfb4493039f49f1d3cae8c/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:277e970e750505ed74c832b4bf75dac7476262ee2a013f5574dd49075879e161", size = 143310, upload-time = "2025-10-14T04:40:43.439Z" }, + { url = "https://files.pythonhosted.org/packages/b5/fe/43dae6144a7e07b87478fdfc4dbe9efd5defb0e7ec29f5f58a55aeef7bf7/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:31fd66405eaf47bb62e8cd575dc621c56c668f27d46a61d975a249930dd5e2a4", size = 162022, upload-time = "2025-10-14T04:40:44.547Z" }, + { url = "https://files.pythonhosted.org/packages/80/e6/7aab83774f5d2bca81f42ac58d04caf44f0cc2b65fc6db2b3b2e8a05f3b3/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:0d3d8f15c07f86e9ff82319b3d9ef6f4bf907608f53fe9d92b28ea9ae3d1fd89", size = 149383, upload-time = "2025-10-14T04:40:46.018Z" }, + { url = "https://files.pythonhosted.org/packages/4f/e8/b289173b4edae05c0dde07f69f8db476a0b511eac556dfe0d6bda3c43384/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:9f7fcd74d410a36883701fafa2482a6af2ff5ba96b9a620e9e0721e28ead5569", size = 159098, upload-time = "2025-10-14T04:40:47.081Z" }, + { url = "https://files.pythonhosted.org/packages/d8/df/fe699727754cae3f8478493c7f45f777b17c3ef0600e28abfec8619eb49c/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ebf3e58c7ec8a8bed6d66a75d7fb37b55e5015b03ceae72a8e7c74495551e224", size = 152991, upload-time = "2025-10-14T04:40:48.246Z" }, + { url = "https://files.pythonhosted.org/packages/1a/86/584869fe4ddb6ffa3bd9f491b87a01568797fb9bd8933f557dba9771beaf/charset_normalizer-3.4.4-cp311-cp311-win32.whl", hash = "sha256:eecbc200c7fd5ddb9a7f16c7decb07b566c29fa2161a16cf67b8d068bd21690a", size = 99456, upload-time = "2025-10-14T04:40:49.376Z" }, + { url = "https://files.pythonhosted.org/packages/65/f6/62fdd5feb60530f50f7e38b4f6a1d5203f4d16ff4f9f0952962c044e919a/charset_normalizer-3.4.4-cp311-cp311-win_amd64.whl", hash = "sha256:5ae497466c7901d54b639cf42d5b8c1b6a4fead55215500d2f486d34db48d016", size = 106978, upload-time = "2025-10-14T04:40:50.844Z" }, + { url = "https://files.pythonhosted.org/packages/7a/9d/0710916e6c82948b3be62d9d398cb4fcf4e97b56d6a6aeccd66c4b2f2bd5/charset_normalizer-3.4.4-cp311-cp311-win_arm64.whl", hash = "sha256:65e2befcd84bc6f37095f5961e68a6f077bf44946771354a28ad434c2cce0ae1", size = 99969, upload-time = "2025-10-14T04:40:52.272Z" }, + { url = "https://files.pythonhosted.org/packages/f3/85/1637cd4af66fa687396e757dec650f28025f2a2f5a5531a3208dc0ec43f2/charset_normalizer-3.4.4-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0a98e6759f854bd25a58a73fa88833fba3b7c491169f86ce1180c948ab3fd394", size = 208425, upload-time = "2025-10-14T04:40:53.353Z" }, + { url = "https://files.pythonhosted.org/packages/9d/6a/04130023fef2a0d9c62d0bae2649b69f7b7d8d24ea5536feef50551029df/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b5b290ccc2a263e8d185130284f8501e3e36c5e02750fc6b6bdeb2e9e96f1e25", size = 148162, upload-time = "2025-10-14T04:40:54.558Z" }, + { url = "https://files.pythonhosted.org/packages/78/29/62328d79aa60da22c9e0b9a66539feae06ca0f5a4171ac4f7dc285b83688/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:74bb723680f9f7a6234dcf67aea57e708ec1fbdf5699fb91dfd6f511b0a320ef", size = 144558, upload-time = "2025-10-14T04:40:55.677Z" }, + { url = "https://files.pythonhosted.org/packages/86/bb/b32194a4bf15b88403537c2e120b817c61cd4ecffa9b6876e941c3ee38fe/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f1e34719c6ed0b92f418c7c780480b26b5d9c50349e9a9af7d76bf757530350d", size = 161497, upload-time = "2025-10-14T04:40:57.217Z" }, + { url = "https://files.pythonhosted.org/packages/19/89/a54c82b253d5b9b111dc74aca196ba5ccfcca8242d0fb64146d4d3183ff1/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:2437418e20515acec67d86e12bf70056a33abdacb5cb1655042f6538d6b085a8", size = 159240, upload-time = "2025-10-14T04:40:58.358Z" }, + { url = "https://files.pythonhosted.org/packages/c0/10/d20b513afe03acc89ec33948320a5544d31f21b05368436d580dec4e234d/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:11d694519d7f29d6cd09f6ac70028dba10f92f6cdd059096db198c283794ac86", size = 153471, upload-time = "2025-10-14T04:40:59.468Z" }, + { url = "https://files.pythonhosted.org/packages/61/fa/fbf177b55bdd727010f9c0a3c49eefa1d10f960e5f09d1d887bf93c2e698/charset_normalizer-3.4.4-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ac1c4a689edcc530fc9d9aa11f5774b9e2f33f9a0c6a57864e90908f5208d30a", size = 150864, upload-time = "2025-10-14T04:41:00.623Z" }, + { url = "https://files.pythonhosted.org/packages/05/12/9fbc6a4d39c0198adeebbde20b619790e9236557ca59fc40e0e3cebe6f40/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:21d142cc6c0ec30d2efee5068ca36c128a30b0f2c53c1c07bd78cb6bc1d3be5f", size = 150647, upload-time = "2025-10-14T04:41:01.754Z" }, + { url = "https://files.pythonhosted.org/packages/ad/1f/6a9a593d52e3e8c5d2b167daf8c6b968808efb57ef4c210acb907c365bc4/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:5dbe56a36425d26d6cfb40ce79c314a2e4dd6211d51d6d2191c00bed34f354cc", size = 145110, upload-time = "2025-10-14T04:41:03.231Z" }, + { url = "https://files.pythonhosted.org/packages/30/42/9a52c609e72471b0fc54386dc63c3781a387bb4fe61c20231a4ebcd58bdd/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:5bfbb1b9acf3334612667b61bd3002196fe2a1eb4dd74d247e0f2a4d50ec9bbf", size = 162839, upload-time = "2025-10-14T04:41:04.715Z" }, + { url = "https://files.pythonhosted.org/packages/c4/5b/c0682bbf9f11597073052628ddd38344a3d673fda35a36773f7d19344b23/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:d055ec1e26e441f6187acf818b73564e6e6282709e9bcb5b63f5b23068356a15", size = 150667, upload-time = "2025-10-14T04:41:05.827Z" }, + { url = "https://files.pythonhosted.org/packages/e4/24/a41afeab6f990cf2daf6cb8c67419b63b48cf518e4f56022230840c9bfb2/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:af2d8c67d8e573d6de5bc30cdb27e9b95e49115cd9baad5ddbd1a6207aaa82a9", size = 160535, upload-time = "2025-10-14T04:41:06.938Z" }, + { url = "https://files.pythonhosted.org/packages/2a/e5/6a4ce77ed243c4a50a1fecca6aaaab419628c818a49434be428fe24c9957/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:780236ac706e66881f3b7f2f32dfe90507a09e67d1d454c762cf642e6e1586e0", size = 154816, upload-time = "2025-10-14T04:41:08.101Z" }, + { url = "https://files.pythonhosted.org/packages/a8/ef/89297262b8092b312d29cdb2517cb1237e51db8ecef2e9af5edbe7b683b1/charset_normalizer-3.4.4-cp312-cp312-win32.whl", hash = "sha256:5833d2c39d8896e4e19b689ffc198f08ea58116bee26dea51e362ecc7cd3ed26", size = 99694, upload-time = "2025-10-14T04:41:09.23Z" }, + { url = "https://files.pythonhosted.org/packages/3d/2d/1e5ed9dd3b3803994c155cd9aacb60c82c331bad84daf75bcb9c91b3295e/charset_normalizer-3.4.4-cp312-cp312-win_amd64.whl", hash = "sha256:a79cfe37875f822425b89a82333404539ae63dbdddf97f84dcbc3d339aae9525", size = 107131, upload-time = "2025-10-14T04:41:10.467Z" }, + { url = "https://files.pythonhosted.org/packages/d0/d9/0ed4c7098a861482a7b6a95603edce4c0d9db2311af23da1fb2b75ec26fc/charset_normalizer-3.4.4-cp312-cp312-win_arm64.whl", hash = "sha256:376bec83a63b8021bb5c8ea75e21c4ccb86e7e45ca4eb81146091b56599b80c3", size = 100390, upload-time = "2025-10-14T04:41:11.915Z" }, + { url = "https://files.pythonhosted.org/packages/97/45/4b3a1239bbacd321068ea6e7ac28875b03ab8bc0aa0966452db17cd36714/charset_normalizer-3.4.4-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:e1f185f86a6f3403aa2420e815904c67b2f9ebc443f045edd0de921108345794", size = 208091, upload-time = "2025-10-14T04:41:13.346Z" }, + { url = "https://files.pythonhosted.org/packages/7d/62/73a6d7450829655a35bb88a88fca7d736f9882a27eacdca2c6d505b57e2e/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6b39f987ae8ccdf0d2642338faf2abb1862340facc796048b604ef14919e55ed", size = 147936, upload-time = "2025-10-14T04:41:14.461Z" }, + { url = "https://files.pythonhosted.org/packages/89/c5/adb8c8b3d6625bef6d88b251bbb0d95f8205831b987631ab0c8bb5d937c2/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:3162d5d8ce1bb98dd51af660f2121c55d0fa541b46dff7bb9b9f86ea1d87de72", size = 144180, upload-time = "2025-10-14T04:41:15.588Z" }, + { url = "https://files.pythonhosted.org/packages/91/ed/9706e4070682d1cc219050b6048bfd293ccf67b3d4f5a4f39207453d4b99/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:81d5eb2a312700f4ecaa977a8235b634ce853200e828fbadf3a9c50bab278328", size = 161346, upload-time = "2025-10-14T04:41:16.738Z" }, + { url = "https://files.pythonhosted.org/packages/d5/0d/031f0d95e4972901a2f6f09ef055751805ff541511dc1252ba3ca1f80cf5/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:5bd2293095d766545ec1a8f612559f6b40abc0eb18bb2f5d1171872d34036ede", size = 158874, upload-time = "2025-10-14T04:41:17.923Z" }, + { url = "https://files.pythonhosted.org/packages/f5/83/6ab5883f57c9c801ce5e5677242328aa45592be8a00644310a008d04f922/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a8a8b89589086a25749f471e6a900d3f662d1d3b6e2e59dcecf787b1cc3a1894", size = 153076, upload-time = "2025-10-14T04:41:19.106Z" }, + { url = "https://files.pythonhosted.org/packages/75/1e/5ff781ddf5260e387d6419959ee89ef13878229732732ee73cdae01800f2/charset_normalizer-3.4.4-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:bc7637e2f80d8530ee4a78e878bce464f70087ce73cf7c1caf142416923b98f1", size = 150601, upload-time = "2025-10-14T04:41:20.245Z" }, + { url = "https://files.pythonhosted.org/packages/d7/57/71be810965493d3510a6ca79b90c19e48696fb1ff964da319334b12677f0/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:f8bf04158c6b607d747e93949aa60618b61312fe647a6369f88ce2ff16043490", size = 150376, upload-time = "2025-10-14T04:41:21.398Z" }, + { url = "https://files.pythonhosted.org/packages/e5/d5/c3d057a78c181d007014feb7e9f2e65905a6c4ef182c0ddf0de2924edd65/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:554af85e960429cf30784dd47447d5125aaa3b99a6f0683589dbd27e2f45da44", size = 144825, upload-time = "2025-10-14T04:41:22.583Z" }, + { url = "https://files.pythonhosted.org/packages/e6/8c/d0406294828d4976f275ffbe66f00266c4b3136b7506941d87c00cab5272/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:74018750915ee7ad843a774364e13a3db91682f26142baddf775342c3f5b1133", size = 162583, upload-time = "2025-10-14T04:41:23.754Z" }, + { url = "https://files.pythonhosted.org/packages/d7/24/e2aa1f18c8f15c4c0e932d9287b8609dd30ad56dbe41d926bd846e22fb8d/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:c0463276121fdee9c49b98908b3a89c39be45d86d1dbaa22957e38f6321d4ce3", size = 150366, upload-time = "2025-10-14T04:41:25.27Z" }, + { url = "https://files.pythonhosted.org/packages/e4/5b/1e6160c7739aad1e2df054300cc618b06bf784a7a164b0f238360721ab86/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:362d61fd13843997c1c446760ef36f240cf81d3ebf74ac62652aebaf7838561e", size = 160300, upload-time = "2025-10-14T04:41:26.725Z" }, + { url = "https://files.pythonhosted.org/packages/7a/10/f882167cd207fbdd743e55534d5d9620e095089d176d55cb22d5322f2afd/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:9a26f18905b8dd5d685d6d07b0cdf98a79f3c7a918906af7cc143ea2e164c8bc", size = 154465, upload-time = "2025-10-14T04:41:28.322Z" }, + { url = "https://files.pythonhosted.org/packages/89/66/c7a9e1b7429be72123441bfdbaf2bc13faab3f90b933f664db506dea5915/charset_normalizer-3.4.4-cp313-cp313-win32.whl", hash = "sha256:9b35f4c90079ff2e2edc5b26c0c77925e5d2d255c42c74fdb70fb49b172726ac", size = 99404, upload-time = "2025-10-14T04:41:29.95Z" }, + { url = "https://files.pythonhosted.org/packages/c4/26/b9924fa27db384bdcd97ab83b4f0a8058d96ad9626ead570674d5e737d90/charset_normalizer-3.4.4-cp313-cp313-win_amd64.whl", hash = "sha256:b435cba5f4f750aa6c0a0d92c541fb79f69a387c91e61f1795227e4ed9cece14", size = 107092, upload-time = "2025-10-14T04:41:31.188Z" }, + { url = "https://files.pythonhosted.org/packages/af/8f/3ed4bfa0c0c72a7ca17f0380cd9e4dd842b09f664e780c13cff1dcf2ef1b/charset_normalizer-3.4.4-cp313-cp313-win_arm64.whl", hash = "sha256:542d2cee80be6f80247095cc36c418f7bddd14f4a6de45af91dfad36d817bba2", size = 100408, upload-time = "2025-10-14T04:41:32.624Z" }, + { url = "https://files.pythonhosted.org/packages/2a/35/7051599bd493e62411d6ede36fd5af83a38f37c4767b92884df7301db25d/charset_normalizer-3.4.4-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:da3326d9e65ef63a817ecbcc0df6e94463713b754fe293eaa03da99befb9a5bd", size = 207746, upload-time = "2025-10-14T04:41:33.773Z" }, + { url = "https://files.pythonhosted.org/packages/10/9a/97c8d48ef10d6cd4fcead2415523221624bf58bcf68a802721a6bc807c8f/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8af65f14dc14a79b924524b1e7fffe304517b2bff5a58bf64f30b98bbc5079eb", size = 147889, upload-time = "2025-10-14T04:41:34.897Z" }, + { url = "https://files.pythonhosted.org/packages/10/bf/979224a919a1b606c82bd2c5fa49b5c6d5727aa47b4312bb27b1734f53cd/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:74664978bb272435107de04e36db5a9735e78232b85b77d45cfb38f758efd33e", size = 143641, upload-time = "2025-10-14T04:41:36.116Z" }, + { url = "https://files.pythonhosted.org/packages/ba/33/0ad65587441fc730dc7bd90e9716b30b4702dc7b617e6ba4997dc8651495/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:752944c7ffbfdd10c074dc58ec2d5a8a4cd9493b314d367c14d24c17684ddd14", size = 160779, upload-time = "2025-10-14T04:41:37.229Z" }, + { url = "https://files.pythonhosted.org/packages/67/ed/331d6b249259ee71ddea93f6f2f0a56cfebd46938bde6fcc6f7b9a3d0e09/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d1f13550535ad8cff21b8d757a3257963e951d96e20ec82ab44bc64aeb62a191", size = 159035, upload-time = "2025-10-14T04:41:38.368Z" }, + { url = "https://files.pythonhosted.org/packages/67/ff/f6b948ca32e4f2a4576aa129d8bed61f2e0543bf9f5f2b7fc3758ed005c9/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ecaae4149d99b1c9e7b88bb03e3221956f68fd6d50be2ef061b2381b61d20838", size = 152542, upload-time = "2025-10-14T04:41:39.862Z" }, + { url = "https://files.pythonhosted.org/packages/16/85/276033dcbcc369eb176594de22728541a925b2632f9716428c851b149e83/charset_normalizer-3.4.4-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:cb6254dc36b47a990e59e1068afacdcd02958bdcce30bb50cc1700a8b9d624a6", size = 149524, upload-time = "2025-10-14T04:41:41.319Z" }, + { url = "https://files.pythonhosted.org/packages/9e/f2/6a2a1f722b6aba37050e626530a46a68f74e63683947a8acff92569f979a/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:c8ae8a0f02f57a6e61203a31428fa1d677cbe50c93622b4149d5c0f319c1d19e", size = 150395, upload-time = "2025-10-14T04:41:42.539Z" }, + { url = "https://files.pythonhosted.org/packages/60/bb/2186cb2f2bbaea6338cad15ce23a67f9b0672929744381e28b0592676824/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:47cc91b2f4dd2833fddaedd2893006b0106129d4b94fdb6af1f4ce5a9965577c", size = 143680, upload-time = "2025-10-14T04:41:43.661Z" }, + { url = "https://files.pythonhosted.org/packages/7d/a5/bf6f13b772fbb2a90360eb620d52ed8f796f3c5caee8398c3b2eb7b1c60d/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:82004af6c302b5d3ab2cfc4cc5f29db16123b1a8417f2e25f9066f91d4411090", size = 162045, upload-time = "2025-10-14T04:41:44.821Z" }, + { url = "https://files.pythonhosted.org/packages/df/c5/d1be898bf0dc3ef9030c3825e5d3b83f2c528d207d246cbabe245966808d/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:2b7d8f6c26245217bd2ad053761201e9f9680f8ce52f0fcd8d0755aeae5b2152", size = 149687, upload-time = "2025-10-14T04:41:46.442Z" }, + { url = "https://files.pythonhosted.org/packages/a5/42/90c1f7b9341eef50c8a1cb3f098ac43b0508413f33affd762855f67a410e/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:799a7a5e4fb2d5898c60b640fd4981d6a25f1c11790935a44ce38c54e985f828", size = 160014, upload-time = "2025-10-14T04:41:47.631Z" }, + { url = "https://files.pythonhosted.org/packages/76/be/4d3ee471e8145d12795ab655ece37baed0929462a86e72372fd25859047c/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:99ae2cffebb06e6c22bdc25801d7b30f503cc87dbd283479e7b606f70aff57ec", size = 154044, upload-time = "2025-10-14T04:41:48.81Z" }, + { url = "https://files.pythonhosted.org/packages/b0/6f/8f7af07237c34a1defe7defc565a9bc1807762f672c0fde711a4b22bf9c0/charset_normalizer-3.4.4-cp314-cp314-win32.whl", hash = "sha256:f9d332f8c2a2fcbffe1378594431458ddbef721c1769d78e2cbc06280d8155f9", size = 99940, upload-time = "2025-10-14T04:41:49.946Z" }, + { url = "https://files.pythonhosted.org/packages/4b/51/8ade005e5ca5b0d80fb4aff72a3775b325bdc3d27408c8113811a7cbe640/charset_normalizer-3.4.4-cp314-cp314-win_amd64.whl", hash = "sha256:8a6562c3700cce886c5be75ade4a5db4214fda19fede41d9792d100288d8f94c", size = 107104, upload-time = "2025-10-14T04:41:51.051Z" }, + { url = "https://files.pythonhosted.org/packages/da/5f/6b8f83a55bb8278772c5ae54a577f3099025f9ade59d0136ac24a0df4bde/charset_normalizer-3.4.4-cp314-cp314-win_arm64.whl", hash = "sha256:de00632ca48df9daf77a2c65a484531649261ec9f25489917f09e455cb09ddb2", size = 100743, upload-time = "2025-10-14T04:41:52.122Z" }, + { url = "https://files.pythonhosted.org/packages/0a/4c/925909008ed5a988ccbb72dcc897407e5d6d3bd72410d69e051fc0c14647/charset_normalizer-3.4.4-py3-none-any.whl", hash = "sha256:7a32c560861a02ff789ad905a2fe94e3f840803362c84fecf1851cb4cf3dc37f", size = 53402, upload-time = "2025-10-14T04:42:31.76Z" }, +] + +[[package]] +name = "colorama" +version = "0.4.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" }, +] + +[[package]] +name = "decorator" +version = "5.2.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/43/fa/6d96a0978d19e17b68d634497769987b16c8f4cd0a7a05048bec693caa6b/decorator-5.2.1.tar.gz", hash = "sha256:65f266143752f734b0a7cc83c46f4618af75b8c5911b00ccb61d0ac9b6da0360", size = 56711, upload-time = "2025-02-24T04:41:34.073Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/4e/8c/f3147f5c4b73e7550fe5f9352eaa956ae838d5c51eb58e7a25b9f3e2643b/decorator-5.2.1-py3-none-any.whl", hash = "sha256:d316bb415a2d9e2d2b3abcc4084c6502fc09240e292cd76a76afc106a1c8e04a", size = 9190, upload-time = "2025-02-24T04:41:32.565Z" }, +] + +[[package]] +name = "executing" +version = "2.2.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/cc/28/c14e053b6762b1044f34a13aab6859bbf40456d37d23aa286ac24cfd9a5d/executing-2.2.1.tar.gz", hash = "sha256:3632cc370565f6648cc328b32435bd120a1e4ebb20c77e3fdde9a13cd1e533c4", size = 1129488, upload-time = "2025-09-01T09:48:10.866Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/c1/ea/53f2148663b321f21b5a606bd5f191517cf40b7072c0497d3c92c4a13b1e/executing-2.2.1-py2.py3-none-any.whl", hash = "sha256:760643d3452b4d777d295bb167ccc74c64a81df23fb5e08eff250c425a4b2017", size = 28317, upload-time = "2025-09-01T09:48:08.5Z" }, +] + +[[package]] +name = "idna" +version = "3.11" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/6f/6d/0703ccc57f3a7233505399edb88de3cbd678da106337b9fcde432b65ed60/idna-3.11.tar.gz", hash = "sha256:795dafcc9c04ed0c1fb032c2aa73654d8e8c5023a7df64a53f39190ada629902", size = 194582, upload-time = "2025-10-12T14:55:20.501Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/0e/61/66938bbb5fc52dbdf84594873d5b51fb1f7c7794e9c0f5bd885f30bc507b/idna-3.11-py3-none-any.whl", hash = "sha256:771a87f49d9defaf64091e6e6fe9c18d4833f140bd19464795bc32d966ca37ea", size = 71008, upload-time = "2025-10-12T14:55:18.883Z" }, +] + +[[package]] +name = "ipython" +version = "9.7.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "colorama", marker = "sys_platform == 'win32'" }, + { name = "decorator" }, + { name = "ipython-pygments-lexers" }, + { name = "jedi" }, + { name = "matplotlib-inline" }, + { name = "pexpect", marker = "sys_platform != 'emscripten' and sys_platform != 'win32'" }, + { name = "prompt-toolkit" }, + { name = "pygments" }, + { name = "stack-data" }, + { name = "traitlets" }, + { name = "typing-extensions", marker = "python_full_version < '3.12'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/29/e6/48c74d54039241a456add616464ea28c6ebf782e4110d419411b83dae06f/ipython-9.7.0.tar.gz", hash = "sha256:5f6de88c905a566c6a9d6c400a8fed54a638e1f7543d17aae2551133216b1e4e", size = 4422115, upload-time = "2025-11-05T12:18:54.646Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/05/aa/62893d6a591d337aa59dcc4c6f6c842f1fe20cd72c8c5c1f980255243252/ipython-9.7.0-py3-none-any.whl", hash = "sha256:bce8ac85eb9521adc94e1845b4c03d88365fd6ac2f4908ec4ed1eb1b0a065f9f", size = 618911, upload-time = "2025-11-05T12:18:52.484Z" }, +] + +[[package]] +name = "ipython-pygments-lexers" +version = "1.1.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "pygments" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ef/4c/5dd1d8af08107f88c7f741ead7a40854b8ac24ddf9ae850afbcf698aa552/ipython_pygments_lexers-1.1.1.tar.gz", hash = "sha256:09c0138009e56b6854f9535736f4171d855c8c08a563a0dcd8022f78355c7e81", size = 8393, upload-time = "2025-01-17T11:24:34.505Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d9/33/1f075bf72b0b747cb3288d011319aaf64083cf2efef8354174e3ed4540e2/ipython_pygments_lexers-1.1.1-py3-none-any.whl", hash = "sha256:a9462224a505ade19a605f71f8fa63c2048833ce50abc86768a0d81d876dc81c", size = 8074, upload-time = "2025-01-17T11:24:33.271Z" }, +] + +[[package]] +name = "jedi" +version = "0.19.2" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "parso" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/72/3a/79a912fbd4d8dd6fbb02bf69afd3bb72cf0c729bb3063c6f4498603db17a/jedi-0.19.2.tar.gz", hash = "sha256:4770dc3de41bde3966b02eb84fbcf557fb33cce26ad23da12c742fb50ecb11f0", size = 1231287, upload-time = "2024-11-11T01:41:42.873Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/c0/5a/9cac0c82afec3d09ccd97c8b6502d48f165f9124db81b4bcb90b4af974ee/jedi-0.19.2-py2.py3-none-any.whl", hash = "sha256:a8ef22bde8490f57fe5c7681a3c83cb58874daf72b4784de3cce5b6ef6edb5b9", size = 1572278, upload-time = "2024-11-11T01:41:40.175Z" }, +] + +[[package]] +name = "matplotlib-inline" +version = "0.2.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "traitlets" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c7/74/97e72a36efd4ae2bccb3463284300f8953f199b5ffbc04cbbb0ec78f74b1/matplotlib_inline-0.2.1.tar.gz", hash = "sha256:e1ee949c340d771fc39e241ea75683deb94762c8fa5f2927ec57c83c4dffa9fe", size = 8110, upload-time = "2025-10-23T09:00:22.126Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/af/33/ee4519fa02ed11a94aef9559552f3b17bb863f2ecfe1a35dc7f548cde231/matplotlib_inline-0.2.1-py3-none-any.whl", hash = "sha256:d56ce5156ba6085e00a9d54fead6ed29a9c47e215cd1bba2e976ef39f5710a76", size = 9516, upload-time = "2025-10-23T09:00:20.675Z" }, +] + +[[package]] +name = "parso" +version = "0.8.5" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d4/de/53e0bcf53d13e005bd8c92e7855142494f41171b34c2536b86187474184d/parso-0.8.5.tar.gz", hash = "sha256:034d7354a9a018bdce352f48b2a8a450f05e9d6ee85db84764e9b6bd96dafe5a", size = 401205, upload-time = "2025-08-23T15:15:28.028Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/16/32/f8e3c85d1d5250232a5d3477a2a28cc291968ff175caeadaf3cc19ce0e4a/parso-0.8.5-py2.py3-none-any.whl", hash = "sha256:646204b5ee239c396d040b90f9e272e9a8017c630092bf59980beb62fd033887", size = 106668, upload-time = "2025-08-23T15:15:25.663Z" }, +] + +[[package]] +name = "pexpect" +version = "4.9.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "ptyprocess" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/42/92/cc564bf6381ff43ce1f4d06852fc19a2f11d180f23dc32d9588bee2f149d/pexpect-4.9.0.tar.gz", hash = "sha256:ee7d41123f3c9911050ea2c2dac107568dc43b2d3b0c7557a33212c398ead30f", size = 166450, upload-time = "2023-11-25T09:07:26.339Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/9e/c3/059298687310d527a58bb01f3b1965787ee3b40dce76752eda8b44e9a2c5/pexpect-4.9.0-py2.py3-none-any.whl", hash = "sha256:7236d1e080e4936be2dc3e326cec0af72acf9212a7e1d060210e70a47e253523", size = 63772, upload-time = "2023-11-25T06:56:14.81Z" }, +] + +[[package]] +name = "prompt-toolkit" +version = "3.0.52" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "wcwidth" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/a1/96/06e01a7b38dce6fe1db213e061a4602dd6032a8a97ef6c1a862537732421/prompt_toolkit-3.0.52.tar.gz", hash = "sha256:28cde192929c8e7321de85de1ddbe736f1375148b02f2e17edd840042b1be855", size = 434198, upload-time = "2025-08-27T15:24:02.057Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/84/03/0d3ce49e2505ae70cf43bc5bb3033955d2fc9f932163e84dc0779cc47f48/prompt_toolkit-3.0.52-py3-none-any.whl", hash = "sha256:9aac639a3bbd33284347de5ad8d68ecc044b91a762dc39b7c21095fcd6a19955", size = 391431, upload-time = "2025-08-27T15:23:59.498Z" }, +] + +[[package]] +name = "ptyprocess" +version = "0.7.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/20/e5/16ff212c1e452235a90aeb09066144d0c5a6a8c0834397e03f5224495c4e/ptyprocess-0.7.0.tar.gz", hash = "sha256:5c5d0a3b48ceee0b48485e0c26037c0acd7d29765ca3fbb5cb3831d347423220", size = 70762, upload-time = "2020-12-28T15:15:30.155Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/22/a6/858897256d0deac81a172289110f31629fc4cee19b6f01283303e18c8db3/ptyprocess-0.7.0-py2.py3-none-any.whl", hash = "sha256:4b41f3967fce3af57cc7e94b888626c18bf37a083e3651ca8feeb66d492fef35", size = 13993, upload-time = "2020-12-28T15:15:28.35Z" }, +] + +[[package]] +name = "pure-eval" +version = "0.2.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/cd/05/0a34433a064256a578f1783a10da6df098ceaa4a57bbeaa96a6c0352786b/pure_eval-0.2.3.tar.gz", hash = "sha256:5f4e983f40564c576c7c8635ae88db5956bb2229d7e9237d03b3c0b0190eaf42", size = 19752, upload-time = "2024-07-21T12:58:21.801Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/8e/37/efad0257dc6e593a18957422533ff0f87ede7c9c6ea010a2177d738fb82f/pure_eval-0.2.3-py3-none-any.whl", hash = "sha256:1db8e35b67b3d218d818ae653e27f06c3aa420901fa7b081ca98cbedc874e0d0", size = 11842, upload-time = "2024-07-21T12:58:20.04Z" }, +] + +[[package]] +name = "pybacklog" +version = "0.1.8" +source = { editable = "." } +dependencies = [ + { name = "requests" }, +] + +[package.dev-dependencies] +dev = [ + { name = "ipython" }, + { name = "ruff" }, +] + +[package.metadata] +requires-dist = [{ name = "requests", specifier = ">=2.12.4,<3.0" }] + +[package.metadata.requires-dev] +dev = [ + { name = "ipython", specifier = ">=9.7.0" }, + { name = "ruff", specifier = ">=0.14.6" }, +] + +[[package]] +name = "pygments" +version = "2.19.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/b0/77/a5b8c569bf593b0140bde72ea885a803b82086995367bf2037de0159d924/pygments-2.19.2.tar.gz", hash = "sha256:636cb2477cec7f8952536970bc533bc43743542f70392ae026374600add5b887", size = 4968631, upload-time = "2025-06-21T13:39:12.283Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217, upload-time = "2025-06-21T13:39:07.939Z" }, +] + +[[package]] +name = "requests" +version = "2.32.5" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "certifi" }, + { name = "charset-normalizer" }, + { name = "idna" }, + { name = "urllib3" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c9/74/b3ff8e6c8446842c3f5c837e9c3dfcfe2018ea6ecef224c710c85ef728f4/requests-2.32.5.tar.gz", hash = "sha256:dbba0bac56e100853db0ea71b82b4dfd5fe2bf6d3754a8893c3af500cec7d7cf", size = 134517, upload-time = "2025-08-18T20:46:02.573Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1e/db/4254e3eabe8020b458f1a747140d32277ec7a271daf1d235b70dc0b4e6e3/requests-2.32.5-py3-none-any.whl", hash = "sha256:2462f94637a34fd532264295e186976db0f5d453d1cdd31473c85a6a161affb6", size = 64738, upload-time = "2025-08-18T20:46:00.542Z" }, +] + +[[package]] +name = "ruff" +version = "0.14.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/52/f0/62b5a1a723fe183650109407fa56abb433b00aa1c0b9ba555f9c4efec2c6/ruff-0.14.6.tar.gz", hash = "sha256:6f0c742ca6a7783a736b867a263b9a7a80a45ce9bee391eeda296895f1b4e1cc", size = 5669501, upload-time = "2025-11-21T14:26:17.903Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/67/d2/7dd544116d107fffb24a0064d41a5d2ed1c9d6372d142f9ba108c8e39207/ruff-0.14.6-py3-none-linux_armv6l.whl", hash = "sha256:d724ac2f1c240dbd01a2ae98db5d1d9a5e1d9e96eba999d1c48e30062df578a3", size = 13326119, upload-time = "2025-11-21T14:25:24.2Z" }, + { url = "https://files.pythonhosted.org/packages/36/6a/ad66d0a3315d6327ed6b01f759d83df3c4d5f86c30462121024361137b6a/ruff-0.14.6-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:9f7539ea257aa4d07b7ce87aed580e485c40143f2473ff2f2b75aee003186004", size = 13526007, upload-time = "2025-11-21T14:25:26.906Z" }, + { url = "https://files.pythonhosted.org/packages/a3/9d/dae6db96df28e0a15dea8e986ee393af70fc97fd57669808728080529c37/ruff-0.14.6-py3-none-macosx_11_0_arm64.whl", hash = "sha256:7f6007e55b90a2a7e93083ba48a9f23c3158c433591c33ee2e99a49b889c6332", size = 12676572, upload-time = "2025-11-21T14:25:29.826Z" }, + { url = "https://files.pythonhosted.org/packages/76/a4/f319e87759949062cfee1b26245048e92e2acce900ad3a909285f9db1859/ruff-0.14.6-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0a8e7b9d73d8728b68f632aa8e824ef041d068d231d8dbc7808532d3629a6bef", size = 13140745, upload-time = "2025-11-21T14:25:32.788Z" }, + { url = "https://files.pythonhosted.org/packages/95/d3/248c1efc71a0a8ed4e8e10b4b2266845d7dfc7a0ab64354afe049eaa1310/ruff-0.14.6-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:d50d45d4553a3ebcbd33e7c5e0fe6ca4aafd9a9122492de357205c2c48f00775", size = 13076486, upload-time = "2025-11-21T14:25:35.601Z" }, + { url = "https://files.pythonhosted.org/packages/a5/19/b68d4563fe50eba4b8c92aa842149bb56dd24d198389c0ed12e7faff4f7d/ruff-0.14.6-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:118548dd121f8a21bfa8ab2c5b80e5b4aed67ead4b7567790962554f38e598ce", size = 13727563, upload-time = "2025-11-21T14:25:38.514Z" }, + { url = "https://files.pythonhosted.org/packages/47/ac/943169436832d4b0e867235abbdb57ce3a82367b47e0280fa7b4eabb7593/ruff-0.14.6-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:57256efafbfefcb8748df9d1d766062f62b20150691021f8ab79e2d919f7c11f", size = 15199755, upload-time = "2025-11-21T14:25:41.516Z" }, + { url = "https://files.pythonhosted.org/packages/c9/b9/288bb2399860a36d4bb0541cb66cce3c0f4156aaff009dc8499be0c24bf2/ruff-0.14.6-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ff18134841e5c68f8e5df1999a64429a02d5549036b394fafbe410f886e1989d", size = 14850608, upload-time = "2025-11-21T14:25:44.428Z" }, + { url = "https://files.pythonhosted.org/packages/ee/b1/a0d549dd4364e240f37e7d2907e97ee80587480d98c7799d2d8dc7a2f605/ruff-0.14.6-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:29c4b7ec1e66a105d5c27bd57fa93203637d66a26d10ca9809dc7fc18ec58440", size = 14118754, upload-time = "2025-11-21T14:25:47.214Z" }, + { url = "https://files.pythonhosted.org/packages/13/ac/9b9fe63716af8bdfddfacd0882bc1586f29985d3b988b3c62ddce2e202c3/ruff-0.14.6-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:167843a6f78680746d7e226f255d920aeed5e4ad9c03258094a2d49d3028b105", size = 13949214, upload-time = "2025-11-21T14:25:50.002Z" }, + { url = "https://files.pythonhosted.org/packages/12/27/4dad6c6a77fede9560b7df6802b1b697e97e49ceabe1f12baf3ea20862e9/ruff-0.14.6-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:16a33af621c9c523b1ae006b1b99b159bf5ac7e4b1f20b85b2572455018e0821", size = 14106112, upload-time = "2025-11-21T14:25:52.841Z" }, + { url = "https://files.pythonhosted.org/packages/6a/db/23e322d7177873eaedea59a7932ca5084ec5b7e20cb30f341ab594130a71/ruff-0.14.6-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:1432ab6e1ae2dc565a7eea707d3b03a0c234ef401482a6f1621bc1f427c2ff55", size = 13035010, upload-time = "2025-11-21T14:25:55.536Z" }, + { url = "https://files.pythonhosted.org/packages/a8/9c/20e21d4d69dbb35e6a1df7691e02f363423658a20a2afacf2a2c011800dc/ruff-0.14.6-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:4c55cfbbe7abb61eb914bfd20683d14cdfb38a6d56c6c66efa55ec6570ee4e71", size = 13054082, upload-time = "2025-11-21T14:25:58.625Z" }, + { url = "https://files.pythonhosted.org/packages/66/25/906ee6a0464c3125c8d673c589771a974965c2be1a1e28b5c3b96cb6ef88/ruff-0.14.6-py3-none-musllinux_1_2_i686.whl", hash = "sha256:efea3c0f21901a685fff4befda6d61a1bf4cb43de16da87e8226a281d614350b", size = 13303354, upload-time = "2025-11-21T14:26:01.816Z" }, + { url = "https://files.pythonhosted.org/packages/4c/58/60577569e198d56922b7ead07b465f559002b7b11d53f40937e95067ca1c/ruff-0.14.6-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:344d97172576d75dc6afc0e9243376dbe1668559c72de1864439c4fc95f78185", size = 14054487, upload-time = "2025-11-21T14:26:05.058Z" }, + { url = "https://files.pythonhosted.org/packages/67/0b/8e4e0639e4cc12547f41cb771b0b44ec8225b6b6a93393176d75fe6f7d40/ruff-0.14.6-py3-none-win32.whl", hash = "sha256:00169c0c8b85396516fdd9ce3446c7ca20c2a8f90a77aa945ba6b8f2bfe99e85", size = 13013361, upload-time = "2025-11-21T14:26:08.152Z" }, + { url = "https://files.pythonhosted.org/packages/fb/02/82240553b77fd1341f80ebb3eaae43ba011c7a91b4224a9f317d8e6591af/ruff-0.14.6-py3-none-win_amd64.whl", hash = "sha256:390e6480c5e3659f8a4c8d6a0373027820419ac14fa0d2713bd8e6c3e125b8b9", size = 14432087, upload-time = "2025-11-21T14:26:10.891Z" }, + { url = "https://files.pythonhosted.org/packages/a5/1f/93f9b0fad9470e4c829a5bb678da4012f0c710d09331b860ee555216f4ea/ruff-0.14.6-py3-none-win_arm64.whl", hash = "sha256:d43c81fbeae52cfa8728d8766bbf46ee4298c888072105815b392da70ca836b2", size = 13520930, upload-time = "2025-11-21T14:26:13.951Z" }, +] + +[[package]] +name = "stack-data" +version = "0.6.3" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "asttokens" }, + { name = "executing" }, + { name = "pure-eval" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/28/e3/55dcc2cfbc3ca9c29519eb6884dd1415ecb53b0e934862d3559ddcb7e20b/stack_data-0.6.3.tar.gz", hash = "sha256:836a778de4fec4dcd1dcd89ed8abff8a221f58308462e1c4aa2a3cf30148f0b9", size = 44707, upload-time = "2023-09-30T13:58:05.479Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f1/7b/ce1eafaf1a76852e2ec9b22edecf1daa58175c090266e9f6c64afcd81d91/stack_data-0.6.3-py3-none-any.whl", hash = "sha256:d5558e0c25a4cb0853cddad3d77da9891a08cb85dd9f9f91b9f8cd66e511e695", size = 24521, upload-time = "2023-09-30T13:58:03.53Z" }, +] + +[[package]] +name = "traitlets" +version = "5.14.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/eb/79/72064e6a701c2183016abbbfedaba506d81e30e232a68c9f0d6f6fcd1574/traitlets-5.14.3.tar.gz", hash = "sha256:9ed0579d3502c94b4b3732ac120375cda96f923114522847de4b3bb98b96b6b7", size = 161621, upload-time = "2024-04-19T11:11:49.746Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/00/c0/8f5d070730d7836adc9c9b6408dec68c6ced86b304a9b26a14df072a6e8c/traitlets-5.14.3-py3-none-any.whl", hash = "sha256:b74e89e397b1ed28cc831db7aea759ba6640cb3de13090ca145426688ff1ac4f", size = 85359, upload-time = "2024-04-19T11:11:46.763Z" }, +] + +[[package]] +name = "typing-extensions" +version = "4.15.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391, upload-time = "2025-08-25T13:49:26.313Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614, upload-time = "2025-08-25T13:49:24.86Z" }, +] + +[[package]] +name = "urllib3" +version = "2.5.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/15/22/9ee70a2574a4f4599c47dd506532914ce044817c7752a79b6a51286319bc/urllib3-2.5.0.tar.gz", hash = "sha256:3fc47733c7e419d4bc3f6b3dc2b4f890bb743906a30d56ba4a5bfa4bbff92760", size = 393185, upload-time = "2025-06-18T14:07:41.644Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a7/c2/fe1e52489ae3122415c51f387e221dd0773709bad6c6cdaa599e8a2c5185/urllib3-2.5.0-py3-none-any.whl", hash = "sha256:e6b01673c0fa6a13e374b50871808eb3bf7046c4b125b216f6bf1cc604cff0dc", size = 129795, upload-time = "2025-06-18T14:07:40.39Z" }, +] + +[[package]] +name = "wcwidth" +version = "0.2.14" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/24/30/6b0809f4510673dc723187aeaf24c7f5459922d01e2f794277a3dfb90345/wcwidth-0.2.14.tar.gz", hash = "sha256:4d478375d31bc5395a3c55c40ccdf3354688364cd61c4f6adacaa9215d0b3605", size = 102293, upload-time = "2025-09-22T16:29:53.023Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/af/b5/123f13c975e9f27ab9c0770f514345bd406d0e8d3b7a0723af9d43f710af/wcwidth-0.2.14-py2.py3-none-any.whl", hash = "sha256:a7bb560c8aee30f9957e5f9895805edd20602f2d7f720186dfd906e82b4982e1", size = 37286, upload-time = "2025-09-22T16:29:51.641Z" }, +] From 291d51bcbd499c35c0246f56d8cb91e6ee4dd55c Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 05:53:55 +0000 Subject: [PATCH 03/16] refactor: apply ruff format and check --- .github/workflows/ci.yml | 2 +- examples/project_activities.py | 16 +-- examples/project_activities2.py | 10 +- pybacklog/__init__.py | 170 +++++++++++++++++--------------- pyproject.toml | 2 +- tests/__init__.py | 13 +-- 6 files changed, 106 insertions(+), 107 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index b32e8d4..827b03c 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -28,7 +28,7 @@ jobs: run: uv sync --group dev - name: Run ruff - run: uv run ruff . + run: uv run ruff format . && uv run ruff check --fix . - name: Run tests run: uv run python3 -m unittest tests \ No newline at end of file diff --git a/examples/project_activities.py b/examples/project_activities.py index 557eb5c..417236c 100644 --- a/examples/project_activities.py +++ b/examples/project_activities.py @@ -9,11 +9,12 @@ client = BacklogClient(_space, _api_key) -activities = client.do("GET", "projects/{project_id_or_key}/activities", - url_params={"project_id_or_key": _project}, - query_params={"activityTypeId[]": [ - 1, 2, 3, 14], "count": 100} - ) +activities = client.do( + "GET", + "projects/{project_id_or_key}/activities", + url_params={"project_id_or_key": _project}, + query_params={"activityTypeId[]": [1, 2, 3, 14], "count": 100}, +) urls = [] items = [] @@ -23,9 +24,8 @@ continue urls.append(url) - item = (activity.get(u"created"), url, - activity.get(u"content").get(u"summary")) + item = (activity.get("created"), url, activity.get("content").get("summary")) items.append(item) for item in items: - print(u"{date}\t{url}\t{summary}".format(date=item[0], url=item[1], summary=item[2])) + print("{date}\t{url}\t{summary}".format(date=item[0], url=item[1], summary=item[2])) diff --git a/examples/project_activities2.py b/examples/project_activities2.py index 1568949..e665308 100644 --- a/examples/project_activities2.py +++ b/examples/project_activities2.py @@ -9,9 +9,7 @@ client = BacklogClient(_space, _api_key) -activities = client.project_activities( - _project, - {"activityTypeId[]": [1, 2, 3, 14], "count": 100}) +activities = client.project_activities(_project, {"activityTypeId[]": [1, 2, 3, 14], "count": 100}) urls = [] items = [] @@ -24,10 +22,8 @@ continue urls.append(url) - item = (activity.get(u"created"), url, - activity.get(u"content").get(u"summary")) + item = (activity.get("created"), url, activity.get("content").get("summary")) items.append(item) for item in items: - print(u"{date}\t{url}\t{summary}".format( - date=item[0], url=item[1], summary=item[2])) + print("{date}\t{url}\t{summary}".format(date=item[0], url=item[1], summary=item[2])) diff --git a/pybacklog/__init__.py b/pybacklog/__init__.py index fbb4a6f..d4d5b8f 100644 --- a/pybacklog/__init__.py +++ b/pybacklog/__init__.py @@ -5,7 +5,6 @@ class BacklogClient(object): - def __init__(self, space_name, api_key): self.space_name = space_name self.api_key = api_key @@ -13,7 +12,6 @@ def __init__(self, space_name, api_key): ## auto detetcion of space location self.endpoint = BacklogClient._detect_endpoint(space_name, api_key) - @staticmethod def _detect_endpoint(space_name, api_key): # at first try .com (new default) @@ -46,7 +44,6 @@ def _detect_endpoint(space_name, api_key): raise Exception("retrive space information failed. maybe space not found in .com nor .jp") - def do(self, method, url, url_params={}, query_params={}, request_params={}): """ - Method: method @@ -67,14 +64,11 @@ def do(self, method, url, url_params={}, query_params={}, request_params={}): if method == "get": resp = requests.get(_endpoint, params=query_params) elif method == "patch": - resp = requests.patch( - _endpoint, params=query_params, data=request_params, headers=_headers) + resp = requests.patch(_endpoint, params=query_params, data=request_params, headers=_headers) elif method == "post": - resp = requests.post( - _endpoint, params=query_params, data=request_params, headers=_headers) + resp = requests.post(_endpoint, params=query_params, data=request_params, headers=_headers) elif method == "delete": - resp = requests.delete( - _endpoint, params=query_params, data=request_params, headers=_headers) + resp = requests.delete(_endpoint, params=query_params, data=request_params, headers=_headers) else: raise Exception("Unsupported Method") @@ -91,24 +85,22 @@ def do(self, method, url, url_params={}, query_params={}, request_params={}): def activity_to_issue_url(self, activity): url = "https://{space}.backlog.jp/view/{project_key}-{content_id}".format( space=self.space_name, - project_key=activity.get(u"project").get(u"projectKey"), - content_id=activity.get(u"content").get(u"key_id"), + project_key=activity.get("project").get("projectKey"), + content_id=activity.get("content").get("key_id"), ) return url @staticmethod def remove_mb4(request_params): # remove 4 byte characters - pattern = re.compile(u"[^\u0000-\uD7FF\uE000-\uFFFF]", re.UNICODE) + pattern = re.compile("[^\u0000-\ud7ff\ue000-\uffff]", re.UNICODE) for key in request_params.keys(): try: if isinstance(request_params[key], unicode): - request_params[key] = pattern.sub( - u"\uFFFD", request_params[key]) + request_params[key] = pattern.sub("\ufffd", request_params[key]) except NameError: # maybe python3 - request_params[key] = pattern.sub( - u"\uFFFD", request_params[key]) + request_params[key] = pattern.sub("\ufffd", request_params[key]) return request_params # ------------------------------- @@ -140,28 +132,34 @@ def project_activities(self, project_id_or_key, extra_query_params={}): client.project_activities("YOUR_PROJECT") client.project_activities("YOUR_PROJECT", {"activityTypeId[]": [1, 2],}) """ - return self.do("get", "projects/{project_id_or_key}/activities", - url_params={"project_id_or_key": project_id_or_key}, - query_params=extra_query_params, - ) + return self.do( + "get", + "projects/{project_id_or_key}/activities", + url_params={"project_id_or_key": project_id_or_key}, + query_params=extra_query_params, + ) def project_users(self, project_id_or_key): """ client = BacklogClient("your_space_name", "your_api_key") client.project_users("YOUR_PROJECT") """ - return self.do("GET", "projects/{project_id_or_key}/users", - url_params={"project_id_or_key": project_id_or_key}, - ) + return self.do( + "GET", + "projects/{project_id_or_key}/users", + url_params={"project_id_or_key": project_id_or_key}, + ) def versions(self, project_id_or_key): """ client = BacklogClient("your_space_name", "your_api_key") client.versions(3) """ - return self.do("GET", "projects/{project_id_or_key}/versions", - url_params={"project_id_or_key": project_id_or_key}, - ) + return self.do( + "GET", + "projects/{project_id_or_key}/versions", + url_params={"project_id_or_key": project_id_or_key}, + ) def create_version(self, project_id_or_key, version_name, extra_request_params={}): """ @@ -173,12 +171,14 @@ def create_version(self, project_id_or_key, version_name, extra_request_params={ """ request_params = extra_request_params request_params["name"] = version_name - return self.do("POST", "projects/{project_id_or_key}/versions", - url_params={"project_id_or_key": project_id_or_key}, - request_params=request_params, - ) + return self.do( + "POST", + "projects/{project_id_or_key}/versions", + url_params={"project_id_or_key": project_id_or_key}, + request_params=request_params, + ) - def update_version(self, project_id_or_key, version_id, version_name, extra_request_params={}): + def update_version(self, project_id_or_key, version_id, version_name, extra_request_params={}): """ client = BacklogClient("your_space_name", "your_api_key") client.update_version("YOUR_PROJECT", @@ -187,24 +187,26 @@ def update_version(self, project_id_or_key, version_id, version_name, extra_requ {"description": "updated description", "archived": "true"}) """ - + request_params = extra_request_params request_params["name"] = version_name - return self.do("PATCH", "projects/{project_id_or_key}/versions/{version_id}", - url_params={"project_id_or_key": project_id_or_key, - "version_id": version_id}, - request_params=request_params, - ) + return self.do( + "PATCH", + "projects/{project_id_or_key}/versions/{version_id}", + url_params={"project_id_or_key": project_id_or_key, "version_id": version_id}, + request_params=request_params, + ) def delete_version(self, project_id_or_key, version_id): """ client = BacklogClient("your_space_name", "your_api_key") client.delete_version("YOUR_PROJECT", 3) """ - return self.do("DELETE", "projects/{project_id_or_key}/versions/{version_id}", - url_params={"project_id_or_key": project_id_or_key, - "version_id": version_id}, - ) + return self.do( + "DELETE", + "projects/{project_id_or_key}/versions/{version_id}", + url_params={"project_id_or_key": project_id_or_key, "version_id": version_id}, + ) def issues(self, extra_query_params={}): """ @@ -221,28 +223,34 @@ def issue(self, issue_id_or_key): client = BacklogClient("your_space_name", "your_api_key") client.issue("YOUR_PROJECT-999") """ - return self.do("GET", "issues/{issue_id_or_key}", - url_params={"issue_id_or_key": issue_id_or_key}, - ) + return self.do( + "GET", + "issues/{issue_id_or_key}", + url_params={"issue_id_or_key": issue_id_or_key}, + ) def issue_comments(self, issue_id_or_key, extra_query_params={}): """ client = BacklogClient("your_space_name", "your_api_key") client.issue_comments("YOUR_PROJECT-999") """ - return self.do("GET", "issues/{issue_id_or_key}/comments", - url_params={"issue_id_or_key": issue_id_or_key}, - query_params=extra_query_params - ) + return self.do( + "GET", + "issues/{issue_id_or_key}/comments", + url_params={"issue_id_or_key": issue_id_or_key}, + query_params=extra_query_params, + ) def project_issue_types(self, project_id_or_key): """ client = BacklogClient("your_space_name", "your_api_key") client.project_issue_types("YOUR_PROJECT") """ - return self.do("GET", "projects/{project_id_or_key}/issueTypes", - url_params={"project_id_or_key": project_id_or_key}, - ) + return self.do( + "GET", + "projects/{project_id_or_key}/issueTypes", + url_params={"project_id_or_key": project_id_or_key}, + ) def priorities(self): """ @@ -272,9 +280,11 @@ def create_issue(self, project_id, summary, issue_type_id, priority_id, extra_re request_params["issueTypeId"] = issue_type_id request_params["priorityId"] = priority_id - return self.do("POST", "issues", - request_params=request_params, - ) + return self.do( + "POST", + "issues", + request_params=request_params, + ) def add_issue_comment(self, issue_id_or_key, content, extra_request_params={}): """ @@ -283,10 +293,12 @@ def add_issue_comment(self, issue_id_or_key, content, extra_request_params={}): """ request_params = extra_request_params request_params["content"] = content - return self.do("POST", "issues/{issue_id_or_key}/comments", - url_params={"issue_id_or_key": issue_id_or_key}, - request_params=request_params, - ) + return self.do( + "POST", + "issues/{issue_id_or_key}/comments", + url_params={"issue_id_or_key": issue_id_or_key}, + request_params=request_params, + ) def users(self): """ @@ -308,9 +320,9 @@ def user_activities(self, user_id, extra_query_params={}): client.user_activities(3) client.user_activities(3, {"count": 2, "order": "asc"}) """ - return self.do("GET", "users/{user_id}/activities", - url_params={"user_id": user_id}, - query_params=extra_query_params) + return self.do( + "GET", "users/{user_id}/activities", url_params={"user_id": user_id}, query_params=extra_query_params + ) def groups(self, extra_query_params={}): """ @@ -332,9 +344,7 @@ def user_stars(self, user_id, extra_query_params={}): client.user_stars(5) client.user_stars(5, {"count": 100, "order": "asc"}) """ - return self.do("GET", "users/{user_id}/stars", - url_params={"user_id": user_id}, - query_params=extra_query_params) + return self.do("GET", "users/{user_id}/stars", url_params={"user_id": user_id}, query_params=extra_query_params) def user_stars_count(self, user_id, extra_query_params={}): """ @@ -342,25 +352,23 @@ def user_stars_count(self, user_id, extra_query_params={}): client.user_stars_count(5) client.user_stars_count(5, {"since": "2017-05-01", "until": "2017-05-31"}) """ - return self.do("GET", "users/{user_id}/stars/count", - url_params={"user_id": user_id}, - query_params=extra_query_params) + return self.do( + "GET", "users/{user_id}/stars/count", url_params={"user_id": user_id}, query_params=extra_query_params + ) def star(self, query_params): """ client = BacklogClient("your_space_name", "your_api_key") client.star({"issueId": 333}) """ - return self.do("POST", "stars", - query_params=query_params) + return self.do("POST", "stars", query_params=query_params) def wikis(self, project_id_or_key): """ client = BacklogClient("your_space_name", "your_api_key") client.wikis(3) """ - return self.do("GET", "wikis", - query_params={"projectIdOrKey": project_id_or_key}) + return self.do("GET", "wikis", query_params={"projectIdOrKey": project_id_or_key}) def wiki(self, wiki_id): """ @@ -376,7 +384,7 @@ def update_wiki(self, wiki_id, extra_request_params={}): """ request_params = extra_request_params return self.do("PATCH", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}, request_params=request_params) - + def wiki_history(self, wiki_id): """ client = BacklogClient("your_space_name", "your_api_key") @@ -396,9 +404,11 @@ def project_statuses(self, project_id_or_key): client = BacklogClient("your_space_name", "your_api_key") client.project_statuses("YOUR_PROJECT") """ - return self.do("GET", "projects/{project_id_or_key}/statuses", - url_params={"project_id_or_key": project_id_or_key}, - ) + return self.do( + "GET", + "projects/{project_id_or_key}/statuses", + url_params={"project_id_or_key": project_id_or_key}, + ) # ------------------------------- # extra utilities (PR welcome) @@ -407,15 +417,15 @@ def project_statuses(self, project_id_or_key): def get_project_id(self, project_key_or_name): projects = self.projects() for p in projects: - if p[u"projectKey"] == project_key_or_name: - return p[u"id"] + if p["projectKey"] == project_key_or_name: + return p["id"] for p in projects: - if p[u"name"] == project_key_or_name: - return p[u"id"] + if p["name"] == project_key_or_name: + return p["id"] return None def get_issue_id(self, issue_key): issue = self.issue(issue_key) if issue: - return int(issue[u"id"]) + return int(issue["id"]) return None diff --git a/pyproject.toml b/pyproject.toml index 1c5ad55..11a48bc 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -36,7 +36,7 @@ dev = [ [tool.ruff] target-version = "py311" -line-length = 88 +line-length = 120 select = ["E", "F", "W"] ignore = [] # Add specific error codes to ignore if needed exclude = [".venv", "build", "dist"] diff --git a/tests/__init__.py b/tests/__init__.py index 254b899..95e73b7 100644 --- a/tests/__init__.py +++ b/tests/__init__.py @@ -5,24 +5,17 @@ class TestBacklogClient(unittest.TestCase): - def test_init(self): try: - client = BacklogClient("my_space_name", "my_api_key") + _ = BacklogClient("my_space_name", "my_api_key") self.fail() except Exception as _ex: self.assertEqual(str(_ex), "retrive space information failed. maybe space not found in .com nor .jp") def test_remove_mb4(self): testing = ( - ( - {"equal1": u"あいう", "equal2": u"123123"}, - {"equal1": u"あいう", "equal2": u"123123"} - ), - ( - {"replaced1": u"あい💔", "replaced2": u"123♥23"}, - {"replaced1": u"あい\uFFFD", "replaced2": u"123♥23"} - ), + ({"equal1": "あいう", "equal2": "123123"}, {"equal1": "あいう", "equal2": "123123"}), + ({"replaced1": "あい💔", "replaced2": "123♥23"}, {"replaced1": "あい\ufffd", "replaced2": "123♥23"}), ) for t in testing: self.assertEqual(BacklogClient.remove_mb4(t[0]), t[1]) From 12c6b6cb18d3968b68c171cc8a0983dbd824e008 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 06:01:36 +0000 Subject: [PATCH 04/16] docs: follow-up changes for pyproject.toml and add dev dependency group --- README.md | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index affb38f..9bdd689 100644 --- a/README.md +++ b/README.md @@ -7,7 +7,7 @@ Backlog API v2 Client Library for Python # Requirements -- Python 2.7 or Python 3.5+ +- Python 3.11+ - requests 2.x # Install @@ -91,10 +91,8 @@ see also [Backlog API Overview \| Backlog Developer API \| Nulab](https://develo # Development ``` -pip install -r requirements.txt -pip install -r requirements_dev.txt - -PYTHONPATH=. python -m unittest tests +uv sync --group dev +uv run python3 -m unittest tests ``` # License From c9fe74739d96e50a70d235ae1b7b4b76251f9106 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 06:35:53 +0000 Subject: [PATCH 05/16] chore: apply ruff to pyproject.toml --- pyproject.toml | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index 11a48bc..0edc03c 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -35,8 +35,8 @@ dev = [ # Additional uv-specific configurations can be added here. [tool.ruff] -target-version = "py311" +target-version = "py314" line-length = 120 -select = ["E", "F", "W"] -ignore = [] # Add specific error codes to ignore if needed -exclude = [".venv", "build", "dist"] +lint.select = ["E", "F", "W"] +lint.ignore = [] # Add specific error codes to ignore if needed +exclude = [".venv", "build", "dist", ".eggs", "*.egg-info", ".ruff_cache", "__pycache__"] From 676a3970a62e8c1e7841ad05b6534c15e4747815 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 06:37:23 +0000 Subject: [PATCH 06/16] refactor: maniac breaking changes, introduce lazy initialization. --- pybacklog/__init__.py | 9 +++++++-- tests/__init__.py | 17 ++++++++++++----- 2 files changed, 19 insertions(+), 7 deletions(-) diff --git a/pybacklog/__init__.py b/pybacklog/__init__.py index d4d5b8f..d2fb986 100644 --- a/pybacklog/__init__.py +++ b/pybacklog/__init__.py @@ -10,7 +10,12 @@ def __init__(self, space_name, api_key): self.api_key = api_key ## auto detetcion of space location - self.endpoint = BacklogClient._detect_endpoint(space_name, api_key) + self._endpoint = "" + + def endpoint(self): + if not self._endpoint: + self._endpoint = BacklogClient._detect_endpoint(self.space_name, self.api_key) + return self._endpoint @staticmethod def _detect_endpoint(space_name, api_key): @@ -52,7 +57,7 @@ def do(self, method, url, url_params={}, query_params={}, request_params={}): - Request Body(data): request_params """ _url = url.format(**url_params).lstrip("/") - _endpoint = self.endpoint.format(path=_url) + _endpoint = self.endpoint().format(path=_url) _headers = {"Content-Type": "application/x-www-form-urlencoded"} request_params = BacklogClient.remove_mb4(request_params) diff --git a/tests/__init__.py b/tests/__init__.py index 95e73b7..7c939da 100644 --- a/tests/__init__.py +++ b/tests/__init__.py @@ -2,15 +2,22 @@ from pybacklog import BacklogClient import unittest +from unittest.mock import patch class TestBacklogClient(unittest.TestCase): def test_init(self): - try: - _ = BacklogClient("my_space_name", "my_api_key") - self.fail() - except Exception as _ex: - self.assertEqual(str(_ex), "retrive space information failed. maybe space not found in .com nor .jp") + # pass + BacklogClient("my_space_name", "my_api_key") + + # raise exception + with patch("pybacklog.requests.get") as mock_get: + mock_get.side_effect = Exception("retrive space information failed. maybe space not found in .com nor .jp") + try: + BacklogClient("my_space_name", "my_api_key").endpoint() + self.fail() + except Exception as _ex: + self.assertEqual(str(_ex), "retrive space information failed. maybe space not found in .com nor .jp") def test_remove_mb4(self): testing = ( From 510d239ccb6b305957d12ab11fc64a4a84bd5bd8 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 07:09:34 +0000 Subject: [PATCH 07/16] refactor: add pyright to be "typed python" --- pyproject.toml | 1 + uv.lock | 24 ++++++++++++++++++++++++ 2 files changed, 25 insertions(+) diff --git a/pyproject.toml b/pyproject.toml index 0edc03c..48c423a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -28,6 +28,7 @@ dependencies = [ [dependency-groups] dev = [ "ipython>=9.7.0", + "pyright>=1.1.407", "ruff>=0.14.6", ] diff --git a/uv.lock b/uv.lock index 9c6a2cd..7fcc183 100644 --- a/uv.lock +++ b/uv.lock @@ -187,6 +187,15 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/af/33/ee4519fa02ed11a94aef9559552f3b17bb863f2ecfe1a35dc7f548cde231/matplotlib_inline-0.2.1-py3-none-any.whl", hash = "sha256:d56ce5156ba6085e00a9d54fead6ed29a9c47e215cd1bba2e976ef39f5710a76", size = 9516, upload-time = "2025-10-23T09:00:20.675Z" }, ] +[[package]] +name = "nodeenv" +version = "1.9.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/43/16/fc88b08840de0e0a72a2f9d8c6bae36be573e475a6326ae854bcc549fc45/nodeenv-1.9.1.tar.gz", hash = "sha256:6ec12890a2dab7946721edbfbcd91f3319c6ccc9aec47be7c7e6b7011ee6645f", size = 47437, upload-time = "2024-06-04T18:44:11.171Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d2/1d/1b658dbd2b9fa9c4c9f32accbfc0205d532c8c6194dc0f2a4c0428e7128a/nodeenv-1.9.1-py2.py3-none-any.whl", hash = "sha256:ba11c9782d29c27c70ffbdda2d7415098754709be8a7056d79a737cd901155c9", size = 22314, upload-time = "2024-06-04T18:44:08.352Z" }, +] + [[package]] name = "parso" version = "0.8.5" @@ -249,6 +258,7 @@ dependencies = [ [package.dev-dependencies] dev = [ { name = "ipython" }, + { name = "pyright" }, { name = "ruff" }, ] @@ -258,6 +268,7 @@ requires-dist = [{ name = "requests", specifier = ">=2.12.4,<3.0" }] [package.metadata.requires-dev] dev = [ { name = "ipython", specifier = ">=9.7.0" }, + { name = "pyright", specifier = ">=1.1.407" }, { name = "ruff", specifier = ">=0.14.6" }, ] @@ -270,6 +281,19 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217, upload-time = "2025-06-21T13:39:07.939Z" }, ] +[[package]] +name = "pyright" +version = "1.1.407" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "nodeenv" }, + { name = "typing-extensions" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/a6/1b/0aa08ee42948b61745ac5b5b5ccaec4669e8884b53d31c8ec20b2fcd6b6f/pyright-1.1.407.tar.gz", hash = "sha256:099674dba5c10489832d4a4b2d302636152a9a42d317986c38474c76fe562262", size = 4122872, upload-time = "2025-10-24T23:17:15.145Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/dc/93/b69052907d032b00c40cb656d21438ec00b3a471733de137a3f65a49a0a0/pyright-1.1.407-py3-none-any.whl", hash = "sha256:6dd419f54fcc13f03b52285796d65e639786373f433e243f8b94cf93a7444d21", size = 5997008, upload-time = "2025-10-24T23:17:13.159Z" }, +] + [[package]] name = "requests" version = "2.32.5" From 21809e531b07be48c516f201ad240f178496d1d2 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 07:11:15 +0000 Subject: [PATCH 08/16] refactor: remove unsafe initial value --- pybacklog/__init__.py | 76 +++++++++++++++++++++++++++++++++---------- 1 file changed, 59 insertions(+), 17 deletions(-) diff --git a/pybacklog/__init__.py b/pybacklog/__init__.py index d2fb986..8ae26b9 100644 --- a/pybacklog/__init__.py +++ b/pybacklog/__init__.py @@ -2,10 +2,11 @@ import requests import re +from typing import Optional class BacklogClient(object): - def __init__(self, space_name, api_key): + def __init__(self, space_name: str, api_key: str): self.space_name = space_name self.api_key = api_key @@ -18,7 +19,7 @@ def endpoint(self): return self._endpoint @staticmethod - def _detect_endpoint(space_name, api_key): + def _detect_endpoint(space_name: str, api_key: str) -> str: # at first try .com (new default) _endpoint = "https://%s.backlog.com/api/v2/{path}" % space_name resp = requests.get(_endpoint.format(path="space"), params={"apiKey": api_key}) @@ -49,13 +50,27 @@ def _detect_endpoint(space_name, api_key): raise Exception("retrive space information failed. maybe space not found in .com nor .jp") - def do(self, method, url, url_params={}, query_params={}, request_params={}): + def do( + self, + method, + url, + url_params: Optional[dict] = None, + query_params: Optional[dict] = None, + request_params: Optional[dict] = None, + ): """ - Method: method - URL: url.format(**url_params) - Parameter: query_params & apiKey=api_key - Request Body(data): request_params """ + if url_params is None: + url_params = {} + if query_params is None: + query_params = {} + if request_params is None: + request_params = {} + _url = url.format(**url_params).lstrip("/") _endpoint = self.endpoint().format(path=_url) _headers = {"Content-Type": "application/x-www-form-urlencoded"} @@ -123,20 +138,24 @@ def space(self): """ return self.do("GET", "space") - def projects(self, extra_query_params={}): + def projects(self, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.projects() client.projects({"archived": "false",}) """ + if extra_query_params is None: + extra_query_params = {} return self.do("GET", "projects", query_params=extra_query_params) - def project_activities(self, project_id_or_key, extra_query_params={}): + def project_activities(self, project_id_or_key, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.project_activities("YOUR_PROJECT") client.project_activities("YOUR_PROJECT", {"activityTypeId[]": [1, 2],}) """ + if extra_query_params is None: + extra_query_params = {} return self.do( "get", "projects/{project_id_or_key}/activities", @@ -166,7 +185,7 @@ def versions(self, project_id_or_key): url_params={"project_id_or_key": project_id_or_key}, ) - def create_version(self, project_id_or_key, version_name, extra_request_params={}): + def create_version(self, project_id_or_key, version_name, extra_request_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") @@ -174,6 +193,8 @@ def create_version(self, project_id_or_key, version_name, extra_request_params={ "VERSION_NAME", {"description": "version description"}) """ + if extra_request_params is None: + extra_request_params = {} request_params = extra_request_params request_params["name"] = version_name return self.do( @@ -183,7 +204,7 @@ def create_version(self, project_id_or_key, version_name, extra_request_params={ request_params=request_params, ) - def update_version(self, project_id_or_key, version_id, version_name, extra_request_params={}): + def update_version(self, project_id_or_key, version_id, version_name, extra_request_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.update_version("YOUR_PROJECT", @@ -192,7 +213,8 @@ def update_version(self, project_id_or_key, version_id, version_name, extra_requ {"description": "updated description", "archived": "true"}) """ - + if extra_request_params is None: + extra_request_params = {} request_params = extra_request_params request_params["name"] = version_name return self.do( @@ -213,7 +235,7 @@ def delete_version(self, project_id_or_key, version_id): url_params={"project_id_or_key": project_id_or_key, "version_id": version_id}, ) - def issues(self, extra_query_params={}): + def issues(self, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.issues() @@ -221,6 +243,8 @@ def issues(self, extra_query_params={}): project_id = client.get_project_id("YOUR_PROJECT") client.issues({"projectId[]":[project_id], "sort": "dueDate"}) """ + if extra_query_params is None: + extra_query_params = {} return self.do("GET", "issues", query_params=extra_query_params) def issue(self, issue_id_or_key): @@ -234,11 +258,13 @@ def issue(self, issue_id_or_key): url_params={"issue_id_or_key": issue_id_or_key}, ) - def issue_comments(self, issue_id_or_key, extra_query_params={}): + def issue_comments(self, issue_id_or_key, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.issue_comments("YOUR_PROJECT-999") """ + if extra_query_params is None: + extra_query_params = {} return self.do( "GET", "issues/{issue_id_or_key}/comments", @@ -264,7 +290,9 @@ def priorities(self): """ return self.do("GET", "priorities") - def create_issue(self, project_id, summary, issue_type_id, priority_id, extra_request_params={}): + def create_issue( + self, project_id, summary, issue_type_id, priority_id, extra_request_params: Optional[dict] = None + ): """ client = BacklogClient("your_space_name", "your_api_key") project_key = "YOUR_PROJECT" @@ -279,6 +307,8 @@ def create_issue(self, project_id, summary, issue_type_id, priority_id, extra_re priority_id, {"description": u"a is b and c or d."}) """ + if extra_request_params is None: + extra_request_params = {} request_params = extra_request_params request_params["projectId"] = project_id request_params["summary"] = summary @@ -291,11 +321,13 @@ def create_issue(self, project_id, summary, issue_type_id, priority_id, extra_re request_params=request_params, ) - def add_issue_comment(self, issue_id_or_key, content, extra_request_params={}): + def add_issue_comment(self, issue_id_or_key, content, extra_request_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.add_issue_comment("YOUR_PROJECT-999", u"or ... else e.") """ + if extra_request_params is None: + extra_request_params = {} request_params = extra_request_params request_params["content"] = content return self.do( @@ -319,21 +351,25 @@ def user(self, user_id): """ return self.do("GET", "users/{user_id}", url_params={"user_id": user_id}) - def user_activities(self, user_id, extra_query_params={}): + def user_activities(self, user_id, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.user_activities(3) client.user_activities(3, {"count": 2, "order": "asc"}) """ + if extra_query_params is None: + extra_query_params = {} return self.do( "GET", "users/{user_id}/activities", url_params={"user_id": user_id}, query_params=extra_query_params ) - def groups(self, extra_query_params={}): + def groups(self, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.groups() """ + if extra_query_params is None: + extra_query_params = {} return self.do("GET", "groups", query_params=extra_query_params) def group(self, group_id): @@ -343,20 +379,24 @@ def group(self, group_id): """ return self.do("GET", "groups/{group_id}", url_params={"group_id": group_id}) - def user_stars(self, user_id, extra_query_params={}): + def user_stars(self, user_id, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.user_stars(5) client.user_stars(5, {"count": 100, "order": "asc"}) """ + if extra_query_params is None: + extra_query_params = {} return self.do("GET", "users/{user_id}/stars", url_params={"user_id": user_id}, query_params=extra_query_params) - def user_stars_count(self, user_id, extra_query_params={}): + def user_stars_count(self, user_id, extra_query_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.user_stars_count(5) client.user_stars_count(5, {"since": "2017-05-01", "until": "2017-05-31"}) """ + if extra_query_params is None: + extra_query_params = {} return self.do( "GET", "users/{user_id}/stars/count", url_params={"user_id": user_id}, query_params=extra_query_params ) @@ -382,11 +422,13 @@ def wiki(self, wiki_id): """ return self.do("GET", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}) - def update_wiki(self, wiki_id, extra_request_params={}): + def update_wiki(self, wiki_id, extra_request_params: Optional[dict] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.update_wiki(3, {"name": "test", "content": "content test", "mailNotify": "true"}) """ + if extra_request_params is None: + extra_request_params = {} request_params = extra_request_params return self.do("PATCH", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}, request_params=request_params) From b825f5f125f486b94d966af3e6e77dba4fd5c6b2 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 07:12:39 +0000 Subject: [PATCH 09/16] refactor: remove python2 compatibility --- pybacklog/__init__.py | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/pybacklog/__init__.py b/pybacklog/__init__.py index 8ae26b9..526cee6 100644 --- a/pybacklog/__init__.py +++ b/pybacklog/__init__.py @@ -115,12 +115,7 @@ def remove_mb4(request_params): # remove 4 byte characters pattern = re.compile("[^\u0000-\ud7ff\ue000-\uffff]", re.UNICODE) for key in request_params.keys(): - try: - if isinstance(request_params[key], unicode): - request_params[key] = pattern.sub("\ufffd", request_params[key]) - except NameError: - # maybe python3 - request_params[key] = pattern.sub("\ufffd", request_params[key]) + request_params[key] = pattern.sub("\ufffd", request_params[key]) return request_params # ------------------------------- From 65c1442a3984ba801d7127ce0632884791abb576 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 07:28:53 +0000 Subject: [PATCH 10/16] refactor: clear pyright findings. --- examples/project_activities.py | 21 +++++++++++---------- examples/project_activities2.py | 27 ++++++++++++++------------- pybacklog/__init__.py | 18 +++++++++--------- 3 files changed, 34 insertions(+), 32 deletions(-) diff --git a/examples/project_activities.py b/examples/project_activities.py index 417236c..7e38578 100644 --- a/examples/project_activities.py +++ b/examples/project_activities.py @@ -3,9 +3,9 @@ from pybacklog import BacklogClient import os -_space = os.getenv("BACKLOG_SPACE") -_api_key = os.getenv("BACKLOG_API_KEY") -_project = os.getenv("BACKLOG_PROJECT") +_space = os.getenv("BACKLOG_SPACE", "") +_api_key = os.getenv("BACKLOG_API_KEY", "") +_project = os.getenv("BACKLOG_PROJECT", "") client = BacklogClient(_space, _api_key) @@ -18,14 +18,15 @@ urls = [] items = [] -for activity in activities: - url = client.activity_to_issue_url(activity) - if url in urls: - continue - urls.append(url) +if activities: + for activity in activities: + url = client.activity_to_issue_url(activity) + if url in urls: + continue + urls.append(url) - item = (activity.get("created"), url, activity.get("content").get("summary")) - items.append(item) + item = (activity.get("created"), url, activity.get("content").get("summary")) + items.append(item) for item in items: print("{date}\t{url}\t{summary}".format(date=item[0], url=item[1], summary=item[2])) diff --git a/examples/project_activities2.py b/examples/project_activities2.py index e665308..4a864c2 100644 --- a/examples/project_activities2.py +++ b/examples/project_activities2.py @@ -3,9 +3,9 @@ from pybacklog import BacklogClient import os -_space = os.getenv("BACKLOG_SPACE") -_api_key = os.getenv("BACKLOG_API_KEY") -_project = os.getenv("BACKLOG_PROJECT") +_space = os.getenv("BACKLOG_SPACE", "") +_api_key = os.getenv("BACKLOG_API_KEY", "") +_project = os.getenv("BACKLOG_PROJECT", "") client = BacklogClient(_space, _api_key) @@ -13,17 +13,18 @@ urls = [] items = [] -for activity in activities: - url = client.activity_to_issue_url(activity) - if url in urls: - continue - if "None" in url: - print(activity) - continue - urls.append(url) +if activities: + for activity in activities: + url = client.activity_to_issue_url(activity) + if url in urls: + continue + if "None" in url: + print(activity) + continue + urls.append(url) - item = (activity.get("created"), url, activity.get("content").get("summary")) - items.append(item) + item = (activity.get("created"), url, activity.get("content").get("summary")) + items.append(item) for item in items: print("{date}\t{url}\t{summary}".format(date=item[0], url=item[1], summary=item[2])) diff --git a/pybacklog/__init__.py b/pybacklog/__init__.py index 526cee6..060f4bd 100644 --- a/pybacklog/__init__.py +++ b/pybacklog/__init__.py @@ -456,18 +456,18 @@ def project_statuses(self, project_id_or_key): # extra utilities (PR welcome) # ------------------------------- - def get_project_id(self, project_key_or_name): - projects = self.projects() + def get_project_id(self, project_key_or_name: str) -> Optional[int | str]: + projects = self.projects() or [] # Ensure projects is an iterable for p in projects: - if p["projectKey"] == project_key_or_name: - return p["id"] + if p.get("projectKey") == project_key_or_name: + return p.get("id") for p in projects: - if p["name"] == project_key_or_name: - return p["id"] + if p.get("name") == project_key_or_name: + return p.get("id") return None - def get_issue_id(self, issue_key): + def get_issue_id(self, issue_key: str) -> Optional[int | str]: issue = self.issue(issue_key) - if issue: - return int(issue["id"]) + if issue is not None and "id" in issue: + return issue.get("id") return None From d596ea61e4e3a5edbe9727d1d238d245cff211ce Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Fri, 28 Nov 2025 08:00:15 +0000 Subject: [PATCH 11/16] refactor: apply strict type annotations --- examples/project_activities.py | 18 ++- examples/project_activities2.py | 18 ++- pybacklog/__init__.py | 194 +++++++++++++++++++------------- pyproject.toml | 8 +- 4 files changed, 150 insertions(+), 88 deletions(-) diff --git a/examples/project_activities.py b/examples/project_activities.py index 7e38578..743d8ae 100644 --- a/examples/project_activities.py +++ b/examples/project_activities.py @@ -1,5 +1,7 @@ # -*- coding: utf-8 -*- +from typing import List, Tuple + from pybacklog import BacklogClient import os @@ -16,16 +18,26 @@ query_params={"activityTypeId[]": [1, 2, 3, 14], "count": 100}, ) -urls = [] -items = [] +urls: List[str] = [] +items: List[Tuple[str, str, str]] = [] if activities: for activity in activities: + if not isinstance(activity, dict): + continue + + created = activity.get("created", "") + url = client.activity_to_issue_url(activity) if url in urls: continue urls.append(url) - item = (activity.get("created"), url, activity.get("content").get("summary")) + try: + summary = activity["content"]["summary"] + except (KeyError, TypeError): + summary = "" + + item = (created, url, summary) items.append(item) for item in items: diff --git a/examples/project_activities2.py b/examples/project_activities2.py index 4a864c2..3acdc3f 100644 --- a/examples/project_activities2.py +++ b/examples/project_activities2.py @@ -1,5 +1,7 @@ # -*- coding: utf-8 -*- +from typing import List, Tuple + from pybacklog import BacklogClient import os @@ -11,19 +13,23 @@ client = BacklogClient(_space, _api_key) activities = client.project_activities(_project, {"activityTypeId[]": [1, 2, 3, 14], "count": 100}) -urls = [] -items = [] +urls: List[str] = [] +items: List[Tuple[str, str, str]] = [] if activities: for activity in activities: url = client.activity_to_issue_url(activity) if url in urls: continue - if "None" in url: - print(activity) - continue urls.append(url) - item = (activity.get("created"), url, activity.get("content").get("summary")) + created = activity.get("created", "") + + try: + summary = activity["content"]["summary"] + except (KeyError, TypeError): + summary = "" + + item = (created, url, summary) items.append(item) for item in items: diff --git a/pybacklog/__init__.py b/pybacklog/__init__.py index 060f4bd..4a2dac3 100644 --- a/pybacklog/__init__.py +++ b/pybacklog/__init__.py @@ -2,7 +2,13 @@ import requests import re -from typing import Optional +from typing import Any, Dict, List, Optional, Union, TypedDict + + +class Project(TypedDict, total=False): + projectKey: str + name: str + id: int class BacklogClient(object): @@ -52,12 +58,12 @@ def _detect_endpoint(space_name: str, api_key: str) -> str: def do( self, - method, - url, - url_params: Optional[dict] = None, - query_params: Optional[dict] = None, - request_params: Optional[dict] = None, - ): + method: str, + url: str, + url_params: Optional[Dict[str, Any]] = None, + query_params: Optional[Dict[str, Any]] = None, + request_params: Optional[Dict[str, Any]] = None, + ) -> Union[Dict[str, Any], List[Dict[str, Any]], None]: """ - Method: method - URL: url.format(**url_params) @@ -102,16 +108,19 @@ def do( return resp.json() - def activity_to_issue_url(self, activity): - url = "https://{space}.backlog.jp/view/{project_key}-{content_id}".format( - space=self.space_name, - project_key=activity.get("project").get("projectKey"), - content_id=activity.get("content").get("key_id"), - ) + def activity_to_issue_url(self, activity: Dict[str, Any]) -> str: + try: + url = "https://{space}.backlog.jp/view/{project_key}-{content_id}".format( + space=self.space_name, + project_key=activity["project"]["projectKey"], + content_id=activity["content"]["key_id"], + ) + except KeyError as e: + raise KeyError(f"Missing required key in activity data: {e}") return url @staticmethod - def remove_mb4(request_params): + def remove_mb4(request_params: Dict[str, str]) -> Dict[str, str]: # remove 4 byte characters pattern = re.compile("[^\u0000-\ud7ff\ue000-\uffff]", re.UNICODE) for key in request_params.keys(): @@ -126,14 +135,14 @@ def remove_mb4(request_params): # - optional values => extra_query_params, extra_request_params # - url_params may always required - def space(self): + def space(self) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.space() """ - return self.do("GET", "space") + return self.do("GET", "space") # pyright: ignore[reportReturnType] - def projects(self, extra_query_params: Optional[dict] = None): + def projects(self, extra_query_params: Optional[Dict[str, Any]] = None) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.projects() @@ -141,9 +150,11 @@ def projects(self, extra_query_params: Optional[dict] = None): """ if extra_query_params is None: extra_query_params = {} - return self.do("GET", "projects", query_params=extra_query_params) + return self.do("GET", "projects", query_params=extra_query_params) # pyright: ignore[reportReturnType] - def project_activities(self, project_id_or_key, extra_query_params: Optional[dict] = None): + def project_activities( + self, project_id_or_key: Union[str, int], extra_query_params: Optional[Dict[str, Any]] = None + ) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.project_activities("YOUR_PROJECT") @@ -156,9 +167,9 @@ def project_activities(self, project_id_or_key, extra_query_params: Optional[dic "projects/{project_id_or_key}/activities", url_params={"project_id_or_key": project_id_or_key}, query_params=extra_query_params, - ) + ) # pyright: ignore[reportReturnType] - def project_users(self, project_id_or_key): + def project_users(self, project_id_or_key: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.project_users("YOUR_PROJECT") @@ -167,9 +178,9 @@ def project_users(self, project_id_or_key): "GET", "projects/{project_id_or_key}/users", url_params={"project_id_or_key": project_id_or_key}, - ) + ) # pyright: ignore[reportReturnType] - def versions(self, project_id_or_key): + def versions(self, project_id_or_key: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.versions(3) @@ -178,9 +189,14 @@ def versions(self, project_id_or_key): "GET", "projects/{project_id_or_key}/versions", url_params={"project_id_or_key": project_id_or_key}, - ) + ) # pyright: ignore[reportReturnType] - def create_version(self, project_id_or_key, version_name, extra_request_params: Optional[dict] = None): + def create_version( + self, + project_id_or_key: Union[str, int], + version_name: str, + extra_request_params: Optional[Dict[str, Any]] = None, + ) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") @@ -197,9 +213,15 @@ def create_version(self, project_id_or_key, version_name, extra_request_params: "projects/{project_id_or_key}/versions", url_params={"project_id_or_key": project_id_or_key}, request_params=request_params, - ) + ) # pyright: ignore[reportReturnType] - def update_version(self, project_id_or_key, version_id, version_name, extra_request_params: Optional[dict] = None): + def update_version( + self, + project_id_or_key: Union[str, int], + version_id: str, + version_name: str, + extra_request_params: Optional[Dict[str, Any]] = None, + ) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.update_version("YOUR_PROJECT", @@ -217,9 +239,9 @@ def update_version(self, project_id_or_key, version_id, version_name, extra_requ "projects/{project_id_or_key}/versions/{version_id}", url_params={"project_id_or_key": project_id_or_key, "version_id": version_id}, request_params=request_params, - ) + ) # pyright: ignore[reportReturnType] - def delete_version(self, project_id_or_key, version_id): + def delete_version(self, project_id_or_key: Union[str, int], version_id: str) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.delete_version("YOUR_PROJECT", 3) @@ -228,9 +250,9 @@ def delete_version(self, project_id_or_key, version_id): "DELETE", "projects/{project_id_or_key}/versions/{version_id}", url_params={"project_id_or_key": project_id_or_key, "version_id": version_id}, - ) + ) # pyright: ignore[reportReturnType] - def issues(self, extra_query_params: Optional[dict] = None): + def issues(self, extra_query_params: Optional[Dict[str, Any]] = None) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.issues() @@ -240,9 +262,9 @@ def issues(self, extra_query_params: Optional[dict] = None): """ if extra_query_params is None: extra_query_params = {} - return self.do("GET", "issues", query_params=extra_query_params) + return self.do("GET", "issues", query_params=extra_query_params) # pyright: ignore[reportReturnType] - def issue(self, issue_id_or_key): + def issue(self, issue_id_or_key: Union[str, int]) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.issue("YOUR_PROJECT-999") @@ -251,9 +273,11 @@ def issue(self, issue_id_or_key): "GET", "issues/{issue_id_or_key}", url_params={"issue_id_or_key": issue_id_or_key}, - ) + ) # pyright: ignore[reportReturnType] - def issue_comments(self, issue_id_or_key, extra_query_params: Optional[dict] = None): + def issue_comments( + self, issue_id_or_key: Union[str, int], extra_query_params: Optional[Dict[str, Any]] = None + ) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.issue_comments("YOUR_PROJECT-999") @@ -265,9 +289,9 @@ def issue_comments(self, issue_id_or_key, extra_query_params: Optional[dict] = N "issues/{issue_id_or_key}/comments", url_params={"issue_id_or_key": issue_id_or_key}, query_params=extra_query_params, - ) + ) # pyright: ignore[reportReturnType] - def project_issue_types(self, project_id_or_key): + def project_issue_types(self, project_id_or_key: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.project_issue_types("YOUR_PROJECT") @@ -276,18 +300,23 @@ def project_issue_types(self, project_id_or_key): "GET", "projects/{project_id_or_key}/issueTypes", url_params={"project_id_or_key": project_id_or_key}, - ) + ) # pyright: ignore[reportReturnType] - def priorities(self): + def priorities(self) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.priorities() """ - return self.do("GET", "priorities") + return self.do("GET", "priorities") # pyright: ignore[reportReturnType] def create_issue( - self, project_id, summary, issue_type_id, priority_id, extra_request_params: Optional[dict] = None - ): + self, + project_id: Union[str, int], + summary: str, + issue_type_id: Union[str, int], + priority_id: Union[str, int], + extra_request_params: Optional[Dict[str, Any]] = None, + ) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") project_key = "YOUR_PROJECT" @@ -314,9 +343,11 @@ def create_issue( "POST", "issues", request_params=request_params, - ) + ) # pyright: ignore[reportReturnType] - def add_issue_comment(self, issue_id_or_key, content, extra_request_params: Optional[dict] = None): + def add_issue_comment( + self, issue_id_or_key: Union[str, int], content: str, extra_request_params: Optional[Dict[str, Any]] = None + ) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.add_issue_comment("YOUR_PROJECT-999", u"or ... else e.") @@ -330,23 +361,25 @@ def add_issue_comment(self, issue_id_or_key, content, extra_request_params: Opti "issues/{issue_id_or_key}/comments", url_params={"issue_id_or_key": issue_id_or_key}, request_params=request_params, - ) + ) # pyright: ignore[reportReturnType] - def users(self): + def users(self) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.users() """ - return self.do("GET", "users") + return self.do("GET", "users") # pyright: ignore[reportReturnType] - def user(self, user_id): + def user(self, user_id: Union[str, int]) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.user(3) """ - return self.do("GET", "users/{user_id}", url_params={"user_id": user_id}) + return self.do("GET", "users/{user_id}", url_params={"user_id": user_id}) # pyright: ignore[reportReturnType] - def user_activities(self, user_id, extra_query_params: Optional[dict] = None): + def user_activities( + self, user_id: Union[str, int], extra_query_params: Optional[Dict[str, Any]] = None + ) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.user_activities(3) @@ -356,9 +389,9 @@ def user_activities(self, user_id, extra_query_params: Optional[dict] = None): extra_query_params = {} return self.do( "GET", "users/{user_id}/activities", url_params={"user_id": user_id}, query_params=extra_query_params - ) + ) # pyright: ignore[reportReturnType] - def groups(self, extra_query_params: Optional[dict] = None): + def groups(self, extra_query_params: Optional[Dict[str, Any]] = None): """ client = BacklogClient("your_space_name", "your_api_key") client.groups() @@ -367,14 +400,16 @@ def groups(self, extra_query_params: Optional[dict] = None): extra_query_params = {} return self.do("GET", "groups", query_params=extra_query_params) - def group(self, group_id): + def group(self, group_id: Union[str, int]): """ client = BacklogClient("your_space_name", "your_api_key") client.group(3) """ return self.do("GET", "groups/{group_id}", url_params={"group_id": group_id}) - def user_stars(self, user_id, extra_query_params: Optional[dict] = None): + def user_stars( + self, user_id: Union[str, int], extra_query_params: Optional[Dict[str, Any]] = None + ) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.user_stars(5) @@ -382,9 +417,11 @@ def user_stars(self, user_id, extra_query_params: Optional[dict] = None): """ if extra_query_params is None: extra_query_params = {} - return self.do("GET", "users/{user_id}/stars", url_params={"user_id": user_id}, query_params=extra_query_params) + return self.do("GET", "users/{user_id}/stars", url_params={"user_id": user_id}, query_params=extra_query_params) # pyright: ignore[reportReturnType] - def user_stars_count(self, user_id, extra_query_params: Optional[dict] = None): + def user_stars_count( + self, user_id: Union[str, int], extra_query_params: Optional[Dict[str, Any]] = None + ) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.user_stars_count(5) @@ -394,30 +431,32 @@ def user_stars_count(self, user_id, extra_query_params: Optional[dict] = None): extra_query_params = {} return self.do( "GET", "users/{user_id}/stars/count", url_params={"user_id": user_id}, query_params=extra_query_params - ) + ) # pyright: ignore[reportReturnType] - def star(self, query_params): + def star(self, query_params: Dict[str, Any]) -> None: """ client = BacklogClient("your_space_name", "your_api_key") client.star({"issueId": 333}) """ - return self.do("POST", "stars", query_params=query_params) + return self.do("POST", "stars", query_params=query_params) # pyright: ignore[reportReturnType] - def wikis(self, project_id_or_key): + def wikis(self, project_id_or_key: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.wikis(3) """ - return self.do("GET", "wikis", query_params={"projectIdOrKey": project_id_or_key}) + return self.do("GET", "wikis", query_params={"projectIdOrKey": project_id_or_key}) # pyright: ignore[reportReturnType] - def wiki(self, wiki_id): + def wiki(self, wiki_id: Union[str, int]) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.wiki(3) """ - return self.do("GET", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}) + return self.do("GET", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}) # pyright: ignore[reportReturnType] - def update_wiki(self, wiki_id, extra_request_params: Optional[dict] = None): + def update_wiki( + self, wiki_id: Union[str, int], extra_request_params: Optional[Dict[str, Any]] = None + ) -> Dict[str, Any]: """ client = BacklogClient("your_space_name", "your_api_key") client.update_wiki(3, {"name": "test", "content": "content test", "mailNotify": "true"}) @@ -425,23 +464,23 @@ def update_wiki(self, wiki_id, extra_request_params: Optional[dict] = None): if extra_request_params is None: extra_request_params = {} request_params = extra_request_params - return self.do("PATCH", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}, request_params=request_params) + return self.do("PATCH", "wikis/{wiki_id}", url_params={"wiki_id": wiki_id}, request_params=request_params) # pyright: ignore[reportReturnType] - def wiki_history(self, wiki_id): + def wiki_history(self, wiki_id: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.wiki_history(3) """ - return self.do("GET", "wikis/{wiki_id}/history", url_params={"wiki_id": wiki_id}) + return self.do("GET", "wikis/{wiki_id}/history", url_params={"wiki_id": wiki_id}) # pyright: ignore[reportReturnType] - def wiki_stars(self, wiki_id): + def wiki_stars(self, wiki_id: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.wiki_stars(3) """ - return self.do("GET", "wikis/{wiki_id}/stars", url_params={"wiki_id": wiki_id}) + return self.do("GET", "wikis/{wiki_id}/stars", url_params={"wiki_id": wiki_id}) # pyright: ignore[reportReturnType] - def project_statuses(self, project_id_or_key): + def project_statuses(self, project_id_or_key: Union[str, int]) -> List[Dict[str, Any]]: """ client = BacklogClient("your_space_name", "your_api_key") client.project_statuses("YOUR_PROJECT") @@ -450,24 +489,25 @@ def project_statuses(self, project_id_or_key): "GET", "projects/{project_id_or_key}/statuses", url_params={"project_id_or_key": project_id_or_key}, - ) + ) # pyright: ignore[reportReturnType] # ------------------------------- # extra utilities (PR welcome) # ------------------------------- def get_project_id(self, project_key_or_name: str) -> Optional[int | str]: - projects = self.projects() or [] # Ensure projects is an iterable + raw_projects = self.projects() or [] + projects: List[Dict[str, Any]] = raw_projects for p in projects: - if p.get("projectKey") == project_key_or_name: - return p.get("id") + if "projectKey" in p and p["projectKey"] == project_key_or_name: + return p["id"] for p in projects: - if p.get("name") == project_key_or_name: - return p.get("id") + if "name" in p and p["name"] == project_key_or_name: + return p["id"] return None def get_issue_id(self, issue_key: str) -> Optional[int | str]: issue = self.issue(issue_key) - if issue is not None and "id" in issue: + if "id" in issue: return issue.get("id") return None diff --git a/pyproject.toml b/pyproject.toml index 48c423a..98eab03 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -38,6 +38,10 @@ dev = [ [tool.ruff] target-version = "py314" line-length = 120 -lint.select = ["E", "F", "W"] -lint.ignore = [] # Add specific error codes to ignore if needed +# lint.select = ["E", "F", "W"] +# lint.ignore = [] exclude = [".venv", "build", "dist", ".eggs", "*.egg-info", ".ruff_cache", "__pycache__"] + +[tool.pyright] +pythonVersion = "3.11" +typeCheckingMode = "strict" \ No newline at end of file From 40617d3982e756fe3ae4fbd1cc3506de9ecea54a Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Sat, 29 Nov 2025 03:50:40 +0000 Subject: [PATCH 12/16] fix: do not forget to run pyright in ci.yml --- .github/workflows/ci.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 827b03c..ca96b6d 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -27,8 +27,8 @@ jobs: - name: Sync dependencies run: uv sync --group dev - - name: Run ruff - run: uv run ruff format . && uv run ruff check --fix . + - name: Run ruff and pyright + run: uv run ruff format . && uv run ruff check --fix . && uv run pyright - name: Run tests run: uv run python3 -m unittest tests \ No newline at end of file From 3673ea04e53327ffb8630658337526a9291b1de2 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Sat, 29 Nov 2025 05:27:26 +0000 Subject: [PATCH 13/16] refactor: follow python EOL schedule. --- .github/workflows/ci.yml | 2 +- README.md | 67 +++++++++++++++++++-------------- examples/project_activities.py | 4 +- examples/project_activities2.py | 5 ++- examples/readme.py | 48 +++++++++++++++++++++++ examples/readme2.py | 16 ++++++++ pyproject.toml | 7 ++-- 7 files changed, 114 insertions(+), 35 deletions(-) create mode 100644 examples/readme.py create mode 100644 examples/readme2.py diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index ca96b6d..7799309 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -10,7 +10,7 @@ jobs: strategy: matrix: - python-version: [3.11, 3.12, 3.13, 3.14] + python-version: [3.10, 3.11, 3.12, 3.13, 3.14] steps: - name: Check out code diff --git a/README.md b/README.md index 9bdd689..019d2e8 100644 --- a/README.md +++ b/README.md @@ -10,57 +10,63 @@ Backlog API v2 Client Library for Python - Python 3.11+ - requests 2.x -# Install +# Usage: install ``` pip install pybacklog ``` -# Usage +# Usage: code ```python from pybacklog import BacklogClient +import os -client = BacklogClient("your_space_name", "your_api_key") +YOUR_SPACE_NAME = os.getenv("BACKLOG_SPACE_NAME", "") +YOUR_API_KEY = os.getenv("BACKLOG_API_KEY", "") +YOUR_PROJECT = os.getenv("BACKLOG_PROJECT", "") +YOUR_ISSUE_KEY = os.getenv("BACKLOG_ISSUE_KEY", "") + +client = BacklogClient(YOUR_SPACE_NAME, YOUR_API_KEY) # space -space = client.do("GET", "space") # GET /api/v2/space -print(space.get(u"spaceKey")) +space = client.space() +print(space.get("spaceKey")) # project projects = client.projects() # activity -activities = client.project_activities("YOUR_PROJECT", {"activityTypeId[]": [1, 2]}) +activities = client.project_activities(YOUR_PROJECT, {"activityTypeId[]": [1, 2]}) # list issue -project_id = client.get_project_id("YOUR_PROJECT") -issues = client.issues({"projectId[]":[project_id], "sort": "dueDate"}) +project_id = client.get_project_id(YOUR_PROJECT) +issues = client.issues({"projectId[]": [project_id], "sort": "dueDate"}) # specified issue -issue = client.issue("YOUR_PROJECT-999") +issue = client.issue(YOUR_ISSUE_KEY) # create issue -project_id = client.get_project_id(project_key) -issue_type_id = client.project_issue_types(project_key)[0][u"id"] -priority_id = client.priorities()[0][u"id"] +project_id = client.get_project_id(YOUR_PROJECT) +issue_type_id = client.project_issue_types(YOUR_PROJECT)[0]["id"] +priority_id = client.priorities()[0]["id"] -client.create_issue(project_id, - u"some summary", - issue_type_id, - priority_id, - {"description": u"a is b and c or d."}) +if project_id and issue_type_id and priority_id: + client.create_issue(project_id, "some summary", issue_type_id, priority_id, {"description": "a is b and c or d."}) # add comment -client.add_issue_comment("YOUR_PROJECT-999", u"or ... else e.") +client.add_issue_comment(YOUR_ISSUE_KEY, "or ... else e.") # top 10 star collector -star_collectors = [(client.user_stars_count(u[u"id"], {"since": "2017-06-01", "until": "2017-06-30"})[u"count"], u[u"name"]) for u in client.users()] +star_collectors = [ + (client.user_stars_count(u["id"], {"since": "2017-06-01", "until": "2017-06-30"})["count"], u["name"]) + for u in client.users() +] star_collectors.sort() star_collectors.reverse() for i, (c, u) in enumerate(star_collectors[:10]): - print(i+1, c, u) + print(i + 1, c, u) ``` supported operations are `pydoc pybacklog.BacklogClient` @@ -74,16 +80,21 @@ Use `do` or let's write code and Pull Request. ```python from pybacklog import BacklogClient +import os + +YOUR_SPACE_NAME = os.getenv("BACKLOG_SPACE_NAME", "") +YOUR_API_KEY = os.getenv("BACKLOG_API_KEY", "") +YOUR_PROJECT = os.getenv("BACKLOG_PROJECT", "") -client = BacklogClient("your_space_name", "your_api_key") +client = BacklogClient(YOUR_SPACE_NAME, YOUR_API_KEY) space = client.do("GET", "space") # GET /api/v2/space -projects = client.do("GET", "projects", - query_params={"archived": false} - ) # GET /api/v2/projects?archived=false -activities = client.do("GET", "projects/{project_id_or_key}/activities", - url_params={"project_id_or_key": "myproj"}, - query_params={"activityTypeId[]": [1, 2]} - ) # GET /api/v2/projects/myproj/activities?activityTypeIds%5B%5D=1&activityTypeIds%5B%5D=2 +projects = client.do("GET", "projects", query_params={"archived": False}) # GET /api/v2/projects?archived=false +activities = client.do( + "GET", + "projects/{project_id_or_key}/activities", + url_params={"project_id_or_key": YOUR_PROJECT}, + query_params={"activityTypeId[]": [1, 2]}, +) # GET /api/v2/projects/myproj/activities?activityTypeIds%5B%5D=1&activityTypeIds%5B%5D=2 ``` see also [Backlog API Overview \| Backlog Developer API \| Nulab](https://developer.nulab-inc.com/docs/backlog/) diff --git a/examples/project_activities.py b/examples/project_activities.py index 743d8ae..2b6fced 100644 --- a/examples/project_activities.py +++ b/examples/project_activities.py @@ -34,7 +34,9 @@ try: summary = activity["content"]["summary"] - except (KeyError, TypeError): + except KeyError: + summary = "" + except TypeError: summary = "" item = (created, url, summary) diff --git a/examples/project_activities2.py b/examples/project_activities2.py index 3acdc3f..8688336 100644 --- a/examples/project_activities2.py +++ b/examples/project_activities2.py @@ -26,9 +26,10 @@ try: summary = activity["content"]["summary"] - except (KeyError, TypeError): + except KeyError: + summary = "" + except TypeError: summary = "" - item = (created, url, summary) items.append(item) diff --git a/examples/readme.py b/examples/readme.py new file mode 100644 index 0000000..1209c28 --- /dev/null +++ b/examples/readme.py @@ -0,0 +1,48 @@ +from pybacklog import BacklogClient +import os + +YOUR_SPACE_NAME = os.getenv("BACKLOG_SPACE_NAME", "") +YOUR_API_KEY = os.getenv("BACKLOG_API_KEY", "") +YOUR_PROJECT = os.getenv("BACKLOG_PROJECT", "") +YOUR_ISSUE_KEY = os.getenv("BACKLOG_ISSUE_KEY", "") + +client = BacklogClient(YOUR_SPACE_NAME, YOUR_API_KEY) + +# space +space = client.space() +print(space.get("spaceKey")) + +# project +projects = client.projects() + +# activity +activities = client.project_activities(YOUR_PROJECT, {"activityTypeId[]": [1, 2]}) + +# list issue +project_id = client.get_project_id(YOUR_PROJECT) +issues = client.issues({"projectId[]": [project_id], "sort": "dueDate"}) + +# specified issue +issue = client.issue(YOUR_ISSUE_KEY) + +# create issue +project_id = client.get_project_id(YOUR_PROJECT) +issue_type_id = client.project_issue_types(YOUR_PROJECT)[0]["id"] +priority_id = client.priorities()[0]["id"] + +if project_id and issue_type_id and priority_id: + client.create_issue(project_id, "some summary", issue_type_id, priority_id, {"description": "a is b and c or d."}) + +# add comment +client.add_issue_comment(YOUR_ISSUE_KEY, "or ... else e.") + +# top 10 star collector +star_collectors = [ + (client.user_stars_count(u["id"], {"since": "2017-06-01", "until": "2017-06-30"})["count"], u["name"]) + for u in client.users() +] +star_collectors.sort() +star_collectors.reverse() + +for i, (c, u) in enumerate(star_collectors[:10]): + print(i + 1, c, u) diff --git a/examples/readme2.py b/examples/readme2.py new file mode 100644 index 0000000..2bcf397 --- /dev/null +++ b/examples/readme2.py @@ -0,0 +1,16 @@ +from pybacklog import BacklogClient +import os + +YOUR_SPACE_NAME = os.getenv("BACKLOG_SPACE_NAME", "") +YOUR_API_KEY = os.getenv("BACKLOG_API_KEY", "") +YOUR_PROJECT = os.getenv("BACKLOG_PROJECT", "") + +client = BacklogClient(YOUR_SPACE_NAME, YOUR_API_KEY) +space = client.do("GET", "space") # GET /api/v2/space +projects = client.do("GET", "projects", query_params={"archived": False}) # GET /api/v2/projects?archived=false +activities = client.do( + "GET", + "projects/{project_id_or_key}/activities", + url_params={"project_id_or_key": YOUR_PROJECT}, + query_params={"activityTypeId[]": [1, 2]}, +) # GET /api/v2/projects/myproj/activities?activityTypeIds%5B%5D=1&activityTypeIds%5B%5D=2 diff --git a/pyproject.toml b/pyproject.toml index 98eab03..a72df52 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -13,13 +13,14 @@ authors = [ ] urls = { "Homepage" = "https://github.com/netmarkjp/pybacklog" } classifiers = [ + "Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.12", "Programming Language :: Python :: 3.13", "Programming Language :: Python :: 3.14", "Topic :: Utilities" ] -requires-python = ">=3.11, <3.15" +requires-python = ">=3.10, <3.15" dependencies = [ "requests >=2.12.4,<3.0" @@ -36,12 +37,12 @@ dev = [ # Additional uv-specific configurations can be added here. [tool.ruff] -target-version = "py314" +target-version = "py310" line-length = 120 # lint.select = ["E", "F", "W"] # lint.ignore = [] exclude = [".venv", "build", "dist", ".eggs", "*.egg-info", ".ruff_cache", "__pycache__"] [tool.pyright] -pythonVersion = "3.11" +pythonVersion = "3.10" typeCheckingMode = "strict" \ No newline at end of file From 11c3b51c0d62613de2465cfd018292a494f5b8c2 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Sat, 29 Nov 2025 05:32:03 +0000 Subject: [PATCH 14/16] fix: 3.10 treated as 3.1 --- .github/workflows/ci.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 7799309..7ed3b39 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -10,7 +10,7 @@ jobs: strategy: matrix: - python-version: [3.10, 3.11, 3.12, 3.13, 3.14] + python-version: ["3.10", "3.11", "3.12", "3.13", "3.14"] steps: - name: Check out code From aeb7196e7bbd3a22fa15da404117abfff40b57a8 Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Sat, 29 Nov 2025 05:38:10 +0000 Subject: [PATCH 15/16] fix: remove unused dependency ipython from dev group --- README.md | 2 +- pyproject.toml | 3 +- uv.lock | 250 ++++++++----------------------------------------- 3 files changed, 41 insertions(+), 214 deletions(-) diff --git a/README.md b/README.md index 019d2e8..10d4caa 100644 --- a/README.md +++ b/README.md @@ -7,7 +7,7 @@ Backlog API v2 Client Library for Python # Requirements -- Python 3.11+ +- Python 3.10+ - requests 2.x # Usage: install diff --git a/pyproject.toml b/pyproject.toml index a72df52..ef4ccdf 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -20,7 +20,7 @@ classifiers = [ "Programming Language :: Python :: 3.14", "Topic :: Utilities" ] -requires-python = ">=3.10, <3.15" +requires-python = ">=3.10" dependencies = [ "requests >=2.12.4,<3.0" @@ -28,7 +28,6 @@ dependencies = [ [dependency-groups] dev = [ - "ipython>=9.7.0", "pyright>=1.1.407", "ruff>=0.14.6", ] diff --git a/uv.lock b/uv.lock index 7fcc183..23a5696 100644 --- a/uv.lock +++ b/uv.lock @@ -1,15 +1,6 @@ version = 1 revision = 3 -requires-python = ">=3.11, <3.15" - -[[package]] -name = "asttokens" -version = "3.0.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/be/a5/8e3f9b6771b0b408517c82d97aed8f2036509bc247d46114925e32fe33f0/asttokens-3.0.1.tar.gz", hash = "sha256:71a4ee5de0bde6a31d64f6b13f2293ac190344478f081c3d1bccfcf5eacb0cb7", size = 62308, upload-time = "2025-11-15T16:43:48.578Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/d2/39/e7eaf1799466a4aef85b6a4fe7bd175ad2b1c6345066aa33f1f58d4b18d0/asttokens-3.0.1-py3-none-any.whl", hash = "sha256:15a3ebc0f43c2d0a50eeafea25e19046c68398e487b9f1f5b517f7c0f40f976a", size = 27047, upload-time = "2025-11-15T16:43:16.109Z" }, -] +requires-python = ">=3.10" [[package]] name = "certifi" @@ -26,6 +17,22 @@ version = "3.4.4" source = { registry = "https://pypi.org/simple" } sdist = { url = "https://files.pythonhosted.org/packages/13/69/33ddede1939fdd074bce5434295f38fae7136463422fe4fd3e0e89b98062/charset_normalizer-3.4.4.tar.gz", hash = "sha256:94537985111c35f28720e43603b8e7b43a6ecfb2ce1d3058bbe955b73404e21a", size = 129418, upload-time = "2025-10-14T04:42:32.879Z" } wheels = [ + { url = "https://files.pythonhosted.org/packages/1f/b8/6d51fc1d52cbd52cd4ccedd5b5b2f0f6a11bbf6765c782298b0f3e808541/charset_normalizer-3.4.4-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:e824f1492727fa856dd6eda4f7cee25f8518a12f3c4a56a74e8095695089cf6d", size = 209709, upload-time = "2025-10-14T04:40:11.385Z" }, + { url = "https://files.pythonhosted.org/packages/5c/af/1f9d7f7faafe2ddfb6f72a2e07a548a629c61ad510fe60f9630309908fef/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4bd5d4137d500351a30687c2d3971758aac9a19208fc110ccb9d7188fbe709e8", size = 148814, upload-time = "2025-10-14T04:40:13.135Z" }, + { url = "https://files.pythonhosted.org/packages/79/3d/f2e3ac2bbc056ca0c204298ea4e3d9db9b4afe437812638759db2c976b5f/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:027f6de494925c0ab2a55eab46ae5129951638a49a34d87f4c3eda90f696b4ad", size = 144467, upload-time = "2025-10-14T04:40:14.728Z" }, + { url = "https://files.pythonhosted.org/packages/ec/85/1bf997003815e60d57de7bd972c57dc6950446a3e4ccac43bc3070721856/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f820802628d2694cb7e56db99213f930856014862f3fd943d290ea8438d07ca8", size = 162280, upload-time = "2025-10-14T04:40:16.14Z" }, + { url = "https://files.pythonhosted.org/packages/3e/8e/6aa1952f56b192f54921c436b87f2aaf7c7a7c3d0d1a765547d64fd83c13/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:798d75d81754988d2565bff1b97ba5a44411867c0cf32b77a7e8f8d84796b10d", size = 159454, upload-time = "2025-10-14T04:40:17.567Z" }, + { url = "https://files.pythonhosted.org/packages/36/3b/60cbd1f8e93aa25d1c669c649b7a655b0b5fb4c571858910ea9332678558/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9d1bb833febdff5c8927f922386db610b49db6e0d4f4ee29601d71e7c2694313", size = 153609, upload-time = "2025-10-14T04:40:19.08Z" }, + { url = "https://files.pythonhosted.org/packages/64/91/6a13396948b8fd3c4b4fd5bc74d045f5637d78c9675585e8e9fbe5636554/charset_normalizer-3.4.4-cp310-cp310-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:9cd98cdc06614a2f768d2b7286d66805f94c48cde050acdbbb7db2600ab3197e", size = 151849, upload-time = "2025-10-14T04:40:20.607Z" }, + { url = "https://files.pythonhosted.org/packages/b7/7a/59482e28b9981d105691e968c544cc0df3b7d6133152fb3dcdc8f135da7a/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:077fbb858e903c73f6c9db43374fd213b0b6a778106bc7032446a8e8b5b38b93", size = 151586, upload-time = "2025-10-14T04:40:21.719Z" }, + { url = "https://files.pythonhosted.org/packages/92/59/f64ef6a1c4bdd2baf892b04cd78792ed8684fbc48d4c2afe467d96b4df57/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:244bfb999c71b35de57821b8ea746b24e863398194a4014e4c76adc2bbdfeff0", size = 145290, upload-time = "2025-10-14T04:40:23.069Z" }, + { url = "https://files.pythonhosted.org/packages/6b/63/3bf9f279ddfa641ffa1962b0db6a57a9c294361cc2f5fcac997049a00e9c/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:64b55f9dce520635f018f907ff1b0df1fdc31f2795a922fb49dd14fbcdf48c84", size = 163663, upload-time = "2025-10-14T04:40:24.17Z" }, + { url = "https://files.pythonhosted.org/packages/ed/09/c9e38fc8fa9e0849b172b581fd9803bdf6e694041127933934184e19f8c3/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_riscv64.whl", hash = "sha256:faa3a41b2b66b6e50f84ae4a68c64fcd0c44355741c6374813a800cd6695db9e", size = 151964, upload-time = "2025-10-14T04:40:25.368Z" }, + { url = "https://files.pythonhosted.org/packages/d2/d1/d28b747e512d0da79d8b6a1ac18b7ab2ecfd81b2944c4c710e166d8dd09c/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:6515f3182dbe4ea06ced2d9e8666d97b46ef4c75e326b79bb624110f122551db", size = 161064, upload-time = "2025-10-14T04:40:26.806Z" }, + { url = "https://files.pythonhosted.org/packages/bb/9a/31d62b611d901c3b9e5500c36aab0ff5eb442043fb3a1c254200d3d397d9/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:cc00f04ed596e9dc0da42ed17ac5e596c6ccba999ba6bd92b0e0aef2f170f2d6", size = 155015, upload-time = "2025-10-14T04:40:28.284Z" }, + { url = "https://files.pythonhosted.org/packages/1f/f3/107e008fa2bff0c8b9319584174418e5e5285fef32f79d8ee6a430d0039c/charset_normalizer-3.4.4-cp310-cp310-win32.whl", hash = "sha256:f34be2938726fc13801220747472850852fe6b1ea75869a048d6f896838c896f", size = 99792, upload-time = "2025-10-14T04:40:29.613Z" }, + { url = "https://files.pythonhosted.org/packages/eb/66/e396e8a408843337d7315bab30dbf106c38966f1819f123257f5520f8a96/charset_normalizer-3.4.4-cp310-cp310-win_amd64.whl", hash = "sha256:a61900df84c667873b292c3de315a786dd8dac506704dea57bc957bd31e22c7d", size = 107198, upload-time = "2025-10-14T04:40:30.644Z" }, + { url = "https://files.pythonhosted.org/packages/b5/58/01b4f815bf0312704c267f2ccb6e5d42bcc7752340cd487bc9f8c3710597/charset_normalizer-3.4.4-cp310-cp310-win_arm64.whl", hash = "sha256:cead0978fc57397645f12578bfd2d5ea9138ea0fac82b2f63f7f7c6877986a69", size = 100262, upload-time = "2025-10-14T04:40:32.108Z" }, { url = "https://files.pythonhosted.org/packages/ed/27/c6491ff4954e58a10f69ad90aca8a1b6fe9c5d3c6f380907af3c37435b59/charset_normalizer-3.4.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:6e1fcf0720908f200cd21aa4e6750a48ff6ce4afe7ff5a79a90d5ed8a08296f8", size = 206988, upload-time = "2025-10-14T04:40:33.79Z" }, { url = "https://files.pythonhosted.org/packages/94/59/2e87300fe67ab820b5428580a53cad894272dbb97f38a7a814a2a1ac1011/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5f819d5fe9234f9f82d75bdfa9aef3a3d72c4d24a6e57aeaebba32a704553aa0", size = 147324, upload-time = "2025-10-14T04:40:34.961Z" }, { url = "https://files.pythonhosted.org/packages/07/fb/0cf61dc84b2b088391830f6274cb57c82e4da8bbc2efeac8c025edb88772/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:a59cb51917aa591b1c4e6a43c132f0cdc3c76dbad6155df4e28ee626cc77a0a3", size = 142742, upload-time = "2025-10-14T04:40:36.105Z" }, @@ -93,33 +100,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/0a/4c/925909008ed5a988ccbb72dcc897407e5d6d3bd72410d69e051fc0c14647/charset_normalizer-3.4.4-py3-none-any.whl", hash = "sha256:7a32c560861a02ff789ad905a2fe94e3f840803362c84fecf1851cb4cf3dc37f", size = 53402, upload-time = "2025-10-14T04:42:31.76Z" }, ] -[[package]] -name = "colorama" -version = "0.4.6" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" }, -] - -[[package]] -name = "decorator" -version = "5.2.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/43/fa/6d96a0978d19e17b68d634497769987b16c8f4cd0a7a05048bec693caa6b/decorator-5.2.1.tar.gz", hash = "sha256:65f266143752f734b0a7cc83c46f4618af75b8c5911b00ccb61d0ac9b6da0360", size = 56711, upload-time = "2025-02-24T04:41:34.073Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/4e/8c/f3147f5c4b73e7550fe5f9352eaa956ae838d5c51eb58e7a25b9f3e2643b/decorator-5.2.1-py3-none-any.whl", hash = "sha256:d316bb415a2d9e2d2b3abcc4084c6502fc09240e292cd76a76afc106a1c8e04a", size = 9190, upload-time = "2025-02-24T04:41:32.565Z" }, -] - -[[package]] -name = "executing" -version = "2.2.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/cc/28/c14e053b6762b1044f34a13aab6859bbf40456d37d23aa286ac24cfd9a5d/executing-2.2.1.tar.gz", hash = "sha256:3632cc370565f6648cc328b32435bd120a1e4ebb20c77e3fdde9a13cd1e533c4", size = 1129488, upload-time = "2025-09-01T09:48:10.866Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c1/ea/53f2148663b321f21b5a606bd5f191517cf40b7072c0497d3c92c4a13b1e/executing-2.2.1-py2.py3-none-any.whl", hash = "sha256:760643d3452b4d777d295bb167ccc74c64a81df23fb5e08eff250c425a4b2017", size = 28317, upload-time = "2025-09-01T09:48:08.5Z" }, -] - [[package]] name = "idna" version = "3.11" @@ -129,64 +109,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/0e/61/66938bbb5fc52dbdf84594873d5b51fb1f7c7794e9c0f5bd885f30bc507b/idna-3.11-py3-none-any.whl", hash = "sha256:771a87f49d9defaf64091e6e6fe9c18d4833f140bd19464795bc32d966ca37ea", size = 71008, upload-time = "2025-10-12T14:55:18.883Z" }, ] -[[package]] -name = "ipython" -version = "9.7.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "colorama", marker = "sys_platform == 'win32'" }, - { name = "decorator" }, - { name = "ipython-pygments-lexers" }, - { name = "jedi" }, - { name = "matplotlib-inline" }, - { name = "pexpect", marker = "sys_platform != 'emscripten' and sys_platform != 'win32'" }, - { name = "prompt-toolkit" }, - { name = "pygments" }, - { name = "stack-data" }, - { name = "traitlets" }, - { name = "typing-extensions", marker = "python_full_version < '3.12'" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/29/e6/48c74d54039241a456add616464ea28c6ebf782e4110d419411b83dae06f/ipython-9.7.0.tar.gz", hash = "sha256:5f6de88c905a566c6a9d6c400a8fed54a638e1f7543d17aae2551133216b1e4e", size = 4422115, upload-time = "2025-11-05T12:18:54.646Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/05/aa/62893d6a591d337aa59dcc4c6f6c842f1fe20cd72c8c5c1f980255243252/ipython-9.7.0-py3-none-any.whl", hash = "sha256:bce8ac85eb9521adc94e1845b4c03d88365fd6ac2f4908ec4ed1eb1b0a065f9f", size = 618911, upload-time = "2025-11-05T12:18:52.484Z" }, -] - -[[package]] -name = "ipython-pygments-lexers" -version = "1.1.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "pygments" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/ef/4c/5dd1d8af08107f88c7f741ead7a40854b8ac24ddf9ae850afbcf698aa552/ipython_pygments_lexers-1.1.1.tar.gz", hash = "sha256:09c0138009e56b6854f9535736f4171d855c8c08a563a0dcd8022f78355c7e81", size = 8393, upload-time = "2025-01-17T11:24:34.505Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/d9/33/1f075bf72b0b747cb3288d011319aaf64083cf2efef8354174e3ed4540e2/ipython_pygments_lexers-1.1.1-py3-none-any.whl", hash = "sha256:a9462224a505ade19a605f71f8fa63c2048833ce50abc86768a0d81d876dc81c", size = 8074, upload-time = "2025-01-17T11:24:33.271Z" }, -] - -[[package]] -name = "jedi" -version = "0.19.2" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "parso" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/72/3a/79a912fbd4d8dd6fbb02bf69afd3bb72cf0c729bb3063c6f4498603db17a/jedi-0.19.2.tar.gz", hash = "sha256:4770dc3de41bde3966b02eb84fbcf557fb33cce26ad23da12c742fb50ecb11f0", size = 1231287, upload-time = "2024-11-11T01:41:42.873Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c0/5a/9cac0c82afec3d09ccd97c8b6502d48f165f9124db81b4bcb90b4af974ee/jedi-0.19.2-py2.py3-none-any.whl", hash = "sha256:a8ef22bde8490f57fe5c7681a3c83cb58874daf72b4784de3cce5b6ef6edb5b9", size = 1572278, upload-time = "2024-11-11T01:41:40.175Z" }, -] - -[[package]] -name = "matplotlib-inline" -version = "0.2.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "traitlets" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/c7/74/97e72a36efd4ae2bccb3463284300f8953f199b5ffbc04cbbb0ec78f74b1/matplotlib_inline-0.2.1.tar.gz", hash = "sha256:e1ee949c340d771fc39e241ea75683deb94762c8fa5f2927ec57c83c4dffa9fe", size = 8110, upload-time = "2025-10-23T09:00:22.126Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/af/33/ee4519fa02ed11a94aef9559552f3b17bb863f2ecfe1a35dc7f548cde231/matplotlib_inline-0.2.1-py3-none-any.whl", hash = "sha256:d56ce5156ba6085e00a9d54fead6ed29a9c47e215cd1bba2e976ef39f5710a76", size = 9516, upload-time = "2025-10-23T09:00:20.675Z" }, -] - [[package]] name = "nodeenv" version = "1.9.1" @@ -196,57 +118,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/d2/1d/1b658dbd2b9fa9c4c9f32accbfc0205d532c8c6194dc0f2a4c0428e7128a/nodeenv-1.9.1-py2.py3-none-any.whl", hash = "sha256:ba11c9782d29c27c70ffbdda2d7415098754709be8a7056d79a737cd901155c9", size = 22314, upload-time = "2024-06-04T18:44:08.352Z" }, ] -[[package]] -name = "parso" -version = "0.8.5" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/d4/de/53e0bcf53d13e005bd8c92e7855142494f41171b34c2536b86187474184d/parso-0.8.5.tar.gz", hash = "sha256:034d7354a9a018bdce352f48b2a8a450f05e9d6ee85db84764e9b6bd96dafe5a", size = 401205, upload-time = "2025-08-23T15:15:28.028Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/16/32/f8e3c85d1d5250232a5d3477a2a28cc291968ff175caeadaf3cc19ce0e4a/parso-0.8.5-py2.py3-none-any.whl", hash = "sha256:646204b5ee239c396d040b90f9e272e9a8017c630092bf59980beb62fd033887", size = 106668, upload-time = "2025-08-23T15:15:25.663Z" }, -] - -[[package]] -name = "pexpect" -version = "4.9.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "ptyprocess" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/42/92/cc564bf6381ff43ce1f4d06852fc19a2f11d180f23dc32d9588bee2f149d/pexpect-4.9.0.tar.gz", hash = "sha256:ee7d41123f3c9911050ea2c2dac107568dc43b2d3b0c7557a33212c398ead30f", size = 166450, upload-time = "2023-11-25T09:07:26.339Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/9e/c3/059298687310d527a58bb01f3b1965787ee3b40dce76752eda8b44e9a2c5/pexpect-4.9.0-py2.py3-none-any.whl", hash = "sha256:7236d1e080e4936be2dc3e326cec0af72acf9212a7e1d060210e70a47e253523", size = 63772, upload-time = "2023-11-25T06:56:14.81Z" }, -] - -[[package]] -name = "prompt-toolkit" -version = "3.0.52" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "wcwidth" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/a1/96/06e01a7b38dce6fe1db213e061a4602dd6032a8a97ef6c1a862537732421/prompt_toolkit-3.0.52.tar.gz", hash = "sha256:28cde192929c8e7321de85de1ddbe736f1375148b02f2e17edd840042b1be855", size = 434198, upload-time = "2025-08-27T15:24:02.057Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/84/03/0d3ce49e2505ae70cf43bc5bb3033955d2fc9f932163e84dc0779cc47f48/prompt_toolkit-3.0.52-py3-none-any.whl", hash = "sha256:9aac639a3bbd33284347de5ad8d68ecc044b91a762dc39b7c21095fcd6a19955", size = 391431, upload-time = "2025-08-27T15:23:59.498Z" }, -] - -[[package]] -name = "ptyprocess" -version = "0.7.0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/20/e5/16ff212c1e452235a90aeb09066144d0c5a6a8c0834397e03f5224495c4e/ptyprocess-0.7.0.tar.gz", hash = "sha256:5c5d0a3b48ceee0b48485e0c26037c0acd7d29765ca3fbb5cb3831d347423220", size = 70762, upload-time = "2020-12-28T15:15:30.155Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/22/a6/858897256d0deac81a172289110f31629fc4cee19b6f01283303e18c8db3/ptyprocess-0.7.0-py2.py3-none-any.whl", hash = "sha256:4b41f3967fce3af57cc7e94b888626c18bf37a083e3651ca8feeb66d492fef35", size = 13993, upload-time = "2020-12-28T15:15:28.35Z" }, -] - -[[package]] -name = "pure-eval" -version = "0.2.3" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/cd/05/0a34433a064256a578f1783a10da6df098ceaa4a57bbeaa96a6c0352786b/pure_eval-0.2.3.tar.gz", hash = "sha256:5f4e983f40564c576c7c8635ae88db5956bb2229d7e9237d03b3c0b0190eaf42", size = 19752, upload-time = "2024-07-21T12:58:21.801Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/8e/37/efad0257dc6e593a18957422533ff0f87ede7c9c6ea010a2177d738fb82f/pure_eval-0.2.3-py3-none-any.whl", hash = "sha256:1db8e35b67b3d218d818ae653e27f06c3aa420901fa7b081ca98cbedc874e0d0", size = 11842, upload-time = "2024-07-21T12:58:20.04Z" }, -] - [[package]] name = "pybacklog" version = "0.1.8" @@ -257,7 +128,6 @@ dependencies = [ [package.dev-dependencies] dev = [ - { name = "ipython" }, { name = "pyright" }, { name = "ruff" }, ] @@ -267,20 +137,10 @@ requires-dist = [{ name = "requests", specifier = ">=2.12.4,<3.0" }] [package.metadata.requires-dev] dev = [ - { name = "ipython", specifier = ">=9.7.0" }, { name = "pyright", specifier = ">=1.1.407" }, { name = "ruff", specifier = ">=0.14.6" }, ] -[[package]] -name = "pygments" -version = "2.19.2" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/b0/77/a5b8c569bf593b0140bde72ea885a803b82086995367bf2037de0159d924/pygments-2.19.2.tar.gz", hash = "sha256:636cb2477cec7f8952536970bc533bc43743542f70392ae026374600add5b887", size = 4968631, upload-time = "2025-06-21T13:39:12.283Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217, upload-time = "2025-06-21T13:39:07.939Z" }, -] - [[package]] name = "pyright" version = "1.1.407" @@ -311,51 +171,28 @@ wheels = [ [[package]] name = "ruff" -version = "0.14.6" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/52/f0/62b5a1a723fe183650109407fa56abb433b00aa1c0b9ba555f9c4efec2c6/ruff-0.14.6.tar.gz", hash = "sha256:6f0c742ca6a7783a736b867a263b9a7a80a45ce9bee391eeda296895f1b4e1cc", size = 5669501, upload-time = "2025-11-21T14:26:17.903Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/67/d2/7dd544116d107fffb24a0064d41a5d2ed1c9d6372d142f9ba108c8e39207/ruff-0.14.6-py3-none-linux_armv6l.whl", hash = "sha256:d724ac2f1c240dbd01a2ae98db5d1d9a5e1d9e96eba999d1c48e30062df578a3", size = 13326119, upload-time = "2025-11-21T14:25:24.2Z" }, - { url = "https://files.pythonhosted.org/packages/36/6a/ad66d0a3315d6327ed6b01f759d83df3c4d5f86c30462121024361137b6a/ruff-0.14.6-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:9f7539ea257aa4d07b7ce87aed580e485c40143f2473ff2f2b75aee003186004", size = 13526007, upload-time = "2025-11-21T14:25:26.906Z" }, - { url = "https://files.pythonhosted.org/packages/a3/9d/dae6db96df28e0a15dea8e986ee393af70fc97fd57669808728080529c37/ruff-0.14.6-py3-none-macosx_11_0_arm64.whl", hash = "sha256:7f6007e55b90a2a7e93083ba48a9f23c3158c433591c33ee2e99a49b889c6332", size = 12676572, upload-time = "2025-11-21T14:25:29.826Z" }, - { url = "https://files.pythonhosted.org/packages/76/a4/f319e87759949062cfee1b26245048e92e2acce900ad3a909285f9db1859/ruff-0.14.6-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0a8e7b9d73d8728b68f632aa8e824ef041d068d231d8dbc7808532d3629a6bef", size = 13140745, upload-time = "2025-11-21T14:25:32.788Z" }, - { url = "https://files.pythonhosted.org/packages/95/d3/248c1efc71a0a8ed4e8e10b4b2266845d7dfc7a0ab64354afe049eaa1310/ruff-0.14.6-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:d50d45d4553a3ebcbd33e7c5e0fe6ca4aafd9a9122492de357205c2c48f00775", size = 13076486, upload-time = "2025-11-21T14:25:35.601Z" }, - { url = "https://files.pythonhosted.org/packages/a5/19/b68d4563fe50eba4b8c92aa842149bb56dd24d198389c0ed12e7faff4f7d/ruff-0.14.6-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:118548dd121f8a21bfa8ab2c5b80e5b4aed67ead4b7567790962554f38e598ce", size = 13727563, upload-time = "2025-11-21T14:25:38.514Z" }, - { url = "https://files.pythonhosted.org/packages/47/ac/943169436832d4b0e867235abbdb57ce3a82367b47e0280fa7b4eabb7593/ruff-0.14.6-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:57256efafbfefcb8748df9d1d766062f62b20150691021f8ab79e2d919f7c11f", size = 15199755, upload-time = "2025-11-21T14:25:41.516Z" }, - { url = "https://files.pythonhosted.org/packages/c9/b9/288bb2399860a36d4bb0541cb66cce3c0f4156aaff009dc8499be0c24bf2/ruff-0.14.6-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ff18134841e5c68f8e5df1999a64429a02d5549036b394fafbe410f886e1989d", size = 14850608, upload-time = "2025-11-21T14:25:44.428Z" }, - { url = "https://files.pythonhosted.org/packages/ee/b1/a0d549dd4364e240f37e7d2907e97ee80587480d98c7799d2d8dc7a2f605/ruff-0.14.6-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:29c4b7ec1e66a105d5c27bd57fa93203637d66a26d10ca9809dc7fc18ec58440", size = 14118754, upload-time = "2025-11-21T14:25:47.214Z" }, - { url = "https://files.pythonhosted.org/packages/13/ac/9b9fe63716af8bdfddfacd0882bc1586f29985d3b988b3c62ddce2e202c3/ruff-0.14.6-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:167843a6f78680746d7e226f255d920aeed5e4ad9c03258094a2d49d3028b105", size = 13949214, upload-time = "2025-11-21T14:25:50.002Z" }, - { url = "https://files.pythonhosted.org/packages/12/27/4dad6c6a77fede9560b7df6802b1b697e97e49ceabe1f12baf3ea20862e9/ruff-0.14.6-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:16a33af621c9c523b1ae006b1b99b159bf5ac7e4b1f20b85b2572455018e0821", size = 14106112, upload-time = "2025-11-21T14:25:52.841Z" }, - { url = "https://files.pythonhosted.org/packages/6a/db/23e322d7177873eaedea59a7932ca5084ec5b7e20cb30f341ab594130a71/ruff-0.14.6-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:1432ab6e1ae2dc565a7eea707d3b03a0c234ef401482a6f1621bc1f427c2ff55", size = 13035010, upload-time = "2025-11-21T14:25:55.536Z" }, - { url = "https://files.pythonhosted.org/packages/a8/9c/20e21d4d69dbb35e6a1df7691e02f363423658a20a2afacf2a2c011800dc/ruff-0.14.6-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:4c55cfbbe7abb61eb914bfd20683d14cdfb38a6d56c6c66efa55ec6570ee4e71", size = 13054082, upload-time = "2025-11-21T14:25:58.625Z" }, - { url = "https://files.pythonhosted.org/packages/66/25/906ee6a0464c3125c8d673c589771a974965c2be1a1e28b5c3b96cb6ef88/ruff-0.14.6-py3-none-musllinux_1_2_i686.whl", hash = "sha256:efea3c0f21901a685fff4befda6d61a1bf4cb43de16da87e8226a281d614350b", size = 13303354, upload-time = "2025-11-21T14:26:01.816Z" }, - { url = "https://files.pythonhosted.org/packages/4c/58/60577569e198d56922b7ead07b465f559002b7b11d53f40937e95067ca1c/ruff-0.14.6-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:344d97172576d75dc6afc0e9243376dbe1668559c72de1864439c4fc95f78185", size = 14054487, upload-time = "2025-11-21T14:26:05.058Z" }, - { url = "https://files.pythonhosted.org/packages/67/0b/8e4e0639e4cc12547f41cb771b0b44ec8225b6b6a93393176d75fe6f7d40/ruff-0.14.6-py3-none-win32.whl", hash = "sha256:00169c0c8b85396516fdd9ce3446c7ca20c2a8f90a77aa945ba6b8f2bfe99e85", size = 13013361, upload-time = "2025-11-21T14:26:08.152Z" }, - { url = "https://files.pythonhosted.org/packages/fb/02/82240553b77fd1341f80ebb3eaae43ba011c7a91b4224a9f317d8e6591af/ruff-0.14.6-py3-none-win_amd64.whl", hash = "sha256:390e6480c5e3659f8a4c8d6a0373027820419ac14fa0d2713bd8e6c3e125b8b9", size = 14432087, upload-time = "2025-11-21T14:26:10.891Z" }, - { url = "https://files.pythonhosted.org/packages/a5/1f/93f9b0fad9470e4c829a5bb678da4012f0c710d09331b860ee555216f4ea/ruff-0.14.6-py3-none-win_arm64.whl", hash = "sha256:d43c81fbeae52cfa8728d8766bbf46ee4298c888072105815b392da70ca836b2", size = 13520930, upload-time = "2025-11-21T14:26:13.951Z" }, -] - -[[package]] -name = "stack-data" -version = "0.6.3" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "asttokens" }, - { name = "executing" }, - { name = "pure-eval" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/28/e3/55dcc2cfbc3ca9c29519eb6884dd1415ecb53b0e934862d3559ddcb7e20b/stack_data-0.6.3.tar.gz", hash = "sha256:836a778de4fec4dcd1dcd89ed8abff8a221f58308462e1c4aa2a3cf30148f0b9", size = 44707, upload-time = "2023-09-30T13:58:05.479Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/f1/7b/ce1eafaf1a76852e2ec9b22edecf1daa58175c090266e9f6c64afcd81d91/stack_data-0.6.3-py3-none-any.whl", hash = "sha256:d5558e0c25a4cb0853cddad3d77da9891a08cb85dd9f9f91b9f8cd66e511e695", size = 24521, upload-time = "2023-09-30T13:58:03.53Z" }, -] - -[[package]] -name = "traitlets" -version = "5.14.3" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/eb/79/72064e6a701c2183016abbbfedaba506d81e30e232a68c9f0d6f6fcd1574/traitlets-5.14.3.tar.gz", hash = "sha256:9ed0579d3502c94b4b3732ac120375cda96f923114522847de4b3bb98b96b6b7", size = 161621, upload-time = "2024-04-19T11:11:49.746Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/00/c0/8f5d070730d7836adc9c9b6408dec68c6ced86b304a9b26a14df072a6e8c/traitlets-5.14.3-py3-none-any.whl", hash = "sha256:b74e89e397b1ed28cc831db7aea759ba6640cb3de13090ca145426688ff1ac4f", size = 85359, upload-time = "2024-04-19T11:11:46.763Z" }, +version = "0.14.7" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/b7/5b/dd7406afa6c95e3d8fa9d652b6d6dd17dd4a6bf63cb477014e8ccd3dcd46/ruff-0.14.7.tar.gz", hash = "sha256:3417deb75d23bd14a722b57b0a1435561db65f0ad97435b4cf9f85ffcef34ae5", size = 5727324, upload-time = "2025-11-28T20:55:10.525Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/8c/b1/7ea5647aaf90106f6d102230e5df874613da43d1089864da1553b899ba5e/ruff-0.14.7-py3-none-linux_armv6l.whl", hash = "sha256:b9d5cb5a176c7236892ad7224bc1e63902e4842c460a0b5210701b13e3de4fca", size = 13414475, upload-time = "2025-11-28T20:54:54.569Z" }, + { url = "https://files.pythonhosted.org/packages/af/19/fddb4cd532299db9cdaf0efdc20f5c573ce9952a11cb532d3b859d6d9871/ruff-0.14.7-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:3f64fe375aefaf36ca7d7250292141e39b4cea8250427482ae779a2aa5d90015", size = 13634613, upload-time = "2025-11-28T20:55:17.54Z" }, + { url = "https://files.pythonhosted.org/packages/40/2b/469a66e821d4f3de0440676ed3e04b8e2a1dc7575cf6fa3ba6d55e3c8557/ruff-0.14.7-py3-none-macosx_11_0_arm64.whl", hash = "sha256:93e83bd3a9e1a3bda64cb771c0d47cda0e0d148165013ae2d3554d718632d554", size = 12765458, upload-time = "2025-11-28T20:55:26.128Z" }, + { url = "https://files.pythonhosted.org/packages/f1/05/0b001f734fe550bcfde4ce845948ac620ff908ab7241a39a1b39bb3c5f49/ruff-0.14.7-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3838948e3facc59a6070795de2ae16e5786861850f78d5914a03f12659e88f94", size = 13236412, upload-time = "2025-11-28T20:55:28.602Z" }, + { url = "https://files.pythonhosted.org/packages/11/36/8ed15d243f011b4e5da75cd56d6131c6766f55334d14ba31cce5461f28aa/ruff-0.14.7-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:24c8487194d38b6d71cd0fd17a5b6715cda29f59baca1defe1e3a03240f851d1", size = 13182949, upload-time = "2025-11-28T20:55:33.265Z" }, + { url = "https://files.pythonhosted.org/packages/3b/cf/fcb0b5a195455729834f2a6eadfe2e4519d8ca08c74f6d2b564a4f18f553/ruff-0.14.7-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:79c73db6833f058a4be8ffe4a0913b6d4ad41f6324745179bd2aa09275b01d0b", size = 13816470, upload-time = "2025-11-28T20:55:08.203Z" }, + { url = "https://files.pythonhosted.org/packages/7f/5d/34a4748577ff7a5ed2f2471456740f02e86d1568a18c9faccfc73bd9ca3f/ruff-0.14.7-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:12eb7014fccff10fc62d15c79d8a6be4d0c2d60fe3f8e4d169a0d2def75f5dad", size = 15289621, upload-time = "2025-11-28T20:55:30.837Z" }, + { url = "https://files.pythonhosted.org/packages/53/53/0a9385f047a858ba133d96f3f8e3c9c66a31cc7c4b445368ef88ebeac209/ruff-0.14.7-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6c623bbdc902de7ff715a93fa3bb377a4e42dd696937bf95669118773dbf0c50", size = 14975817, upload-time = "2025-11-28T20:55:24.107Z" }, + { url = "https://files.pythonhosted.org/packages/a8/d7/2f1c32af54c3b46e7fadbf8006d8b9bcfbea535c316b0bd8813d6fb25e5d/ruff-0.14.7-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f53accc02ed2d200fa621593cdb3c1ae06aa9b2c3cae70bc96f72f0000ae97a9", size = 14284549, upload-time = "2025-11-28T20:55:06.08Z" }, + { url = "https://files.pythonhosted.org/packages/92/05/434ddd86becd64629c25fb6b4ce7637dd52a45cc4a4415a3008fe61c27b9/ruff-0.14.7-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:281f0e61a23fcdcffca210591f0f53aafaa15f9025b5b3f9706879aaa8683bc4", size = 14071389, upload-time = "2025-11-28T20:55:35.617Z" }, + { url = "https://files.pythonhosted.org/packages/ff/50/fdf89d4d80f7f9d4f420d26089a79b3bb1538fe44586b148451bc2ba8d9c/ruff-0.14.7-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:dbbaa5e14148965b91cb090236931182ee522a5fac9bc5575bafc5c07b9f9682", size = 14202679, upload-time = "2025-11-28T20:55:01.472Z" }, + { url = "https://files.pythonhosted.org/packages/77/54/87b34988984555425ce967f08a36df0ebd339bb5d9d0e92a47e41151eafc/ruff-0.14.7-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:1464b6e54880c0fe2f2d6eaefb6db15373331414eddf89d6b903767ae2458143", size = 13147677, upload-time = "2025-11-28T20:55:19.933Z" }, + { url = "https://files.pythonhosted.org/packages/67/29/f55e4d44edfe053918a16a3299e758e1c18eef216b7a7092550d7a9ec51c/ruff-0.14.7-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:f217ed871e4621ea6128460df57b19ce0580606c23aeab50f5de425d05226784", size = 13151392, upload-time = "2025-11-28T20:55:21.967Z" }, + { url = "https://files.pythonhosted.org/packages/36/69/47aae6dbd4f1d9b4f7085f4d9dcc84e04561ee7ad067bf52e0f9b02e3209/ruff-0.14.7-py3-none-musllinux_1_2_i686.whl", hash = "sha256:6be02e849440ed3602d2eb478ff7ff07d53e3758f7948a2a598829660988619e", size = 13412230, upload-time = "2025-11-28T20:55:12.749Z" }, + { url = "https://files.pythonhosted.org/packages/b7/4b/6e96cb6ba297f2ba502a231cd732ed7c3de98b1a896671b932a5eefa3804/ruff-0.14.7-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:19a0f116ee5e2b468dfe80c41c84e2bbd6b74f7b719bee86c2ecde0a34563bcc", size = 14195397, upload-time = "2025-11-28T20:54:56.896Z" }, + { url = "https://files.pythonhosted.org/packages/69/82/251d5f1aa4dcad30aed491b4657cecd9fb4274214da6960ffec144c260f7/ruff-0.14.7-py3-none-win32.whl", hash = "sha256:e33052c9199b347c8937937163b9b149ef6ab2e4bb37b042e593da2e6f6cccfa", size = 13126751, upload-time = "2025-11-28T20:55:03.47Z" }, + { url = "https://files.pythonhosted.org/packages/a8/b5/d0b7d145963136b564806f6584647af45ab98946660d399ec4da79cae036/ruff-0.14.7-py3-none-win_amd64.whl", hash = "sha256:e17a20ad0d3fad47a326d773a042b924d3ac31c6ca6deb6c72e9e6b5f661a7c6", size = 14531726, upload-time = "2025-11-28T20:54:59.121Z" }, + { url = "https://files.pythonhosted.org/packages/1d/d2/1637f4360ada6a368d3265bf39f2cf737a0aaab15ab520fc005903e883f8/ruff-0.14.7-py3-none-win_arm64.whl", hash = "sha256:be4d653d3bea1b19742fcc6502354e32f65cd61ff2fbdb365803ef2c2aec6228", size = 13609215, upload-time = "2025-11-28T20:55:15.375Z" }, ] [[package]] @@ -375,12 +212,3 @@ sdist = { url = "https://files.pythonhosted.org/packages/15/22/9ee70a2574a4f4599 wheels = [ { url = "https://files.pythonhosted.org/packages/a7/c2/fe1e52489ae3122415c51f387e221dd0773709bad6c6cdaa599e8a2c5185/urllib3-2.5.0-py3-none-any.whl", hash = "sha256:e6b01673c0fa6a13e374b50871808eb3bf7046c4b125b216f6bf1cc604cff0dc", size = 129795, upload-time = "2025-06-18T14:07:40.39Z" }, ] - -[[package]] -name = "wcwidth" -version = "0.2.14" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/24/30/6b0809f4510673dc723187aeaf24c7f5459922d01e2f794277a3dfb90345/wcwidth-0.2.14.tar.gz", hash = "sha256:4d478375d31bc5395a3c55c40ccdf3354688364cd61c4f6adacaa9215d0b3605", size = 102293, upload-time = "2025-09-22T16:29:53.023Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/af/b5/123f13c975e9f27ab9c0770f514345bd406d0e8d3b7a0723af9d43f710af/wcwidth-0.2.14-py2.py3-none-any.whl", hash = "sha256:a7bb560c8aee30f9957e5f9895805edd20602f2d7f720186dfd906e82b4982e1", size = 37286, upload-time = "2025-09-22T16:29:51.641Z" }, -] From 71764bcf4f3ec8846f7102c174d57c4c2ac2762d Mon Sep 17 00:00:00 2001 From: BABA Toshiaki Date: Sun, 30 Nov 2025 05:58:04 +0000 Subject: [PATCH 16/16] bump version --- pyproject.toml | 35 ++++++++++++++++++++--------------- uv.lock | 2 +- 2 files changed, 21 insertions(+), 16 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index ef4ccdf..d28987f 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -4,33 +4,30 @@ build-backend = "setuptools.build_meta" [project] name = "pybacklog" -version = "0.1.8" +version = "1.0.0" description = "Backlog API v2 Client" readme = "README.md" license = { file = "LICENSE" } -authors = [ - { name = "Toshiaki Baba", email = "toshiaki@netmark.jp" } -] -urls = { "Homepage" = "https://github.com/netmarkjp/pybacklog" } +authors = [{ name = "BABA Toshiaki", email = "toshiaki@netmark.jp" }] +keywords = ["backlog", "api"] classifiers = [ "Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.12", "Programming Language :: Python :: 3.13", "Programming Language :: Python :: 3.14", - "Topic :: Utilities" + "Topic :: Utilities", ] requires-python = ">=3.10" -dependencies = [ - "requests >=2.12.4,<3.0" -] +dependencies = ["requests >=2.12.4,<3.0"] + +[project.urls] +Homepage = "https://github.com/netmarkjp/pybacklog" +Repository = "https://github.com/netmarkjp/pybacklog" [dependency-groups] -dev = [ - "pyright>=1.1.407", - "ruff>=0.14.6", -] +dev = ["pyright>=1.1.407", "ruff>=0.14.6"] [tool.uv] # Additional uv-specific configurations can be added here. @@ -40,8 +37,16 @@ target-version = "py310" line-length = 120 # lint.select = ["E", "F", "W"] # lint.ignore = [] -exclude = [".venv", "build", "dist", ".eggs", "*.egg-info", ".ruff_cache", "__pycache__"] +exclude = [ + ".venv", + "build", + "dist", + ".eggs", + "*.egg-info", + ".ruff_cache", + "__pycache__", +] [tool.pyright] pythonVersion = "3.10" -typeCheckingMode = "strict" \ No newline at end of file +typeCheckingMode = "strict" diff --git a/uv.lock b/uv.lock index 23a5696..d1704f6 100644 --- a/uv.lock +++ b/uv.lock @@ -120,7 +120,7 @@ wheels = [ [[package]] name = "pybacklog" -version = "0.1.8" +version = "1.0.0" source = { editable = "." } dependencies = [ { name = "requests" },