#!/usr/bin/env python3
"""
Update the wheel index for PR SGLang releases.
This script generates a single PyPI-compatible index.html file at pr/index.html
containing all PR builds, ordered by PR number and commit count (newest first).
Similar to update_nightly_whl_index.py but for PR builds.
"""
import argparse
import hashlib
import pathlib
import re
def compute_sha256(file_path: pathlib.Path) -> str:
"""Compute SHA256 hash of a file."""
sha256_hash = hashlib.sha256()
with open(file_path, "rb") as f:
for byte_block in iter(lambda: f.read(4096), b""):
sha256_hash.update(byte_block)
return sha256_hash.hexdigest()
def update_wheel_index(
pr_number: str, commit_hash: str, wheel_version: str, build_date: str
):
"""Update the wheel index for PR releases.
Creates a single index at pr/index.html containing all PR builds.
Args:
pr_number: PR number (e.g., '123')
commit_hash: Short git commit hash (e.g., 'c5f1e86')
wheel_version: Full wheel version string (e.g., '0.5.6.dev7716+pr-123.gc5f1e86')
build_date: Build date in YYYY-MM-DD format (e.g., '2025-12-13')
"""
dist_dir = pathlib.Path("dist")
whl_repo_dir = pathlib.Path("sgl-whl")
if not dist_dir.exists():
print(f"Warning: {dist_dir} does not exist, skipping index update")
return
# Base URL for wheels stored in GitHub Releases
base_url = "https://github.com/sgl-project/whl/releases/download"
release_tag = f"pr-{pr_number}-{build_date}-{commit_hash}"
# Create pr directory structure following PEP 503
# /pr/index.html -> links to sglang/
# /pr/sglang/index.html -> contains wheel links
pr_dir = whl_repo_dir / "pr"
pr_dir.mkdir(parents=True, exist_ok=True)
sglang_dir = pr_dir / "sglang"
sglang_dir.mkdir(parents=True, exist_ok=True)
root_index = pr_dir / "index.html"
package_index = sglang_dir / "index.html"
print(f"\nUpdating PR wheel index")
print(f" Root index: {root_index}")
print(f" Package index: {package_index}")
# Read existing package index if it exists
existing_links = []
if package_index.exists():
with open(package_index, "r") as f:
content = f.read()
# Extract existing links (skip header and HTML boilerplate)
existing_links = [
line for line in content.split("\n") if line.startswith("{filename}
'
new_links.append(link)
print(f" Added: {filename}")
except Exception as e:
print(f" Error processing {wheel_path.name}: {e}")
continue
if not new_links:
print(" No new wheels to add")
return
# Combine existing and new links (new links first for latest)
all_links = new_links + existing_links
# Remove duplicates while preserving order (newer first)
seen = set()
unique_links = []
for link in all_links:
# Extract filename from link to check for duplicates
filename_match = re.search(r">([^<]+\.whl)", link)
if filename_match:
filename = filename_match.group(1)
if filename not in seen:
seen.add(filename)
unique_links.append(link)
# Write root index (links to sglang package directory)
with open(root_index, "w") as f:
f.write("\n")
f.write('sglang\n')
print(f" Written root index: {root_index}")
# Write package index in minimal format
with open(package_index, "w") as f:
f.write("\n")
f.write("