252 lines
5.6 KiB
Bash
252 lines
5.6 KiB
Bash
#!/usr/bin/env bash
|
|
|
|
if [ -z "${REPO_ROOT:-}" ]; then
|
|
REPO_ROOT="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)"
|
|
fi
|
|
|
|
release_info() {
|
|
echo "$@"
|
|
}
|
|
|
|
release_warn() {
|
|
echo "Warning: $*" >&2
|
|
}
|
|
|
|
release_fail() {
|
|
echo "Error: $*" >&2
|
|
exit 1
|
|
}
|
|
|
|
git_remote_exists() {
|
|
git -C "$REPO_ROOT" remote get-url "$1" >/dev/null 2>&1
|
|
}
|
|
|
|
github_repo_from_remote() {
|
|
local remote_url
|
|
|
|
remote_url="$(git -C "$REPO_ROOT" remote get-url "$1" 2>/dev/null || true)"
|
|
[ -n "$remote_url" ] || return 1
|
|
|
|
remote_url="${remote_url%.git}"
|
|
remote_url="${remote_url#ssh://}"
|
|
|
|
node - "$remote_url" <<'NODE'
|
|
const remoteUrl = process.argv[2];
|
|
|
|
const patterns = [
|
|
/^https?:\/\/github\.com\/([^/]+\/[^/]+)$/,
|
|
/^git@github\.com:([^/]+\/[^/]+)$/,
|
|
/^[^:]+:([^/]+\/[^/]+)$/
|
|
];
|
|
|
|
for (const pattern of patterns) {
|
|
const match = remoteUrl.match(pattern);
|
|
if (!match) continue;
|
|
process.stdout.write(match[1]);
|
|
process.exit(0);
|
|
}
|
|
|
|
process.exit(1);
|
|
NODE
|
|
}
|
|
|
|
resolve_release_remote() {
|
|
local remote="${RELEASE_REMOTE:-${PUBLISH_REMOTE:-}}"
|
|
|
|
if [ -n "$remote" ]; then
|
|
git_remote_exists "$remote" || release_fail "git remote '$remote' does not exist."
|
|
printf '%s\n' "$remote"
|
|
return
|
|
fi
|
|
|
|
if git_remote_exists public-gh; then
|
|
printf 'public-gh\n'
|
|
return
|
|
fi
|
|
|
|
if git_remote_exists origin; then
|
|
printf 'origin\n'
|
|
return
|
|
fi
|
|
|
|
release_fail "no git remote found. Configure RELEASE_REMOTE or PUBLISH_REMOTE."
|
|
}
|
|
|
|
fetch_release_remote() {
|
|
git -C "$REPO_ROOT" fetch "$1" --prune --tags
|
|
}
|
|
|
|
get_last_stable_tag() {
|
|
git -C "$REPO_ROOT" tag --list 'v*' --sort=-version:refname | head -1
|
|
}
|
|
|
|
get_current_stable_version() {
|
|
local tag
|
|
tag="$(get_last_stable_tag)"
|
|
if [ -z "$tag" ]; then
|
|
printf '0.0.0\n'
|
|
else
|
|
printf '%s\n' "${tag#v}"
|
|
fi
|
|
}
|
|
|
|
compute_bumped_version() {
|
|
node - "$1" "$2" <<'NODE'
|
|
const current = process.argv[2];
|
|
const bump = process.argv[3];
|
|
const match = current.match(/^(\d+)\.(\d+)\.(\d+)$/);
|
|
|
|
if (!match) {
|
|
throw new Error(`invalid semver version: ${current}`);
|
|
}
|
|
|
|
let [major, minor, patch] = match.slice(1).map(Number);
|
|
|
|
if (bump === 'patch') {
|
|
patch += 1;
|
|
} else if (bump === 'minor') {
|
|
minor += 1;
|
|
patch = 0;
|
|
} else if (bump === 'major') {
|
|
major += 1;
|
|
minor = 0;
|
|
patch = 0;
|
|
} else {
|
|
throw new Error(`unsupported bump type: ${bump}`);
|
|
}
|
|
|
|
process.stdout.write(`${major}.${minor}.${patch}`);
|
|
NODE
|
|
}
|
|
|
|
next_canary_version() {
|
|
local stable_version="$1"
|
|
local versions_json
|
|
|
|
versions_json="$(npm view paperclipai versions --json 2>/dev/null || echo '[]')"
|
|
|
|
node - "$stable_version" "$versions_json" <<'NODE'
|
|
const stable = process.argv[2];
|
|
const versionsArg = process.argv[3];
|
|
|
|
let versions = [];
|
|
try {
|
|
const parsed = JSON.parse(versionsArg);
|
|
versions = Array.isArray(parsed) ? parsed : [parsed];
|
|
} catch {
|
|
versions = [];
|
|
}
|
|
|
|
const pattern = new RegExp(`^${stable.replace(/\./g, '\\.')}-canary\\.(\\d+)$`);
|
|
let max = -1;
|
|
|
|
for (const version of versions) {
|
|
const match = version.match(pattern);
|
|
if (!match) continue;
|
|
max = Math.max(max, Number(match[1]));
|
|
}
|
|
|
|
process.stdout.write(`${stable}-canary.${max + 1}`);
|
|
NODE
|
|
}
|
|
|
|
release_branch_name() {
|
|
printf 'release/%s\n' "$1"
|
|
}
|
|
|
|
release_notes_file() {
|
|
printf '%s/releases/v%s.md\n' "$REPO_ROOT" "$1"
|
|
}
|
|
|
|
default_release_worktree_path() {
|
|
local version="$1"
|
|
local parent_dir
|
|
local repo_name
|
|
|
|
parent_dir="$(cd "$REPO_ROOT/.." && pwd)"
|
|
repo_name="$(basename "$REPO_ROOT")"
|
|
printf '%s/%s-release-%s\n' "$parent_dir" "$repo_name" "$version"
|
|
}
|
|
|
|
git_current_branch() {
|
|
git -C "$REPO_ROOT" symbolic-ref --quiet --short HEAD 2>/dev/null || true
|
|
}
|
|
|
|
git_local_branch_exists() {
|
|
git -C "$REPO_ROOT" show-ref --verify --quiet "refs/heads/$1"
|
|
}
|
|
|
|
git_remote_branch_exists() {
|
|
git -C "$REPO_ROOT" ls-remote --exit-code --heads "$2" "refs/heads/$1" >/dev/null 2>&1
|
|
}
|
|
|
|
git_local_tag_exists() {
|
|
git -C "$REPO_ROOT" show-ref --verify --quiet "refs/tags/$1"
|
|
}
|
|
|
|
git_remote_tag_exists() {
|
|
git -C "$REPO_ROOT" ls-remote --exit-code --tags "$2" "refs/tags/$1" >/dev/null 2>&1
|
|
}
|
|
|
|
npm_version_exists() {
|
|
local version="$1"
|
|
local resolved
|
|
|
|
resolved="$(npm view "paperclipai@${version}" version 2>/dev/null || true)"
|
|
[ "$resolved" = "$version" ]
|
|
}
|
|
|
|
require_clean_worktree() {
|
|
if [ -n "$(git -C "$REPO_ROOT" status --porcelain)" ]; then
|
|
release_fail "working tree is not clean. Commit, stash, or remove changes before releasing."
|
|
fi
|
|
}
|
|
|
|
git_worktree_path_for_branch() {
|
|
local branch_ref="refs/heads/$1"
|
|
|
|
git -C "$REPO_ROOT" worktree list --porcelain | awk -v branch_ref="$branch_ref" '
|
|
$1 == "worktree" { path = substr($0, 10) }
|
|
$1 == "branch" && $2 == branch_ref { print path; exit }
|
|
'
|
|
}
|
|
|
|
path_is_worktree_for_branch() {
|
|
local path="$1"
|
|
local branch="$2"
|
|
local current_branch
|
|
|
|
[ -d "$path" ] || return 1
|
|
current_branch="$(git -C "$path" symbolic-ref --quiet --short HEAD 2>/dev/null || true)"
|
|
[ "$current_branch" = "$branch" ]
|
|
}
|
|
|
|
ensure_release_branch_for_version() {
|
|
local stable_version="$1"
|
|
local current_branch
|
|
local expected_branch
|
|
|
|
current_branch="$(git_current_branch)"
|
|
expected_branch="$(release_branch_name "$stable_version")"
|
|
|
|
if [ -z "$current_branch" ]; then
|
|
release_fail "release work must run from branch $expected_branch, but HEAD is detached."
|
|
fi
|
|
|
|
if [ "$current_branch" != "$expected_branch" ]; then
|
|
release_fail "release work must run from branch $expected_branch, but current branch is $current_branch."
|
|
fi
|
|
}
|
|
|
|
stable_release_exists_anywhere() {
|
|
local stable_version="$1"
|
|
local remote="$2"
|
|
local tag="v$stable_version"
|
|
|
|
git_local_tag_exists "$tag" || git_remote_tag_exists "$tag" "$remote" || npm_version_exists "$stable_version"
|
|
}
|
|
|
|
release_train_is_frozen() {
|
|
stable_release_exists_anywhere "$1" "$2"
|
|
}
|