mirror of
https://github.com/kennethreitz/pipenv.git
synced 2026-06-05 06:46:15 +00:00
240 lines
7.7 KiB
Python
240 lines
7.7 KiB
Python
import collections
|
|
import contextlib
|
|
import io
|
|
import json
|
|
import os
|
|
|
|
from tarfile import is_tarfile
|
|
from zipfile import is_zipfile
|
|
|
|
import distlib.wheel
|
|
import requests
|
|
from six.moves import xmlrpc_client
|
|
|
|
from flask import Flask, redirect, abort, render_template, send_file, jsonify
|
|
|
|
|
|
ReleaseTuple = collections.namedtuple("ReleaseTuple", ["path", "requires_python", "hash"])
|
|
|
|
app = Flask(__name__)
|
|
session = requests.Session()
|
|
|
|
packages = {}
|
|
ARTIFACTS = {}
|
|
|
|
|
|
@contextlib.contextmanager
|
|
def xml_pypi_server(server):
|
|
transport = xmlrpc_client.Transport()
|
|
client = xmlrpc_client.ServerProxy(server, transport)
|
|
try:
|
|
yield client
|
|
finally:
|
|
transport.close()
|
|
|
|
|
|
def get_pypi_package_names():
|
|
pypi_packages = set()
|
|
with xml_pypi_server("https://pypi.org/pypi") as client:
|
|
pypi_packages = set(client.list_packages())
|
|
return pypi_packages
|
|
|
|
|
|
class Package:
|
|
"""Package represents a collection of releases from one or more directories"""
|
|
|
|
def __init__(self, name):
|
|
super().__init__()
|
|
self.name = name
|
|
self.releases = {}
|
|
self._package_dirs = set()
|
|
|
|
@property
|
|
def json(self):
|
|
for path, _ in self._package_dirs:
|
|
try:
|
|
with open(os.path.join(path, 'api.json')) as f:
|
|
return json.load(f)
|
|
except FileNotFoundError:
|
|
r = session.get(f'https://pypi.org/pypi/{self.name}/json')
|
|
response = r.json()
|
|
releases = response["releases"]
|
|
files = {
|
|
pkg for pkg_dir in self._package_dirs
|
|
for pkg in os.listdir(pkg_dir.path)
|
|
}
|
|
for release in list(releases.keys()):
|
|
values = (
|
|
r for r in releases[release] if r["filename"] in files
|
|
)
|
|
values = list(values)
|
|
if values:
|
|
releases[release] = values
|
|
else:
|
|
del releases[release]
|
|
response["releases"] = releases
|
|
with open(os.path.join(path, "api.json"), "w") as fh:
|
|
json.dump(response, fh, indent=4)
|
|
return response
|
|
|
|
def __repr__(self):
|
|
return f"<Package name={self.name!r} releases={len(self.releases)!r}"
|
|
|
|
def add_release(self, path_to_binary):
|
|
path_to_binary = os.path.abspath(path_to_binary)
|
|
path, release = os.path.split(path_to_binary)
|
|
requires_python = ""
|
|
hash_value = ""
|
|
if path_to_binary.endswith(".whl"):
|
|
pkg = distlib.wheel.Wheel(path_to_binary)
|
|
md_dict = pkg.metadata.todict()
|
|
requires_python = md_dict.get("requires_python", "")
|
|
if requires_python.count(".") > 1:
|
|
requires_python, _, _ = requires_python.rpartition(".")
|
|
if os.path.isfile(path_to_binary + ".sha256"):
|
|
with open(path_to_binary + ".sha256") as f:
|
|
hash_value = f.read().strip()
|
|
self.releases[release] = ReleaseTuple(path_to_binary, requires_python, hash_value)
|
|
self._package_dirs.add(ReleaseTuple(path, requires_python, hash_value))
|
|
|
|
|
|
class Artifact:
|
|
"""Represents an artifact for download"""
|
|
|
|
def __init__(self, name):
|
|
super().__init__()
|
|
self.name = name
|
|
self.files = {}
|
|
self._artifact_dirs = set()
|
|
|
|
def __repr__(self):
|
|
return f"<Artifact name={self.name!r} files={len(self.files)!r}"
|
|
|
|
def add_file(self, path):
|
|
path = os.path.abspath(path)
|
|
base_path, fn = os.path.split(path)
|
|
self.files[fn] = path
|
|
self._artifact_dirs.add(base_path)
|
|
|
|
|
|
def prepare_fixtures(path):
|
|
path = os.path.abspath(path)
|
|
if not (os.path.exists(path) and os.path.isdir(path)):
|
|
raise ValueError(f"{path} is not a directory!")
|
|
for root, dirs, files in os.walk(path):
|
|
package_name, _, _ = os.path.relpath(root, start=path).partition(os.path.sep)
|
|
if package_name not in ARTIFACTS:
|
|
ARTIFACTS[package_name] = Artifact(package_name)
|
|
for file in files:
|
|
file_path = os.path.join(root, file)
|
|
rel_path = os.path.relpath(file_path, start=path)
|
|
_, _, subpkg = rel_path.partition(os.path.sep)
|
|
subpkg, _, _ = subpkg.partition(os.path.sep)
|
|
pkg, ext = os.path.splitext(subpkg)
|
|
if not (is_tarfile(file_path) or is_zipfile(file_path) or ext == ".git"):
|
|
continue
|
|
if subpkg not in ARTIFACTS[package_name].files:
|
|
ARTIFACTS[package_name].add_file(os.path.join(root, file))
|
|
ARTIFACTS[package_name].add_file(os.path.join(root, file))
|
|
|
|
|
|
def prepare_packages(path):
|
|
"""Add packages in path to the registry."""
|
|
path = os.path.abspath(path)
|
|
if not (os.path.exists(path) and os.path.isdir(path)):
|
|
raise ValueError(f"{path} is not a directory!")
|
|
for root, dirs, files in os.walk(path):
|
|
if all([setup_file in list(files) for setup_file in ("setup.py", "setup.cfg")]):
|
|
continue
|
|
for file in files:
|
|
if not file.startswith('.') and not file.endswith('.json'):
|
|
package_name = os.path.basename(root)
|
|
if package_name and package_name == "fixtures":
|
|
prepare_fixtures(root)
|
|
continue
|
|
package_name = package_name.replace("_", "-")
|
|
if package_name not in packages:
|
|
packages[package_name] = Package(package_name)
|
|
|
|
packages[package_name].add_release(os.path.join(root, file))
|
|
# remaining = get_pypi_package_names() - set(list(packages.keys()))
|
|
# for pypi_pkg in remaining:
|
|
# packages[pypi_pkg] = Package(pypi_pkg)
|
|
|
|
|
|
@app.route('/')
|
|
def hello_world():
|
|
return redirect('/simple', code=302)
|
|
|
|
|
|
@app.route('/simple')
|
|
def simple():
|
|
return render_template('simple.html', packages=packages.values())
|
|
|
|
|
|
@app.route('/artifacts')
|
|
def artifacts():
|
|
return render_template('artifacts.html', artifacts=ARTIFACTS.values())
|
|
|
|
|
|
@app.route('/simple/<package>/')
|
|
def simple_package(package):
|
|
if package in packages and packages[package].releases:
|
|
return render_template('package.html', package=packages[package])
|
|
else:
|
|
try:
|
|
r = requests.get(f"https://pypi.org/simple/{package}")
|
|
r.raise_for_status()
|
|
except Exception:
|
|
abort(404)
|
|
else:
|
|
return render_template(
|
|
'package_pypi.html', package_contents=r.text
|
|
)
|
|
|
|
|
|
@app.route('/artifacts/<artifact>/')
|
|
def simple_artifact(artifact):
|
|
if artifact in ARTIFACTS:
|
|
return render_template('artifact.html', artifact=ARTIFACTS[artifact])
|
|
else:
|
|
abort(404)
|
|
|
|
|
|
@app.route('/<package>/<release>')
|
|
def serve_package(package, release):
|
|
if package in packages:
|
|
package = packages[package]
|
|
|
|
if release in package.releases:
|
|
return send_file(package.releases[release].path)
|
|
|
|
abort(404)
|
|
|
|
|
|
@app.route('/artifacts/<artifact>/<fn>')
|
|
def serve_artifact(artifact, fn):
|
|
if artifact in ARTIFACTS:
|
|
artifact = ARTIFACTS[artifact]
|
|
if fn in artifact.files:
|
|
return send_file(artifact.files[fn])
|
|
abort(404)
|
|
|
|
|
|
@app.route('/pypi/<package>/json')
|
|
def json_for_package(package):
|
|
return jsonify(packages[package].json)
|
|
# try:
|
|
# except Exception:
|
|
# r = session.get('https://pypi.org/pypi/{0}/json'.format(package))
|
|
# return jsonify(r.json())
|
|
|
|
|
|
if __name__ == '__main__':
|
|
PYPI_VENDOR_DIR = os.environ.get('PYPI_VENDOR_DIR', './pypi')
|
|
PYPI_VENDOR_DIR = os.path.abspath(PYPI_VENDOR_DIR)
|
|
prepare_packages(PYPI_VENDOR_DIR)
|
|
prepare_fixtures(os.path.join(PYPI_VENDOR_DIR, "fixtures"))
|
|
|
|
app.run()
|