Skip to main content

Offline deep validator for pyproject.toml: file existence, URL and email format, PEP 440 version and PEP 508 specifier validity, version-constraint satisfiability, and entry-point format.

Project description

pyproject-doctor

pyproject-doctor logo

Offline deep validator for pyproject.toml that catches the semantic mistakes other tools miss.

Most validators only check TOML syntax. pyproject-doctor goes further: it validates that your versions are PEP 440 compliant, your requires-python is a valid specifier set, your dependencies are PEP 508 compliant, your version constraints are actually satisfiable, your referenced files exist, your URLs are real URLs, your email addresses look right, and your entry-point references are correctly formatted.

What it checks

Code Description
version-invalid project.version is not a valid PEP 440 version
requires-python-invalid project.requires-python is empty, malformed, not a valid PEP 440 specifier set, or unsatisfiable (e.g. >=3.12,<3.8)
dep-invalid A dependency in project.dependencies, project.optional-dependencies, or build-system.requires is not a valid PEP 508 requirement
constraint-unsatisfiable A dependency's version specifiers form an impossible range (e.g. >=2.0,<1.0)
file-missing A file referenced by project.readme, project.license, or an entry-point module does not exist
url-invalid A value in project.urls is not a valid absolute URL
email-invalid An author or maintainer email address is malformed
entry-point-invalid A script or entry-point value is not in valid module:attr format
classifier-unknown A classifier in project.classifiers is not a known trove classifier (requires pip install 'pyproject-doctor[classifiers]')
dynamic-malformed project.dynamic is not a list of strings
dynamic-name-forbidden name appears in project.dynamic (PEP 621 requires it to always be static)
dynamic-field-unknown An entry in project.dynamic is not a recognized [project] field name
dynamic-static-conflict A field listed in project.dynamic is also set statically in [project]

Install

pip install pyproject-doctor

Or with classifier validation:

pip install "pyproject-doctor[classifiers]"

Usage

# Validate pyproject.toml in the current directory
pyproject-doctor

# Validate a specific file
pyproject-doctor /path/to/pyproject.toml

# JSON output
pyproject-doctor --format json

# SARIF 2.1.0 output (for GitHub code scanning and other SARIF consumers)
pyproject-doctor --format sarif

Exit code is 1 if any error-level diagnostic is found, 0 otherwise.

The sarif format emits a SARIF 2.1.0 log: each diagnostic becomes one result, the diagnostic code is the ruleId, every result points at the analyzed pyproject.toml, and the SARIF level (error, warning, note) is derived from the diagnostic's level.

Pre-commit

Add to .pre-commit-config.yaml:

repos:
  - repo: https://github.com/amaar-mc/pyproject-doctor
    rev: v0.3.0
    hooks:
      - id: pyproject-doctor

Example output

error project.version: version-invalid: 'not.a.version' is not a valid PEP 440 version
error project.dependencies[0]: constraint-unsatisfiable: Dependency 'requests': constraint '>=3.0,<2.0' is unsatisfiable (lower bound 3.0 exceeds upper bound 2.0)
error project.urls.Homepage: url-invalid: URL 'not-a-url' is not a valid absolute URL (must have scheme and host)

License

MIT. Copyright (c) 2026 Amaar Chughtai.

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

pyproject_doctor-0.3.0.tar.gz (952.1 kB view details)

Uploaded Source

Built Distribution

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

pyproject_doctor-0.3.0-py3-none-any.whl (18.8 kB view details)

Uploaded Python 3

File details

Details for the file pyproject_doctor-0.3.0.tar.gz.

File metadata

  • Download URL: pyproject_doctor-0.3.0.tar.gz
  • Upload date:
  • Size: 952.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.3

File hashes

Hashes for pyproject_doctor-0.3.0.tar.gz
Algorithm Hash digest
SHA256 97de084b2f29097e293b4f60a11874def6d084b9bebed19e8f69f201ffa6a676
MD5 446a0e4570621c17a4e97d91ff4f06dc
BLAKE2b-256 2465ae0372f656fe28c4a3fdb96bcaca606448e740c0012ddebba743207a8e25

See more details on using hashes here.

File details

Details for the file pyproject_doctor-0.3.0-py3-none-any.whl.

File metadata

File hashes

Hashes for pyproject_doctor-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 f9da1449ceae05558076a8efcac0b4bd4b2b1bf584a3c7620824bef58b3b915c
MD5 925ba2cbd5b093e2e357db9688e14ea4
BLAKE2b-256 bcc53fd0475efaf4c2a91af9c716a3666083dbf87a493f3e3bd98f25e9d94d4d

See more details on using hashes here.

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