mirror of
https://github.com/astral-sh/ruff.git
synced 2025-07-07 21:25:08 +00:00
168 lines
5 KiB
Python
168 lines
5 KiB
Python
"""Update ruff.json in schemastore.
|
|
|
|
This script will clone `astral-sh/schemastore`, update the schema and push the changes
|
|
to a new branch tagged with the ruff git hash. You should see a URL to create the PR
|
|
to schemastore in the CLI.
|
|
|
|
Usage:
|
|
|
|
uv run --only-dev scripts/update_schemastore.py
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import enum
|
|
import json
|
|
from pathlib import Path
|
|
from subprocess import check_call, check_output
|
|
from tempfile import TemporaryDirectory
|
|
from typing import NamedTuple, assert_never
|
|
|
|
# The remote URL for the `ruff` repository.
|
|
RUFF_REPO = "https://github.com/astral-sh/ruff"
|
|
|
|
# The path to the root of the `ruff` repository.
|
|
RUFF_ROOT = Path(__file__).parent.parent
|
|
|
|
# The path to the JSON schema in the `ruff` repository.
|
|
RUFF_SCHEMA = RUFF_ROOT / "ruff.schema.json"
|
|
|
|
# The path to the JSON schema in the `schemastore` repository.
|
|
RUFF_JSON = Path("schemas/json/ruff.json")
|
|
|
|
|
|
class SchemastoreRepos(NamedTuple):
|
|
fork: str
|
|
upstream: str
|
|
|
|
|
|
class GitProtocol(enum.Enum):
|
|
SSH = "ssh"
|
|
HTTPS = "https"
|
|
|
|
def schemastore_repos(self) -> SchemastoreRepos:
|
|
match self:
|
|
case GitProtocol.SSH:
|
|
return SchemastoreRepos(
|
|
fork="git@github.com:astral-sh/schemastore.git",
|
|
upstream="git@github.com:SchemaStore/schemastore.git",
|
|
)
|
|
case GitProtocol.HTTPS:
|
|
return SchemastoreRepos(
|
|
fork="https://github.com/astral-sh/schemastore.git",
|
|
upstream="https://github.com/SchemaStore/schemastore.git",
|
|
)
|
|
case _:
|
|
assert_never(self)
|
|
|
|
|
|
def update_schemastore(
|
|
schemastore_path: Path, schemastore_repos: SchemastoreRepos
|
|
) -> None:
|
|
if not (schemastore_path / ".git").is_dir():
|
|
check_call(
|
|
["git", "clone", schemastore_repos.fork, schemastore_path, "--depth=1"]
|
|
)
|
|
check_call(
|
|
[
|
|
"git",
|
|
"remote",
|
|
"add",
|
|
"upstream",
|
|
schemastore_repos.upstream,
|
|
],
|
|
cwd=schemastore_path,
|
|
)
|
|
# Create a new branch tagged with the current ruff commit up to date with the latest
|
|
# upstream schemastore
|
|
check_call(["git", "fetch", "upstream"], cwd=schemastore_path)
|
|
current_sha = check_output(["git", "rev-parse", "HEAD"], text=True).strip()
|
|
branch = f"update-ruff-{current_sha}"
|
|
check_call(
|
|
["git", "switch", "-c", branch],
|
|
cwd=schemastore_path,
|
|
)
|
|
check_call(
|
|
["git", "reset", "--hard", "upstream/master"],
|
|
cwd=schemastore_path,
|
|
)
|
|
|
|
# Run npm install
|
|
src = schemastore_path / "src"
|
|
check_call(["npm", "install"], cwd=schemastore_path)
|
|
|
|
# Update the schema and format appropriately
|
|
schema = json.loads(RUFF_SCHEMA.read_text())
|
|
schema["$id"] = "https://json.schemastore.org/ruff.json"
|
|
(src / RUFF_JSON).write_text(
|
|
json.dumps(dict(schema.items()), indent=2, ensure_ascii=False),
|
|
)
|
|
check_call(
|
|
[
|
|
"../node_modules/prettier/bin/prettier.cjs",
|
|
"--plugin",
|
|
"prettier-plugin-sort-json",
|
|
"--write",
|
|
RUFF_JSON,
|
|
],
|
|
cwd=src,
|
|
)
|
|
|
|
# Check if the schema has changed
|
|
# https://stackoverflow.com/a/9393642/3549270
|
|
if check_output(["git", "status", "-s"], cwd=schemastore_path).strip():
|
|
# Schema has changed, commit and push
|
|
commit_url = f"{RUFF_REPO}/commit/{current_sha}"
|
|
commit_body = (
|
|
f"This updates ruff's JSON schema to [{current_sha}]({commit_url})"
|
|
)
|
|
# https://stackoverflow.com/a/22909204/3549270
|
|
check_call(
|
|
[
|
|
"git",
|
|
"commit",
|
|
"-a",
|
|
"-m",
|
|
"Update ruff's JSON schema",
|
|
"-m",
|
|
commit_body,
|
|
],
|
|
cwd=schemastore_path,
|
|
)
|
|
# This should show the link to create a PR
|
|
check_call(
|
|
["git", "push", "--set-upstream", "origin", branch, "--force"],
|
|
cwd=schemastore_path,
|
|
)
|
|
else:
|
|
print("No changes")
|
|
|
|
|
|
def determine_git_protocol(argv: list[str] | None = None) -> GitProtocol:
|
|
import argparse
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description=__doc__, formatter_class=argparse.RawDescriptionHelpFormatter
|
|
)
|
|
parser.add_argument(
|
|
"--proto",
|
|
choices=[proto.value for proto in GitProtocol],
|
|
default="https",
|
|
help="Protocol to use for git authentication",
|
|
)
|
|
args = parser.parse_args(argv)
|
|
return GitProtocol(args.proto)
|
|
|
|
|
|
def main() -> None:
|
|
schemastore_repos = determine_git_protocol().schemastore_repos()
|
|
schemastore_existing = RUFF_ROOT / "schemastore"
|
|
if schemastore_existing.is_dir():
|
|
update_schemastore(schemastore_existing, schemastore_repos)
|
|
else:
|
|
with TemporaryDirectory(prefix="ruff-schemastore-") as temp_dir:
|
|
update_schemastore(Path(temp_dir), schemastore_repos)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|