Skip to main content

A lightweight script runner for uv — define and run project scripts from pyproject.toml

Project description

uv-script

PyPI

A lightweight, zero-dependency script runner for uv. Define project scripts in pyproject.toml and run them through uv run.

$ uvs --list
  check   lint -> test
  format  ruff format src/
  lint    ruff check src/
  test    pytest tests/ -v

Why?

uv is a fantastic package manager but has no built-in task runner. If you've been using Hatch just for its scripts, or reaching for Poe the Poet / Taskipy to fill the gap, uvs offers a simpler alternative:

  • Zero runtime dependencies — stdlib only (tomllib, argparse, subprocess)
  • uv-native — every command runs through uv run, so your venv, lockfile, and Python version are always in sync
  • Familiar — if you've used npm scripts or Hatch scripts, you already know how this works

Installation

# As a dev dependency in your project
uv add --dev uv-script

# Or run without installing
uvx uv-script --list

Quick start

Add a [tool.uvs.scripts] section to your pyproject.toml:

[tool.uvs.scripts]
test = "pytest tests/ -v"
lint = "ruff check src/"
format = "ruff format src/"
check = ["lint", "test"]

Then run:

uvs test       # runs: uv run pytest tests/ -v
uvs check      # runs lint, then test (stops on first failure)
uvs --list     # shows all available scripts

Configuration

Scripts are defined under [tool.uvs.scripts] in pyproject.toml. Three formats are supported:

Simple command

A string value runs a single command through uv run:

[tool.uvs.scripts]
test = "pytest tests/ -v"
lint = "ruff check ."

Composite script

An array of strings runs multiple steps sequentially. Items can reference other script names or be raw commands. Execution stops on the first non-zero exit code.

[tool.uvs.scripts]
lint = "ruff check ."
test = "pytest tests/"
check = ["lint", "test"]
full = ["ruff format --check .", "lint", "test"]

Table with options

A table gives you control over environment variables and help text:

[tool.uvs.scripts.serve]
cmd = "python -m flask run"
env = { FLASK_DEBUG = "1", FLASK_APP = "app.py" }
help = "Start the development server"

[tool.uvs.scripts.deploy]
cmd = "python scripts/deploy.py"
env = { ENV = "production" }
help = "Deploy to production"
Key Type Required Description
cmd string yes The command to run
env table of strings no Environment variables (overlays current)
help string no Description shown in --list output

Editable installs

For monorepos with multiple packages, you can configure editable installs under [tool.uvs]. These are passed as --with-editable flags to every uv run invocation, so local packages take priority over PyPI versions:

[tool.uvs]
editable = ["../shared-lib", "../auth-service"]

[tool.uvs.scripts]
test = "pytest tests/ -v"

Paths are resolved relative to the pyproject.toml directory. To skip editable installs for a single run, use --no-editable:

uvs --no-editable test

Usage

uvs [options] <script> [-- extra-args...]
Flag Description
-l, --list List all available scripts
-v, --verbose Print each command before running
--no-editable Ignore editable installs from config
-V, --version Show version and exit

Passing extra arguments

Use -- to forward arguments to the underlying command:

uvs test -- -k "test_parse" --no-header
# runs: uv run pytest tests/ -v -k test_parse --no-header

For composite scripts, extra arguments are appended to the last command in the chain.

Running from subdirectories

uvs walks up from your current directory to find the nearest pyproject.toml, just like uv does. You can run uvs test from anywhere inside your project.

How it works

uvs is a thin orchestration layer. Every command is prefixed with uv run, which means:

  1. Your virtual environment is automatically activated
  2. Dependencies are synced from the lockfile if needed
  3. The correct Python version is used

There is no magic — uvs test is equivalent to typing uv run pytest tests/ -v yourself.

Development

This project uses uvs to manage its own scripts:

git clone <repo-url> && cd uv-script
uv sync
uvs test       # run tests
uvs lint       # run linter
uvs check      # lint + test

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

uv_script-0.1.8.tar.gz (5.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

uv_script-0.1.8-py3-none-any.whl (7.5 kB view details)

Uploaded Python 3

File details

Details for the file uv_script-0.1.8.tar.gz.

File metadata

  • Download URL: uv_script-0.1.8.tar.gz
  • Upload date:
  • Size: 5.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for uv_script-0.1.8.tar.gz
Algorithm Hash digest
SHA256 d1e9d84aa27980d7db4b35426a91806b533542e02eec7c1f9b0621e92c779fc1
MD5 9f2351c004218bcfe5aa84408aca694d
BLAKE2b-256 5bb3b2fb45be4b3020f250daeee34f2a805f8b621a975308faf65a0b569b13d0

See more details on using hashes here.

Provenance

The following attestation bundles were made for uv_script-0.1.8.tar.gz:

Publisher: python-publish.yml on fsimkovic/uv-script

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file uv_script-0.1.8-py3-none-any.whl.

File metadata

  • Download URL: uv_script-0.1.8-py3-none-any.whl
  • Upload date:
  • Size: 7.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for uv_script-0.1.8-py3-none-any.whl
Algorithm Hash digest
SHA256 9d58670a377b962d34a02859b28d37311311a9f5b8352f50646a6195a015afc0
MD5 6f326bfd7d1edde6400a10df7bdeca6f
BLAKE2b-256 da739a20cb8f39f30f3ad71a7f19bb6689aec0972888500b643c707eb6371fdc

See more details on using hashes here.

Provenance

The following attestation bundles were made for uv_script-0.1.8-py3-none-any.whl:

Publisher: python-publish.yml on fsimkovic/uv-script

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page