mirror of
				https://github.com/astral-sh/ruff.git
				synced 2025-11-04 05:34:54 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			160 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
	
		
			4.8 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.
 | 
						|
"""
 | 
						|
 | 
						|
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
 | 
						|
 | 
						|
ruff_repo = "https://github.com/astral-sh/ruff"
 | 
						|
root = Path(
 | 
						|
    check_output(["git", "rev-parse", "--show-toplevel"], text=True).strip(),
 | 
						|
)
 | 
						|
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.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.joinpath("src")
 | 
						|
    check_call(["npm", "install"], cwd=schemastore_path)
 | 
						|
 | 
						|
    # Update the schema and format appropriately
 | 
						|
    schema = json.loads(root.joinpath("ruff.schema.json").read_text())
 | 
						|
    schema["$id"] = "https://json.schemastore.org/ruff.json"
 | 
						|
    src.joinpath(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 = root.joinpath("schemastore")
 | 
						|
    if schemastore_existing.is_dir():
 | 
						|
        update_schemastore(schemastore_existing, schemastore_repos)
 | 
						|
    else:
 | 
						|
        with TemporaryDirectory() as temp_dir:
 | 
						|
            update_schemastore(
 | 
						|
                Path(temp_dir).joinpath("schemastore"), schemastore_repos
 | 
						|
            )
 | 
						|
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    main()
 |