又用 vscode copilot 重写 startb

至少这次可读性还不错。
This commit is contained in:
2026-03-28 04:23:22 +08:00
parent e8e33898be
commit 10d03bc051
3 changed files with 92 additions and 33 deletions

View File

@@ -1,3 +1,3 @@
#!/bin/bash #!/bin/bash
option=$(fuzzel --dmenu --prompt-only="Run command: ") option=$(fuzzel --dmenu --prompt-only="Run command: ")
[ -z "$option" ] || startb $option [ -z "$option" ] || exec startb $option

View File

@@ -15,4 +15,4 @@ for item in $(ls $desktop); do
done done
fsarg=$(echo -en "$options" | fuzzel --dmenu --prompt="Open with VSCode: Desktop/") fsarg=$(echo -en "$options" | fuzzel --dmenu --prompt="Open with VSCode: Desktop/")
[ -z "$fsarg" ] || startb code "$desktop/$fsarg" [ -z "$fsarg" ] || exec startb code "$desktop/$fsarg"

View File

@@ -1,40 +1,99 @@
#!/bin/bash #!/usr/bin/env bash
# like `start /b` in Windows CMD. # startb: start a command in the background with journald or file logging fallback.
# Usage:
# startb [--log logfile] COMMAND [ARGS...]
# Environment:
# STARTB_LOGFILE=... to force a log file.
# PATH 不完整问题: set -euo pipefail
# 根据 niri 启动方式的不同niri 读到的环境变量可能也不一样。
# - tty 进终端再 niri-session 拉起来大概是最完整的。毕竟已经加载 .xxxrc 了嘛。
# - 而像我这样用 sddm 进的 niri由于直接从 systemd 启动,中间也没有任何机会更新 PATH相比就最麻烦了。
# 根据谷鸽 AI 的回答,我这种情况只能在 ~/.config/environment.d/ 里静态补上环境变量。
# 之前不懂的时候用的是如下的奇技淫巧:
# PATH=$(zsh -c -i 'echo $PATH') # PATH may be incomplete from systemd login env; restore from interactive shell.
# export PATH PATH=$(zsh -c -i 'echo $PATH')
export PATH
# https://vescrity.github.io/post/systemd-desktop-suspend/ SCRIPT_NAME="${0##*/}"
#systemd-run --user --scope --slice=YukiLauncher.slice --unit="$1-$$".scope /bin/sh -c '"$@"' _ "$@"
usage() {
cat <<EOF
Usage: ${SCRIPT_NAME} [--log logfile] COMMAND [ARGS...]
--log logfile Write logs to this file instead of journald.
Environment variable STARTB_LOGFILE can also provide a logfile path.
EOF
exit 1
}
if [ $# -lt 1 ]; then if [ $# -lt 1 ]; then
echo "Usage: daemon-run COMMAND [ARGS...]" usage
exit 1
fi fi
UNIT_NAME="${1##*/}-$(date +%s%N)" LOG_FILE=${STARTB_LOGFILE:-}
ENV_ARGS=(
--setenv=PATH="$PATH"
--setenv=NVM_DIR="$NVM_DIR"
--setenv=NVM_BIN="$NVM_BIN"
)
systemd-run --user \ if [ "$1" = "--log" ] || [ "$1" = "-l" ]; then
"${ENV_ARGS[@]}" \ shift
--working-directory="$PWD" \ if [ $# -lt 1 ]; then
--slice=StartProcess.slice \ echo "Error: --log requires a logfile path." >&2
--unit="$UNIT_NAME" \ usage
--collect \ fi
-- /bin/sh -c 'exec "$@"' _ "$@" LOG_FILE=$1
shift
if [ $? -eq 0 ]; then
echo "Logs : journalctl --user -u $UNIT_NAME -f"
echo "Note : '$UNIT_NAME' will auto-destruct on exit."
fi fi
if [ $# -lt 1 ]; then
usage
fi
COMMAND=("$@")
UNIT_BASE=$(basename "${COMMAND[0]}")
UNIT_BASE=${UNIT_BASE:-cmd}
UNIT_NAME="${UNIT_BASE}-$(date +%Y%m%d_%H%M%S_%N)"
log_to_file() {
local path=$1
mkdir -p "$(dirname "$path")" 2>/dev/null || true
touch "$path" || {
echo "Failed to touch logfile: $path" >&2
return 1
}
nohup setsid "${COMMAND[@]}" >>"$path" 2>&1 &
local pid=$!
disown "$pid" 2>/dev/null || true
echo "$UNIT_NAME started in background (pid=$pid), logs: $path"
return 0
}
log_to_journal() {
# 不按雪叶那样写基本记录不到输出,懒得调了,直接屏蔽。
# if command -v systemd-run >/dev/null 2>&1; then
# systemd-run --user --scope --unit="$UNIT_NAME" -- "${COMMAND[@]}" >/dev/null 2>&1 &
# local pid=$!
# echo "$UNIT_NAME started in systemd scope (pid=$pid), journald logging enabled"
# return 0
# fi
if command -v systemd-cat >/dev/null 2>&1; then
nohup setsid systemd-cat -t "$UNIT_NAME" -- "${COMMAND[@]}" >/dev/null 2>&1 &
local pid=$!
disown "$pid" 2>/dev/null || true
echo "$UNIT_NAME started with systemd-cat (pid=$pid), journald logging enabled"
return 0
fi
return 1
}
if [ -n "$LOG_FILE" ]; then
log_to_file "$LOG_FILE" || exit 1
exit 0
fi
if log_to_journal; then
exit 0
fi
# Fallback log path when journald is unavailable.
LOG_FILE="/tmp/${USER:-unknown}/startb/${UNIT_NAME}.log"
log_to_file "$LOG_FILE"