mirror of
https://github.com/obra/superpowers.git
synced 2026-04-26 03:29:06 +08:00
rewrites sync tool to clone the fork, open a PR, and regenerate overlays inline
The previous version was a local rsync helper that required a hand-maintained destination path. This rewrite makes it path/user-agnostic and gives every team member the same flow: - Clones prime-radiant-inc/openai-codex-plugins fresh into a temp dir per run (trap EXIT cleans up) - Auto-detects upstream from the script's own location - Preflight: rsync, git, gh auth, python3, upstream package.json - Reads upstream version from package.json and bakes it into the regenerated .codex-plugin/plugin.json, so version bumps flow through - Regenerates both overlay files (.codex-plugin/plugin.json and agents/openai.yaml) inline via heredoc — single source of truth - Pushes a sync/superpowers-<sha>-<UTC-timestamp> branch and opens a PR via gh pr create; prints PR URL and /files diff URL on completion - --dry-run, --yes, --base BRANCH, --local PATH flags for all the usual modes - Deterministic: two runs against the same upstream SHA produce PRs with identical diffs, so the tool itself can be sanity-checked by running twice Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -2,27 +2,37 @@
|
|||||||
#
|
#
|
||||||
# sync-to-codex-plugin.sh
|
# sync-to-codex-plugin.sh
|
||||||
#
|
#
|
||||||
# Syncs this superpowers checkout into a Codex plugin mirror directory.
|
# Sync this superpowers checkout → prime-radiant-inc/openai-codex-plugins.
|
||||||
# Pulls every file except the EXCLUDES list, never touches the PROTECTS list.
|
# Clones the fork fresh into a temp dir, rsyncs upstream content, regenerates
|
||||||
# Leaves changes unstaged in the destination so a human can review before committing.
|
# the Codex overlay files (.codex-plugin/plugin.json + agents/openai.yaml)
|
||||||
|
# inline, commits, pushes a sync branch, and opens a PR.
|
||||||
|
# Path/user agnostic — auto-detects upstream from script location.
|
||||||
|
#
|
||||||
|
# Deterministic: running twice against the same upstream SHA produces PRs with
|
||||||
|
# identical diffs, so two back-to-back runs can verify the tool itself.
|
||||||
#
|
#
|
||||||
# Usage:
|
# Usage:
|
||||||
# ./scripts/sync-to-codex-plugin.sh # sync with confirmation
|
# ./scripts/sync-to-codex-plugin.sh # full run with confirm
|
||||||
# ./scripts/sync-to-codex-plugin.sh -n # dry run, show changes only
|
# ./scripts/sync-to-codex-plugin.sh -n # dry run, no clone/push/PR
|
||||||
# ./scripts/sync-to-codex-plugin.sh -y # skip confirmation prompt
|
# ./scripts/sync-to-codex-plugin.sh -y # skip confirmation
|
||||||
# ./scripts/sync-to-codex-plugin.sh --dest /path/to/plugins/superpowers
|
# ./scripts/sync-to-codex-plugin.sh --local PATH # use existing checkout
|
||||||
|
# ./scripts/sync-to-codex-plugin.sh --base BRANCH # target branch (default: main)
|
||||||
#
|
#
|
||||||
# Environment:
|
# Requires: bash, rsync, git, gh (authenticated), python3.
|
||||||
# CODEX_PLUGIN_DEST Destination plugin path (default: sibling openai-codex-plugins checkout)
|
|
||||||
|
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Config — edit these lists as the upstream or canonical shape evolves
|
# Config — edit as upstream or canonical plugin shape evolves
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
|
FORK="prime-radiant-inc/openai-codex-plugins"
|
||||||
|
DEFAULT_BASE="main"
|
||||||
|
DEST_REL="plugins/superpowers"
|
||||||
|
|
||||||
# Paths in upstream that should NOT land in the embedded plugin.
|
# Paths in upstream that should NOT land in the embedded plugin.
|
||||||
# Rsync --exclude patterns (trailing slash = directory).
|
# Both the Codex-overlay files are here too — they're managed by the
|
||||||
|
# generate-overlays step, not by rsync.
|
||||||
EXCLUDES=(
|
EXCLUDES=(
|
||||||
# Dotfiles and infra
|
# Dotfiles and infra
|
||||||
".claude/"
|
".claude/"
|
||||||
@@ -38,7 +48,7 @@ EXCLUDES=(
|
|||||||
".worktrees/"
|
".worktrees/"
|
||||||
".DS_Store"
|
".DS_Store"
|
||||||
|
|
||||||
# Root ceremony files (not part of a canonical Codex plugin)
|
# Root ceremony files
|
||||||
"AGENTS.md"
|
"AGENTS.md"
|
||||||
"CHANGELOG.md"
|
"CHANGELOG.md"
|
||||||
"CLAUDE.md"
|
"CLAUDE.md"
|
||||||
@@ -56,33 +66,93 @@ EXCLUDES=(
|
|||||||
"scripts/"
|
"scripts/"
|
||||||
"tests/"
|
"tests/"
|
||||||
"tmp/"
|
"tmp/"
|
||||||
)
|
|
||||||
|
|
||||||
# Paths in the destination that are hand-authored Codex overlays.
|
# Codex-overlay files — regenerated below, not synced
|
||||||
# Rsync will never touch these — including when --delete would otherwise
|
|
||||||
# remove them because they don't exist in upstream.
|
|
||||||
PROTECTS=(
|
|
||||||
".codex-plugin/"
|
".codex-plugin/"
|
||||||
"agents/openai.yaml"
|
"agents/openai.yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Paths
|
# Generated overlay files
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
# Writes the Codex plugin manifest to "$1" with the given upstream version.
|
||||||
UPSTREAM="$(cd "$SCRIPT_DIR/.." && pwd)"
|
# Args: dest_path, version
|
||||||
|
generate_plugin_json() {
|
||||||
|
local dest="$1"
|
||||||
|
local version="$2"
|
||||||
|
mkdir -p "$(dirname "$dest")"
|
||||||
|
cat > "$dest" <<EOF
|
||||||
|
{
|
||||||
|
"name": "superpowers",
|
||||||
|
"version": "$version",
|
||||||
|
"description": "Core skills library for Codex: planning, TDD, debugging, and collaboration workflows.",
|
||||||
|
"author": {
|
||||||
|
"name": "Jesse Vincent",
|
||||||
|
"email": "jesse@fsck.com",
|
||||||
|
"url": "https://github.com/obra"
|
||||||
|
},
|
||||||
|
"homepage": "https://github.com/obra/superpowers",
|
||||||
|
"repository": "https://github.com/obra/superpowers",
|
||||||
|
"license": "MIT",
|
||||||
|
"keywords": [
|
||||||
|
"skills",
|
||||||
|
"planning",
|
||||||
|
"tdd",
|
||||||
|
"debugging",
|
||||||
|
"code-review",
|
||||||
|
"workflow"
|
||||||
|
],
|
||||||
|
"skills": "./skills/",
|
||||||
|
"interface": {
|
||||||
|
"displayName": "Superpowers",
|
||||||
|
"shortDescription": "Planning, TDD, debugging, and delivery workflows for coding agents",
|
||||||
|
"longDescription": "Use Superpowers to guide agent work through brainstorming, implementation planning, test-driven development, systematic debugging, parallel execution, code review, and finish-the-branch workflows adapted for Codex.",
|
||||||
|
"developerName": "Jesse Vincent",
|
||||||
|
"category": "Coding",
|
||||||
|
"capabilities": [
|
||||||
|
"Interactive",
|
||||||
|
"Read",
|
||||||
|
"Write"
|
||||||
|
],
|
||||||
|
"websiteURL": "https://github.com/obra/superpowers",
|
||||||
|
"privacyPolicyURL": "https://docs.github.com/site-policy/privacy-policies/github-general-privacy-statement",
|
||||||
|
"termsOfServiceURL": "https://docs.github.com/en/site-policy/github-terms/github-terms-of-service",
|
||||||
|
"defaultPrompt": [
|
||||||
|
"Use Superpowers to plan this feature before we code",
|
||||||
|
"Debug this bug with a systematic root-cause workflow",
|
||||||
|
"Turn this approved design into an implementation plan"
|
||||||
|
],
|
||||||
|
"brandColor": "#F59E0B",
|
||||||
|
"screenshots": []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
|
||||||
# Default dest: sibling openai-codex-plugins checkout, if it exists
|
# Writes the plugin-level agents/openai.yaml to "$1".
|
||||||
DEFAULT_DEST="${CODEX_PLUGIN_DEST:-$(dirname "$UPSTREAM")/openai-codex-plugins/plugins/superpowers}"
|
# Args: dest_path
|
||||||
|
generate_agents_openai_yaml() {
|
||||||
|
local dest="$1"
|
||||||
|
mkdir -p "$(dirname "$dest")"
|
||||||
|
cat > "$dest" <<'EOF'
|
||||||
|
interface:
|
||||||
|
display_name: "Superpowers"
|
||||||
|
short_description: "Planning, TDD, debugging, and delivery workflows for coding agents"
|
||||||
|
default_prompt: "Use Superpowers to brainstorm a design, write an implementation plan, run test-driven development, debug bugs systematically, or finish and ship a development branch."
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Args
|
# Args
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
DEST="$DEFAULT_DEST"
|
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
||||||
|
UPSTREAM="$(cd "$SCRIPT_DIR/.." && pwd)"
|
||||||
|
BASE="$DEFAULT_BASE"
|
||||||
DRY_RUN=0
|
DRY_RUN=0
|
||||||
YES=0
|
YES=0
|
||||||
|
LOCAL_CHECKOUT=""
|
||||||
|
|
||||||
usage() {
|
usage() {
|
||||||
sed -n 's/^# \{0,1\}//;2,20p' "$0"
|
sed -n 's/^# \{0,1\}//;2,20p' "$0"
|
||||||
@@ -91,84 +161,122 @@ usage() {
|
|||||||
|
|
||||||
while [[ $# -gt 0 ]]; do
|
while [[ $# -gt 0 ]]; do
|
||||||
case "$1" in
|
case "$1" in
|
||||||
--dest) DEST="$2"; shift 2 ;;
|
|
||||||
-n|--dry-run) DRY_RUN=1; shift ;;
|
-n|--dry-run) DRY_RUN=1; shift ;;
|
||||||
-y|--yes) YES=1; shift ;;
|
-y|--yes) YES=1; shift ;;
|
||||||
|
--local) LOCAL_CHECKOUT="$2"; shift 2 ;;
|
||||||
|
--base) BASE="$2"; shift 2 ;;
|
||||||
-h|--help) usage 0 ;;
|
-h|--help) usage 0 ;;
|
||||||
*) echo "Unknown arg: $1" >&2; usage 2 ;;
|
*) echo "Unknown arg: $1" >&2; usage 2 ;;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Validate environment
|
# Preflight
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
if [[ ! -d "$UPSTREAM/.git" ]]; then
|
die() { echo "ERROR: $*" >&2; exit 1; }
|
||||||
echo "ERROR: Upstream '$UPSTREAM' is not a git checkout." >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -d "$DEST" ]]; then
|
command -v rsync >/dev/null || die "rsync not found in PATH"
|
||||||
echo "ERROR: Destination '$DEST' does not exist." >&2
|
command -v git >/dev/null || die "git not found in PATH"
|
||||||
echo "Set CODEX_PLUGIN_DEST or pass --dest <path>." >&2
|
command -v gh >/dev/null || die "gh not found — install GitHub CLI"
|
||||||
exit 1
|
command -v python3 >/dev/null || die "python3 not found in PATH"
|
||||||
fi
|
|
||||||
|
|
||||||
confirm() {
|
gh auth status >/dev/null 2>&1 || die "gh not authenticated — run 'gh auth login'"
|
||||||
local prompt="$1"
|
|
||||||
[[ $YES -eq 1 ]] && return 0
|
[[ -d "$UPSTREAM/.git" ]] || die "upstream '$UPSTREAM' is not a git checkout"
|
||||||
read -rp "$prompt [y/N] " ans
|
[[ -f "$UPSTREAM/package.json" ]] || die "upstream has no package.json — cannot read version"
|
||||||
[[ "$ans" == "y" || "$ans" == "Y" ]]
|
|
||||||
}
|
# Read the upstream version from package.json
|
||||||
|
UPSTREAM_VERSION="$(python3 -c 'import json,sys; print(json.load(open(sys.argv[1]))["version"])' "$UPSTREAM/package.json")"
|
||||||
|
[[ -n "$UPSTREAM_VERSION" ]] || die "could not read 'version' from upstream package.json"
|
||||||
|
|
||||||
# Check upstream branch
|
|
||||||
UPSTREAM_BRANCH="$(cd "$UPSTREAM" && git branch --show-current)"
|
UPSTREAM_BRANCH="$(cd "$UPSTREAM" && git branch --show-current)"
|
||||||
UPSTREAM_SHA="$(cd "$UPSTREAM" && git rev-parse HEAD)"
|
UPSTREAM_SHA="$(cd "$UPSTREAM" && git rev-parse HEAD)"
|
||||||
UPSTREAM_SHORT="$(cd "$UPSTREAM" && git rev-parse --short HEAD)"
|
UPSTREAM_SHORT="$(cd "$UPSTREAM" && git rev-parse --short HEAD)"
|
||||||
|
|
||||||
|
confirm() {
|
||||||
|
[[ $YES -eq 1 ]] && return 0
|
||||||
|
read -rp "$1 [y/N] " ans
|
||||||
|
[[ "$ans" == "y" || "$ans" == "Y" ]]
|
||||||
|
}
|
||||||
|
|
||||||
if [[ "$UPSTREAM_BRANCH" != "main" ]]; then
|
if [[ "$UPSTREAM_BRANCH" != "main" ]]; then
|
||||||
echo "WARNING: Upstream is on branch '$UPSTREAM_BRANCH', not 'main'."
|
echo "WARNING: upstream is on '$UPSTREAM_BRANCH', not 'main'"
|
||||||
confirm "Sync from '$UPSTREAM_BRANCH' anyway?" || exit 1
|
confirm "Sync from '$UPSTREAM_BRANCH' anyway?" || exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Check upstream working tree is clean
|
|
||||||
UPSTREAM_STATUS="$(cd "$UPSTREAM" && git status --porcelain)"
|
UPSTREAM_STATUS="$(cd "$UPSTREAM" && git status --porcelain)"
|
||||||
if [[ -n "$UPSTREAM_STATUS" ]]; then
|
if [[ -n "$UPSTREAM_STATUS" ]]; then
|
||||||
echo "WARNING: Upstream has uncommitted changes:"
|
echo "WARNING: upstream has uncommitted changes:"
|
||||||
echo "$UPSTREAM_STATUS" | sed 's/^/ /'
|
echo "$UPSTREAM_STATUS" | sed 's/^/ /'
|
||||||
echo "Sync will use the working-tree state, not HEAD ($UPSTREAM_SHORT)."
|
echo "Sync will use working-tree state, not HEAD ($UPSTREAM_SHORT)."
|
||||||
confirm "Continue anyway?" || exit 1
|
confirm "Continue anyway?" || exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Build rsync args
|
# Prepare destination (clone fork fresh, or use --local)
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
CLEANUP_DIR=""
|
||||||
|
cleanup() {
|
||||||
|
[[ -n "$CLEANUP_DIR" ]] && rm -rf "$CLEANUP_DIR"
|
||||||
|
}
|
||||||
|
trap cleanup EXIT
|
||||||
|
|
||||||
|
if [[ -n "$LOCAL_CHECKOUT" ]]; then
|
||||||
|
DEST_REPO="$(cd "$LOCAL_CHECKOUT" && pwd)"
|
||||||
|
[[ -d "$DEST_REPO/.git" ]] || die "--local path '$DEST_REPO' is not a git checkout"
|
||||||
|
else
|
||||||
|
echo "Cloning $FORK..."
|
||||||
|
CLEANUP_DIR="$(mktemp -d)"
|
||||||
|
DEST_REPO="$CLEANUP_DIR/openai-codex-plugins"
|
||||||
|
gh repo clone "$FORK" "$DEST_REPO" >/dev/null
|
||||||
|
fi
|
||||||
|
|
||||||
|
DEST="$DEST_REPO/$DEST_REL"
|
||||||
|
|
||||||
|
# Checkout base branch
|
||||||
|
cd "$DEST_REPO"
|
||||||
|
git checkout -q "$BASE" 2>/dev/null || die "base branch '$BASE' doesn't exist in $FORK"
|
||||||
|
|
||||||
|
[[ -d "$DEST" ]] || die "base branch '$BASE' has no '$DEST_REL/' — merge the bootstrap PR first, or pass --base <branch>"
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# Create sync branch
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
TIMESTAMP="$(date -u +%Y%m%d-%H%M%S)"
|
||||||
|
SYNC_BRANCH="sync/superpowers-${UPSTREAM_SHORT}-${TIMESTAMP}"
|
||||||
|
git checkout -q -b "$SYNC_BRANCH"
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# Build rsync args (excludes only — no protects, overlays are regenerated)
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
RSYNC_ARGS=(-av --delete)
|
RSYNC_ARGS=(-av --delete)
|
||||||
|
for pat in "${EXCLUDES[@]}"; do RSYNC_ARGS+=(--exclude="$pat"); done
|
||||||
for pat in "${EXCLUDES[@]}"; do
|
|
||||||
RSYNC_ARGS+=(--exclude="$pat")
|
|
||||||
done
|
|
||||||
|
|
||||||
for pat in "${PROTECTS[@]}"; do
|
|
||||||
RSYNC_ARGS+=(--filter="protect $pat")
|
|
||||||
done
|
|
||||||
|
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Dry run first, always
|
# Dry run preview (always shown)
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
echo ""
|
echo ""
|
||||||
echo "Upstream: $UPSTREAM ($UPSTREAM_BRANCH @ $UPSTREAM_SHORT)"
|
echo "Upstream: $UPSTREAM ($UPSTREAM_BRANCH @ $UPSTREAM_SHORT)"
|
||||||
echo "Dest: $DEST"
|
echo "Version: $UPSTREAM_VERSION"
|
||||||
|
echo "Fork: $FORK"
|
||||||
|
echo "Base: $BASE"
|
||||||
|
echo "Branch: $SYNC_BRANCH"
|
||||||
echo ""
|
echo ""
|
||||||
echo "=== Preview (rsync --dry-run) ==="
|
echo "=== Preview (rsync --dry-run) ==="
|
||||||
rsync "${RSYNC_ARGS[@]}" --dry-run --itemize-changes "$UPSTREAM/" "$DEST/"
|
rsync "${RSYNC_ARGS[@]}" --dry-run --itemize-changes "$UPSTREAM/" "$DEST/"
|
||||||
echo "=== End preview ==="
|
echo "=== End preview ==="
|
||||||
|
echo ""
|
||||||
|
echo "Overlay files (.codex-plugin/plugin.json, agents/openai.yaml) will be"
|
||||||
|
echo "regenerated with version $UPSTREAM_VERSION regardless of rsync output."
|
||||||
|
|
||||||
if [[ $DRY_RUN -eq 1 ]]; then
|
if [[ $DRY_RUN -eq 1 ]]; then
|
||||||
echo ""
|
echo ""
|
||||||
echo "Dry run only. Nothing was changed."
|
echo "Dry run only. Nothing was changed or pushed."
|
||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -177,36 +285,56 @@ fi
|
|||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
echo ""
|
echo ""
|
||||||
confirm "Apply these changes?" || { echo "Aborted."; exit 1; }
|
confirm "Apply changes, push branch, and open PR?" || { echo "Aborted."; exit 1; }
|
||||||
|
|
||||||
echo ""
|
echo ""
|
||||||
echo "Syncing..."
|
echo "Syncing upstream content..."
|
||||||
rsync "${RSYNC_ARGS[@]}" "$UPSTREAM/" "$DEST/"
|
rsync "${RSYNC_ARGS[@]}" "$UPSTREAM/" "$DEST/"
|
||||||
echo "Done."
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
# =============================================================================
|
echo "Regenerating overlay files..."
|
||||||
# Report
|
generate_plugin_json "$DEST/.codex-plugin/plugin.json" "$UPSTREAM_VERSION"
|
||||||
# =============================================================================
|
generate_agents_openai_yaml "$DEST/agents/openai.yaml"
|
||||||
|
|
||||||
DEST_GIT_ROOT="$(cd "$DEST" && git rev-parse --show-toplevel 2>/dev/null || echo "")"
|
# Bail early if nothing actually changed
|
||||||
if [[ -n "$DEST_GIT_ROOT" ]]; then
|
cd "$DEST_REPO"
|
||||||
DEST_REL="${DEST#$DEST_GIT_ROOT/}"
|
if [[ -z "$(git status --porcelain "$DEST_REL")" ]]; then
|
||||||
CHANGES="$(cd "$DEST_GIT_ROOT" && git status --porcelain "$DEST_REL")"
|
echo "No changes — embedded plugin was already in sync with upstream $UPSTREAM_SHORT (v$UPSTREAM_VERSION)."
|
||||||
if [[ -z "$CHANGES" ]]; then
|
exit 0
|
||||||
echo "No changes — destination was already in sync with upstream $UPSTREAM_SHORT."
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "Changes pending review:"
|
|
||||||
echo "$CHANGES" | sed 's/^/ /'
|
|
||||||
echo ""
|
|
||||||
echo "Upstream SHA: $UPSTREAM_SHA"
|
|
||||||
echo ""
|
|
||||||
echo "Suggested commit message:"
|
|
||||||
echo " sync superpowers from upstream main @ $UPSTREAM_SHORT"
|
|
||||||
echo ""
|
|
||||||
echo "Review with: git -C $DEST_GIT_ROOT diff -- $DEST_REL"
|
|
||||||
else
|
|
||||||
echo "Destination is not a git checkout — cannot report changes."
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# Commit, push, open PR
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
git add "$DEST_REL"
|
||||||
|
git commit --quiet -m "sync superpowers v$UPSTREAM_VERSION from upstream main @ $UPSTREAM_SHORT
|
||||||
|
|
||||||
|
Automated sync via scripts/sync-to-codex-plugin.sh
|
||||||
|
Upstream: https://github.com/obra/superpowers/commit/$UPSTREAM_SHA
|
||||||
|
Branch: $SYNC_BRANCH"
|
||||||
|
|
||||||
|
echo "Pushing $SYNC_BRANCH to $FORK..."
|
||||||
|
git push -u origin "$SYNC_BRANCH" --quiet
|
||||||
|
|
||||||
|
PR_TITLE="sync superpowers v$UPSTREAM_VERSION from upstream main @ $UPSTREAM_SHORT"
|
||||||
|
PR_BODY="Automated sync from superpowers upstream \`main\` @ \`$UPSTREAM_SHORT\` (v$UPSTREAM_VERSION).
|
||||||
|
|
||||||
|
Run via: \`scripts/sync-to-codex-plugin.sh\`
|
||||||
|
Upstream commit: https://github.com/obra/superpowers/commit/$UPSTREAM_SHA
|
||||||
|
|
||||||
|
Running the sync tool again against the same upstream SHA should produce a PR with an identical diff — use that to verify the tool is behaving."
|
||||||
|
|
||||||
|
echo "Opening PR..."
|
||||||
|
PR_URL="$(gh pr create \
|
||||||
|
--repo "$FORK" \
|
||||||
|
--base "$BASE" \
|
||||||
|
--head "$SYNC_BRANCH" \
|
||||||
|
--title "$PR_TITLE" \
|
||||||
|
--body "$PR_BODY")"
|
||||||
|
|
||||||
|
PR_NUM="${PR_URL##*/}"
|
||||||
|
DIFF_URL="https://github.com/$FORK/pull/$PR_NUM/files"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "PR opened: $PR_URL"
|
||||||
|
echo "Diff view: $DIFF_URL"
|
||||||
|
|||||||
Reference in New Issue
Block a user