187 lines
5.3 KiB
Python
187 lines
5.3 KiB
Python
"""Rule for running JS/TS scripts with Bun in watch mode for development."""
|
|
|
|
def _bun_dev_impl(ctx):
|
|
toolchain = ctx.toolchains["//bun:toolchain_type"]
|
|
bun_bin = toolchain.bun.bun_bin
|
|
entry_point = ctx.file.entry_point
|
|
|
|
restart_watch_paths = "\n".join([path.short_path for path in ctx.files.restart_on])
|
|
|
|
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"
|
|
bun_bin="${{runfiles_dir}}/_main/{bun_short_path}"
|
|
entry_point="${{runfiles_dir}}/_main/{entry_short_path}"
|
|
|
|
resolve_entrypoint_workdir() {{
|
|
local dir
|
|
dir="$(dirname "${{entry_point}}")"
|
|
while [[ "${{dir}}" == "${{workspace_root}}"* ]]; do
|
|
if [[ -f "${{dir}}/.env" || -f "${{dir}}/package.json" ]]; then
|
|
echo "${{dir}}"
|
|
return 0
|
|
fi
|
|
if [[ "${{dir}}" == "${{workspace_root}}" ]]; then
|
|
break
|
|
fi
|
|
dir="$(dirname "${{dir}}")"
|
|
done
|
|
echo "$(dirname "${{entry_point}}")"
|
|
}}
|
|
|
|
working_dir="{working_dir}"
|
|
if [[ "${{working_dir}}" == "entry_point" ]]; then
|
|
cd "$(resolve_entrypoint_workdir)"
|
|
else
|
|
cd "${{workspace_root}}"
|
|
fi
|
|
|
|
watch_mode="{watch_mode}"
|
|
if [[ "${{watch_mode}}" == "hot" ]]; then
|
|
dev_flag="--hot"
|
|
else
|
|
dev_flag="--watch"
|
|
fi
|
|
|
|
run_dev() {{
|
|
exec "${{bun_bin}}" --bun "${{dev_flag}}" run "${{entry_point}}" "$@"
|
|
}}
|
|
|
|
if [[ {restart_count} -eq 0 ]]; then
|
|
run_dev "$@"
|
|
fi
|
|
|
|
readarray -t restart_paths <<'EOF_RESTART_PATHS'
|
|
{restart_watch_paths}
|
|
EOF_RESTART_PATHS
|
|
|
|
file_mtime() {{
|
|
local p="$1"
|
|
if stat -f '%m' "${{p}}" >/dev/null 2>&1; then
|
|
stat -f '%m' "${{p}}"
|
|
return 0
|
|
fi
|
|
stat -c '%Y' "${{p}}"
|
|
}}
|
|
|
|
declare -A mtimes
|
|
for rel in "${{restart_paths[@]}}"; do
|
|
path="${{runfiles_dir}}/_main/${{rel}}"
|
|
if [[ -e "${{path}}" ]]; then
|
|
mtimes["${{rel}}"]="$(file_mtime "${{path}}")"
|
|
else
|
|
mtimes["${{rel}}"]="missing"
|
|
fi
|
|
done
|
|
|
|
child_pid=""
|
|
restart_child() {{
|
|
if [[ -n "${{child_pid}}" ]] && kill -0 "${{child_pid}}" 2>/dev/null; then
|
|
kill "${{child_pid}}"
|
|
wait "${{child_pid}}" || true
|
|
fi
|
|
"${{bun_bin}}" --bun "${{dev_flag}}" run "${{entry_point}}" "$@" &
|
|
child_pid=$!
|
|
}}
|
|
|
|
cleanup() {{
|
|
if [[ -n "${{child_pid}}" ]] && kill -0 "${{child_pid}}" 2>/dev/null; then
|
|
kill "${{child_pid}}"
|
|
wait "${{child_pid}}" || true
|
|
fi
|
|
}}
|
|
|
|
trap cleanup EXIT INT TERM
|
|
|
|
restart_child "$@"
|
|
|
|
while true; do
|
|
sleep 1
|
|
changed=0
|
|
for rel in "${{restart_paths[@]}}"; do
|
|
path="${{runfiles_dir}}/_main/${{rel}}"
|
|
if [[ -e "${{path}}" ]]; then
|
|
current="$(file_mtime "${{path}}")"
|
|
else
|
|
current="missing"
|
|
fi
|
|
if [[ "${{current}}" != "${{mtimes[${{rel}}]}}" ]]; then
|
|
mtimes["${{rel}}"]="${{current}}"
|
|
changed=1
|
|
fi
|
|
done
|
|
if [[ "${{changed}}" -eq 1 ]]; then
|
|
restart_child "$@"
|
|
fi
|
|
done
|
|
""".format(
|
|
bun_short_path = bun_bin.short_path,
|
|
entry_short_path = entry_point.short_path,
|
|
watch_mode = ctx.attr.watch_mode,
|
|
working_dir = ctx.attr.working_dir,
|
|
restart_count = len(ctx.files.restart_on),
|
|
restart_watch_paths = restart_watch_paths,
|
|
),
|
|
)
|
|
|
|
transitive_files = []
|
|
if ctx.attr.node_modules:
|
|
transitive_files.append(ctx.attr.node_modules[DefaultInfo].files)
|
|
|
|
runfiles = ctx.runfiles(
|
|
files = [bun_bin, entry_point] + ctx.files.data + ctx.files.restart_on,
|
|
transitive_files = depset(transitive = transitive_files),
|
|
)
|
|
|
|
return [
|
|
DefaultInfo(
|
|
executable = launcher,
|
|
runfiles = runfiles,
|
|
),
|
|
]
|
|
|
|
bun_dev = rule(
|
|
implementation = _bun_dev_impl,
|
|
doc = """Runs a JS/TS entry point in Bun development watch mode.
|
|
|
|
This rule is intended for local dev loops (`bazel run`) and supports Bun
|
|
watch/HMR plus optional full restarts on selected file changes.
|
|
""",
|
|
attrs = {
|
|
"entry_point": attr.label(
|
|
mandatory = True,
|
|
allow_single_file = [".js", ".ts", ".jsx", ".tsx", ".mjs", ".cjs"],
|
|
doc = "Path to the main JS/TS file to execute in dev mode.",
|
|
),
|
|
"watch_mode": attr.string(
|
|
default = "watch",
|
|
values = ["watch", "hot"],
|
|
doc = "Bun live-reload mode: `watch` (default) or `hot`.",
|
|
),
|
|
"restart_on": attr.label_list(
|
|
allow_files = True,
|
|
doc = "Files that trigger a full Bun process restart when they change.",
|
|
),
|
|
"node_modules": attr.label(
|
|
doc = "Optional label providing package files from a `node_modules` tree, typically produced by `bun_install`, in runfiles.",
|
|
),
|
|
"data": attr.label_list(
|
|
allow_files = True,
|
|
doc = "Additional runtime files required by the dev process.",
|
|
),
|
|
"working_dir": attr.string(
|
|
default = "workspace",
|
|
values = ["workspace", "entry_point"],
|
|
doc = "Working directory at runtime: `workspace` root or nearest `entry_point` ancestor containing `.env`/`package.json`.",
|
|
),
|
|
},
|
|
executable = True,
|
|
toolchains = ["//bun:toolchain_type"],
|
|
)
|