mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-06-14 02:19:58 +08:00
- 发布链路新增 driver release 资产自检,校验已发布二进制与 manifest revision/sha256 一致 - dev-build 与 release 工作流在复用已发布驱动前先校验实际资产,发现旧二进制混入时强制全量重建 - driver manifest 新增 sha256 字段,避免仅凭 source commit 与 manifest 一致就误判发布有效 - 驱动变更检测与全量重建判定纳入 release asset 校验脚本变更,确保链路修复提交会触发自愈重建 - 驱动下载优先使用 GitHub release asset API 地址,并对资产接口按 octet-stream 下载,降低 dev-latest 同名资产命中旧缓存的风险 - 保持 DuckDB 无主键编辑修复不回退,并通过 internal/db 与前端相关回归测试
185 lines
6.2 KiB
Python
185 lines
6.2 KiB
Python
#!/usr/bin/env python3
|
|
|
|
import argparse
|
|
import hashlib
|
|
import json
|
|
import os
|
|
import shutil
|
|
import stat
|
|
import subprocess
|
|
import sys
|
|
import tempfile
|
|
import urllib.error
|
|
import urllib.parse
|
|
import urllib.request
|
|
from pathlib import Path
|
|
|
|
|
|
MANIFEST_ASSET_NAME = "GoNavi-DriverAgents-Manifest.json"
|
|
|
|
|
|
def github_headers(binary: bool = False):
|
|
headers = {
|
|
"Accept": "application/octet-stream" if binary else "application/vnd.github+json",
|
|
"User-Agent": "GoNavi-CI",
|
|
}
|
|
token = os.environ.get("DRIVER_RELEASE_TOKEN") or os.environ.get("GITHUB_TOKEN")
|
|
if token:
|
|
headers["Authorization"] = f"Bearer {token}"
|
|
return headers
|
|
|
|
|
|
def fetch_json(url: str):
|
|
req = urllib.request.Request(url, headers=github_headers(False))
|
|
with urllib.request.urlopen(req, timeout=30) as resp:
|
|
return json.loads(resp.read().decode("utf-8"))
|
|
|
|
|
|
def download_url(url: str, destination: Path):
|
|
req = urllib.request.Request(url, headers=github_headers(False))
|
|
with urllib.request.urlopen(req, timeout=120) as resp, open(destination, "wb") as out:
|
|
shutil.copyfileobj(resp, out)
|
|
|
|
|
|
def load_release(repo: str, tag: str):
|
|
owner_repo = repo.strip()
|
|
if not owner_repo:
|
|
raise ValueError("repo is required")
|
|
if tag == "latest":
|
|
url = f"https://api.github.com/repos/{owner_repo}/releases/latest"
|
|
else:
|
|
url = f"https://api.github.com/repos/{owner_repo}/releases/tags/{urllib.parse.quote(tag, safe='')}"
|
|
return fetch_json(url)
|
|
|
|
|
|
def asset_map(release: dict):
|
|
result = {}
|
|
for asset in release.get("assets", []):
|
|
name = str(asset.get("name") or "").strip()
|
|
if name:
|
|
result[name] = asset
|
|
return result
|
|
|
|
|
|
def infer_asset_path(name: str):
|
|
trimmed = str(name or "").strip()
|
|
if not trimmed:
|
|
return None
|
|
if trimmed == "duckdb.dll":
|
|
return "Windows/duckdb.dll"
|
|
if trimmed == "duckdb-driver.zip":
|
|
return None
|
|
if trimmed.endswith("-driver-agent-windows-amd64.exe") or trimmed.endswith("-driver-agent-windows-arm64.exe"):
|
|
return f"Windows/{trimmed}"
|
|
if trimmed.endswith("-driver-agent-darwin-amd64") or trimmed.endswith("-driver-agent-darwin-arm64"):
|
|
return f"MacOS/{trimmed}"
|
|
if trimmed.endswith("-driver-agent-linux-amd64"):
|
|
return f"Linux/{trimmed}"
|
|
return None
|
|
|
|
|
|
def sha256_file(path: Path):
|
|
h = hashlib.sha256()
|
|
with open(path, "rb") as fh:
|
|
for chunk in iter(lambda: fh.read(1024 * 1024), b""):
|
|
if not chunk:
|
|
break
|
|
h.update(chunk)
|
|
return h.hexdigest()
|
|
|
|
|
|
def probe_metadata_revision(path: Path):
|
|
current_mode = path.stat().st_mode
|
|
path.chmod(current_mode | stat.S_IXUSR)
|
|
proc = subprocess.run(
|
|
[str(path)],
|
|
input=b'{"id":1,"method":"metadata"}\n',
|
|
capture_output=True,
|
|
timeout=20,
|
|
check=True,
|
|
)
|
|
if not proc.stdout:
|
|
raise RuntimeError(f"{path.name}: metadata output is empty")
|
|
payload = json.loads(proc.stdout.decode("utf-8"))
|
|
return str(((payload.get("data") or {}).get("agentRevision") or "")).strip()
|
|
|
|
|
|
def validate_release_assets(release: dict, manifest: dict):
|
|
assets = asset_map(release)
|
|
manifest_assets = manifest.get("assets") or {}
|
|
if not isinstance(manifest_assets, dict) or not manifest_assets:
|
|
raise RuntimeError("manifest assets is empty")
|
|
|
|
mismatches = []
|
|
skipped = []
|
|
with tempfile.TemporaryDirectory(prefix="gonavi-release-assets-") as tmp:
|
|
tmp_root = Path(tmp)
|
|
for name, meta in sorted(manifest_assets.items()):
|
|
if name == MANIFEST_ASSET_NAME:
|
|
continue
|
|
asset = assets.get(name)
|
|
if asset is None:
|
|
mismatches.append((name, "missing_release_asset", "", "present in manifest"))
|
|
continue
|
|
|
|
expected_sha = str(meta.get("sha256") or "").strip().lower()
|
|
expected_revision = str(meta.get("revision") or "").strip()
|
|
path_hint = infer_asset_path(name)
|
|
if path_hint is None:
|
|
skipped.append(name)
|
|
continue
|
|
|
|
local_path = tmp_root / name
|
|
download_url(str(asset.get("browser_download_url") or "").strip(), local_path)
|
|
|
|
actual_sha = sha256_file(local_path).lower()
|
|
if expected_sha and actual_sha != expected_sha:
|
|
mismatches.append((name, "sha256", actual_sha, expected_sha))
|
|
continue
|
|
|
|
if expected_revision:
|
|
actual_revision = probe_metadata_revision(local_path)
|
|
if actual_revision != expected_revision:
|
|
mismatches.append((name, "revision", actual_revision, expected_revision))
|
|
|
|
return mismatches, skipped
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("--repo", default="Syngnat/GoNavi-DriverAgents")
|
|
parser.add_argument("--tag", required=True)
|
|
args = parser.parse_args()
|
|
|
|
release = load_release(args.repo, args.tag)
|
|
assets = asset_map(release)
|
|
manifest_asset = assets.get(MANIFEST_ASSET_NAME)
|
|
if manifest_asset is None:
|
|
raise SystemExit(f"release {args.repo}@{args.tag} missing {MANIFEST_ASSET_NAME}")
|
|
|
|
with tempfile.TemporaryDirectory(prefix="gonavi-release-manifest-") as tmp:
|
|
manifest_path = Path(tmp) / MANIFEST_ASSET_NAME
|
|
download_url(str(manifest_asset.get("browser_download_url") or "").strip(), manifest_path)
|
|
manifest = json.loads(manifest_path.read_text(encoding="utf-8"))
|
|
|
|
mismatches, skipped = validate_release_assets(release, manifest)
|
|
if mismatches:
|
|
print("published driver release assets mismatch manifest:", file=sys.stderr)
|
|
for name, field, actual, expected in mismatches:
|
|
print(f" - {name} [{field}] actual={actual or '<empty>'} expected={expected or '<empty>'}", file=sys.stderr)
|
|
raise SystemExit(1)
|
|
|
|
checked = len((manifest.get("assets") or {})) - len(skipped)
|
|
print(f"driver release assets validation passed: checked={checked} skipped={len(skipped)}")
|
|
if skipped:
|
|
print("skipped assets: " + ", ".join(sorted(skipped)))
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
try:
|
|
raise SystemExit(main())
|
|
except urllib.error.HTTPError as exc:
|
|
print(f"http error: {exc.code}", file=sys.stderr)
|
|
raise
|