The virtual engine now detects local members (repo_type=local) and
generates their package index in-memory instead of trying to fetch
from a non-existent upstream.
- MemberIndex gains RepoType field so mergers use correct URL prefix
(/api/v1/local/ vs /api/v1/remote/)
- Virtual engine accepts a LocalIndexGenerator interface for producing
local PyPI indexes
- LocalHandler implements GeneratePyPIPackageHTML for reuse by both
the direct serving path and the virtual merger
- Includes local PyPI upload support (cherry-picked from benvin/local-pypi)
Tested e2e: local wheel upload + virtual merge + uv pip install from
both direct local and virtual URLs
Upload Python wheels and sdists to local PyPI repos. The simple index
(PEP 503) is computed on-demand from stored files.
- Upload validates .whl/.tar.gz/.zip filenames, parses and normalizes
package names per PEP 503, stores under {package}/{filename}
- GET /api/v1/local/{name}/simple/ serves root index listing all packages
- GET /api/v1/local/{name}/simple/{pkg}/ serves per-package file listing
with sha256 hashes for integrity verification
- Files are downloadable at /api/v1/local/{name}/{package}/{filename}
- Overwrites rejected with 409
Tested e2e: uv build wheel → upload → uv pip install from local repo