348 lines
11 KiB
Python
348 lines
11 KiB
Python
"""Rule for running package.json scripts with Bun."""
|
|
|
|
|
|
def _shell_quote(value):
|
|
return "'" + value.replace("'", "'\"'\"'") + "'"
|
|
|
|
|
|
def _bun_script_impl(ctx):
|
|
toolchain = ctx.toolchains["//bun:toolchain_type"]
|
|
bun_bin = toolchain.bun.bun_bin
|
|
package_json = ctx.file.package_json
|
|
|
|
launcher = ctx.actions.declare_file(ctx.label.name)
|
|
ctx.actions.write(
|
|
output = launcher,
|
|
is_executable = True,
|
|
content = """#!/usr/bin/env bash
|
|
set -euo pipefail
|
|
|
|
runfiles_dir="${{RUNFILES_DIR:-$0.runfiles}}"
|
|
workspace_root="${{runfiles_dir}}/_main"
|
|
workspace_root="$(cd "${{workspace_root}}" && pwd -P)"
|
|
bun_bin="${{runfiles_dir}}/_main/{bun_short_path}"
|
|
package_json="${{runfiles_dir}}/_main/{package_json_short_path}"
|
|
package_dir="$(cd "$(dirname "${{package_json}}")" && pwd -P)"
|
|
package_rel_dir="{package_rel_dir}"
|
|
|
|
select_primary_node_modules() {{
|
|
local selected=""
|
|
local fallback=""
|
|
while IFS= read -r node_modules_dir; do
|
|
if [[ -z "${{fallback}}" ]]; then
|
|
fallback="${{node_modules_dir}}"
|
|
fi
|
|
|
|
if [[ ! -d "${{node_modules_dir}}/.bun" ]]; then
|
|
continue
|
|
fi
|
|
|
|
if [[ "${{node_modules_dir}}" != *"/runfiles/_main/"* ]]; then
|
|
selected="${{node_modules_dir}}"
|
|
break
|
|
fi
|
|
|
|
if [[ -z "${{selected}}" ]]; then
|
|
selected="${{node_modules_dir}}"
|
|
fi
|
|
done < <(find -L "${{runfiles_dir}}" -type d -name node_modules 2>/dev/null | sort)
|
|
|
|
if [[ -n "${{selected}}" ]]; then
|
|
echo "${{selected}}"
|
|
else
|
|
echo "${{fallback}}"
|
|
fi
|
|
}}
|
|
|
|
primary_node_modules="$(select_primary_node_modules)"
|
|
|
|
runtime_workspace="$(mktemp -d)"
|
|
cleanup_runtime_workspace() {{
|
|
rm -rf "${{runtime_workspace}}"
|
|
}}
|
|
trap cleanup_runtime_workspace EXIT
|
|
|
|
runtime_package_dir="${{runtime_workspace}}/${{package_rel_dir}}"
|
|
mkdir -p "${{runtime_package_dir}}"
|
|
cp -RL "${{package_dir}}/." "${{runtime_package_dir}}/"
|
|
|
|
workspace_package_map="${{runtime_workspace}}/workspace-packages.tsv"
|
|
python3 - "${{runtime_package_dir}}" >"${{workspace_package_map}}" <<'PY'
|
|
import json
|
|
import os
|
|
import sys
|
|
|
|
root = sys.argv[1]
|
|
|
|
for dirpath, dirnames, filenames in os.walk(root):
|
|
dirnames[:] = [name for name in dirnames if name != "node_modules"]
|
|
if "package.json" not in filenames:
|
|
continue
|
|
|
|
manifest_path = os.path.join(dirpath, "package.json")
|
|
try:
|
|
with open(manifest_path, "r", encoding="utf-8") as manifest_file:
|
|
package_name = json.load(manifest_file).get("name")
|
|
except Exception:
|
|
continue
|
|
|
|
if isinstance(package_name, str):
|
|
print(f"{{package_name}}\t{{dirpath}}")
|
|
PY
|
|
|
|
install_repo_root=""
|
|
if [[ -n "${{primary_node_modules}}" ]]; then
|
|
install_repo_root="$(dirname "${{primary_node_modules}}")"
|
|
ln -s "${{primary_node_modules}}" "${{runtime_workspace}}/node_modules"
|
|
fi
|
|
|
|
workspace_package_dir_for_source() {{
|
|
local source="$1"
|
|
local manifest_path="${{source}}/package.json"
|
|
local package_name=""
|
|
local workspace_dir=""
|
|
|
|
if [[ ! -f "${{manifest_path}}" ]]; then
|
|
return 1
|
|
fi
|
|
|
|
package_name="$(python3 - "${{manifest_path}}" <<'PY'
|
|
import json
|
|
import sys
|
|
|
|
try:
|
|
with open(sys.argv[1], "r", encoding="utf-8") as manifest_file:
|
|
package_name = json.load(manifest_file).get("name", "")
|
|
except Exception:
|
|
package_name = ""
|
|
|
|
if isinstance(package_name, str):
|
|
print(package_name)
|
|
PY
|
|
)"
|
|
|
|
workspace_dir="$(awk -F '\t' -v name="$package_name" '$1 == name {{ print $2; exit }}' "${{workspace_package_map}}")"
|
|
if [[ -n "${{package_name}}" && -n "${{workspace_dir}}" ]]; then
|
|
echo "${{workspace_dir}}"
|
|
return 0
|
|
fi
|
|
|
|
return 1
|
|
}}
|
|
|
|
link_node_modules_entry() {{
|
|
local source="$1"
|
|
local destination="$2"
|
|
local workspace_target=""
|
|
|
|
rm -rf "${{destination}}"
|
|
workspace_target="$(workspace_package_dir_for_source "${{source}}" || true)"
|
|
if [[ -n "${{workspace_target}}" ]]; then
|
|
ln -s "${{workspace_target}}" "${{destination}}"
|
|
return 0
|
|
fi
|
|
|
|
if [[ -L "${{source}}" ]]; then
|
|
ln -s "$(readlink "${{source}}")" "${{destination}}"
|
|
else
|
|
ln -s "${{source}}" "${{destination}}"
|
|
fi
|
|
}}
|
|
|
|
mirror_node_modules_dir() {{
|
|
local source_dir="$1"
|
|
local destination_dir="$2"
|
|
local entry=""
|
|
local scoped_entry=""
|
|
|
|
rm -rf "${{destination_dir}}"
|
|
mkdir -p "${{destination_dir}}"
|
|
|
|
shopt -s dotglob nullglob
|
|
for entry in "${{source_dir}}"/* "${{source_dir}}"/.[!.]* "${{source_dir}}"/..?*; do
|
|
local entry_name="$(basename "${{entry}}")"
|
|
if [[ "${{entry_name}}" == "." || "${{entry_name}}" == ".." ]]; then
|
|
continue
|
|
fi
|
|
|
|
if [[ -d "${{entry}}" && ! -L "${{entry}}" && "${{entry_name}}" == @* ]]; then
|
|
mkdir -p "${{destination_dir}}/${{entry_name}}"
|
|
for scoped_entry in "${{entry}}"/* "${{entry}}"/.[!.]* "${{entry}}"/..?*; do
|
|
local scoped_name="$(basename "${{scoped_entry}}")"
|
|
if [[ "${{scoped_name}}" == "." || "${{scoped_name}}" == ".." ]]; then
|
|
continue
|
|
fi
|
|
|
|
link_node_modules_entry "${{scoped_entry}}" "${{destination_dir}}/${{entry_name}}/${{scoped_name}}"
|
|
done
|
|
continue
|
|
fi
|
|
|
|
link_node_modules_entry "${{entry}}" "${{destination_dir}}/${{entry_name}}"
|
|
done
|
|
shopt -u dotglob nullglob
|
|
}}
|
|
|
|
find_node_modules() {{
|
|
local dir="$1"
|
|
local root="$2"
|
|
|
|
while [[ "$dir" == "$root"* ]]; do
|
|
if [[ -d "$dir/node_modules" ]]; then
|
|
echo "$dir/node_modules"
|
|
return 0
|
|
fi
|
|
|
|
if [[ "$dir" == "$root" ]]; then
|
|
break
|
|
fi
|
|
|
|
dir="$(dirname "$dir")"
|
|
done
|
|
|
|
return 1
|
|
}}
|
|
|
|
find_install_repo_node_modules() {{
|
|
local repo_root="$1"
|
|
local rel_dir="$2"
|
|
local candidate="${{rel_dir}}"
|
|
|
|
while [[ -n "${{candidate}}" ]]; do
|
|
if [[ -d "${{repo_root}}/${{candidate}}/node_modules" ]]; then
|
|
echo "${{repo_root}}/${{candidate}}/node_modules"
|
|
return 0
|
|
fi
|
|
|
|
if [[ "${{candidate}}" != */* ]]; then
|
|
break
|
|
fi
|
|
|
|
candidate="${{candidate#*/}}"
|
|
done
|
|
|
|
if [[ -d "${{repo_root}}/node_modules" ]]; then
|
|
echo "${{repo_root}}/node_modules"
|
|
return 0
|
|
fi
|
|
|
|
return 1
|
|
}}
|
|
|
|
mirror_install_repo_workspace_node_modules() {{
|
|
local repo_root="$1"
|
|
local destination_root="$2"
|
|
|
|
while IFS= read -r install_node_modules; do
|
|
local rel_path="${{install_node_modules#${{repo_root}}/}}"
|
|
local destination="${{destination_root}}/${{rel_path}}"
|
|
|
|
mkdir -p "$(dirname "${{destination}}")"
|
|
mirror_node_modules_dir "${{install_node_modules}}" "${{destination}}"
|
|
done < <(find "${{repo_root}}" \
|
|
-path "${{repo_root}}/node_modules" -prune -o \
|
|
-type d -name node_modules -print 2>/dev/null | sort)
|
|
}}
|
|
|
|
resolved_install_node_modules=""
|
|
if [[ -n "${{install_repo_root}}" ]]; then
|
|
resolved_install_node_modules="$(find_install_repo_node_modules "${{install_repo_root}}" "${{package_rel_dir}}" || true)"
|
|
fi
|
|
|
|
if [[ -n "${{resolved_install_node_modules}}" ]]; then
|
|
mirror_node_modules_dir "${{resolved_install_node_modules}}" "${{runtime_package_dir}}/node_modules"
|
|
else
|
|
resolved_node_modules="$(find_node_modules "${{runtime_package_dir}}" "${{runtime_workspace}}" || true)"
|
|
if [[ -n "${{resolved_node_modules}}" && "${{resolved_node_modules}}" != "${{runtime_package_dir}}/node_modules" ]]; then
|
|
mirror_node_modules_dir "${{resolved_node_modules}}" "${{runtime_package_dir}}/node_modules"
|
|
fi
|
|
fi
|
|
|
|
if [[ -n "${{install_repo_root}}" ]]; then
|
|
mirror_install_repo_workspace_node_modules "${{install_repo_root}}" "${{runtime_package_dir}}"
|
|
fi
|
|
|
|
path_entries=()
|
|
if [[ -d "${{runtime_package_dir}}/node_modules/.bin" ]]; then
|
|
path_entries+=("${{runtime_package_dir}}/node_modules/.bin")
|
|
fi
|
|
|
|
if [[ -d "${{runtime_workspace}}/node_modules/.bin" && "${{runtime_workspace}}/node_modules/.bin" != "${{runtime_package_dir}}/node_modules/.bin" ]]; then
|
|
path_entries+=("${{runtime_workspace}}/node_modules/.bin")
|
|
fi
|
|
|
|
if [[ ${{#path_entries[@]}} -gt 0 ]]; then
|
|
export PATH="$(IFS=:; echo "${{path_entries[*]}}"):${{PATH}}"
|
|
fi
|
|
|
|
working_dir="{working_dir}"
|
|
if [[ "${{working_dir}}" == "package" ]]; then
|
|
cd "${{runtime_package_dir}}"
|
|
else
|
|
cd "${{runtime_workspace}}"
|
|
fi
|
|
|
|
exec "${{bun_bin}}" --bun run {script} "$@"
|
|
""".format(
|
|
bun_short_path = bun_bin.short_path,
|
|
package_json_short_path = package_json.short_path,
|
|
package_rel_dir = package_json.dirname,
|
|
working_dir = ctx.attr.working_dir,
|
|
script = _shell_quote(ctx.attr.script),
|
|
),
|
|
)
|
|
|
|
transitive_files = []
|
|
if ctx.attr.node_modules:
|
|
transitive_files.append(ctx.attr.node_modules[DefaultInfo].files)
|
|
|
|
runfiles = ctx.runfiles(
|
|
files = [bun_bin, package_json] + ctx.files.data,
|
|
transitive_files = depset(transitive = transitive_files),
|
|
)
|
|
|
|
return [
|
|
DefaultInfo(
|
|
executable = launcher,
|
|
runfiles = runfiles,
|
|
),
|
|
]
|
|
|
|
|
|
bun_script = rule(
|
|
implementation = _bun_script_impl,
|
|
doc = """Runs a named `package.json` script with Bun as an executable target.
|
|
|
|
Use this rule to expose existing package scripts such as `dev`, `build`, or
|
|
`check` via `bazel run` without adding wrapper shell scripts. This is a good fit
|
|
for Vite-style workflows, where scripts like `vite dev` or `vite build` are
|
|
declared in `package.json` and expect to run from the package directory with
|
|
`node_modules/.bin` available on `PATH`.
|
|
""",
|
|
attrs = {
|
|
"script": attr.string(
|
|
mandatory = True,
|
|
doc = "Name of the `package.json` script to execute via `bun run <script>`.",
|
|
),
|
|
"package_json": attr.label(
|
|
mandatory = True,
|
|
allow_single_file = True,
|
|
doc = "Label of the `package.json` file containing the named script.",
|
|
),
|
|
"node_modules": attr.label(
|
|
doc = "Optional label providing package files from a `node_modules` tree, typically produced by `bun_install`, in runfiles. Executables from `node_modules/.bin` are added to `PATH`, which is useful for scripts such as `vite`.",
|
|
),
|
|
"data": attr.label_list(
|
|
allow_files = True,
|
|
doc = "Additional runtime files required by the script.",
|
|
),
|
|
"working_dir": attr.string(
|
|
default = "package",
|
|
values = ["workspace", "package"],
|
|
doc = "Working directory at runtime: Bazel runfiles `workspace` root or the directory containing `package.json`. The default `package` mode matches tools such as Vite that resolve config and assets relative to the package directory.",
|
|
),
|
|
},
|
|
executable = True,
|
|
toolchains = ["//bun:toolchain_type"],
|
|
)
|