feat: publish gitea issue devops skill with docs and workflow templates
This commit is contained in:
164
.gitea/workflows/issue-branch-preview.yml
Normal file
164
.gitea/workflows/issue-branch-preview.yml
Normal file
@@ -0,0 +1,164 @@
|
||||
name: issue-branch-preview
|
||||
|
||||
on:
|
||||
push:
|
||||
branches-ignore:
|
||||
- main
|
||||
workflow_dispatch:
|
||||
inputs:
|
||||
branch:
|
||||
description: "Target branch (optional, default current ref)"
|
||||
required: false
|
||||
type: string
|
||||
issue:
|
||||
description: "Issue number (optional, auto-parse from branch)"
|
||||
required: false
|
||||
type: string
|
||||
|
||||
jobs:
|
||||
allocate-and-deploy:
|
||||
runs-on: ubuntu-latest
|
||||
env:
|
||||
PREVIEW_SLOTS: ${{ vars.PREVIEW_SLOTS }}
|
||||
PREVIEW_URL_TEMPLATE: ${{ vars.PREVIEW_URL_TEMPLATE }}
|
||||
PREVIEW_TTL_HOURS: ${{ vars.PREVIEW_TTL_HOURS }}
|
||||
PREVIEW_STATE_FILE: .tmp/preview-slots.json
|
||||
CLIENT_DEPLOY_CMD: ${{ vars.CLIENT_DEPLOY_CMD }}
|
||||
SERVER_DEPLOY_CMD: ${{ vars.SERVER_DEPLOY_CMD }}
|
||||
FULL_STACK_DEPLOY_CMD: ${{ vars.FULL_STACK_DEPLOY_CMD }}
|
||||
INFRA_APPLY_CMD: ${{ vars.INFRA_APPLY_CMD }}
|
||||
steps:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Setup Python
|
||||
uses: actions/setup-python@v5
|
||||
with:
|
||||
python-version: "3.11"
|
||||
|
||||
- name: Resolve branch and issue
|
||||
id: target
|
||||
shell: bash
|
||||
run: |
|
||||
BRANCH="${{ inputs.branch }}"
|
||||
ISSUE_INPUT="${{ inputs.issue }}"
|
||||
if [ -z "$BRANCH" ]; then
|
||||
BRANCH="${GITHUB_REF_NAME:-$(git rev-parse --abbrev-ref HEAD)}"
|
||||
fi
|
||||
|
||||
ISSUE_ID="$ISSUE_INPUT"
|
||||
if [ -z "$ISSUE_ID" ]; then
|
||||
ISSUE_ID="$(echo "$BRANCH" | sed -nE 's#^issue[-/ ]?([0-9]+).*$#\1#p')"
|
||||
fi
|
||||
if [ -z "$ISSUE_ID" ]; then
|
||||
ISSUE_ID="$(echo "$BRANCH" | sed -nE 's#^.*/([0-9]+).*$#\1#p')"
|
||||
fi
|
||||
if [ -z "$ISSUE_ID" ]; then
|
||||
ISSUE_ID="0"
|
||||
fi
|
||||
|
||||
echo "branch=$BRANCH" >> "$GITHUB_OUTPUT"
|
||||
echo "issue=$ISSUE_ID" >> "$GITHUB_OUTPUT"
|
||||
|
||||
- name: Detect change scope
|
||||
id: scope
|
||||
shell: bash
|
||||
run: |
|
||||
git fetch origin main --depth=1 || true
|
||||
mkdir -p .tmp
|
||||
python skills/gitea-issue-devops-agent/scripts/change_scope.py \
|
||||
--repo-path . \
|
||||
--base-ref origin/main \
|
||||
--head-ref "${{ steps.target.outputs.branch }}" > .tmp/change-scope.json
|
||||
SCOPE="$(python -c "import json;print(json.load(open('.tmp/change-scope.json', encoding='utf-8'))['scope'])")"
|
||||
echo "scope=$SCOPE" >> "$GITHUB_OUTPUT"
|
||||
cat .tmp/change-scope.json
|
||||
|
||||
- name: Allocate preview slot
|
||||
id: slot
|
||||
shell: bash
|
||||
run: |
|
||||
SLOTS="${PREVIEW_SLOTS:-preview-a,preview-b}"
|
||||
TTL="${PREVIEW_TTL_HOURS:-24}"
|
||||
URL_TEMPLATE="${PREVIEW_URL_TEMPLATE:-https://{slot}.qa.example.com}"
|
||||
mkdir -p .tmp
|
||||
|
||||
python skills/gitea-issue-devops-agent/scripts/preview_slot_allocator.py \
|
||||
--state-file "$PREVIEW_STATE_FILE" \
|
||||
--slots "$SLOTS" \
|
||||
--repo "${GITHUB_REPOSITORY}" \
|
||||
--issue "${{ steps.target.outputs.issue }}" \
|
||||
--branch "${{ steps.target.outputs.branch }}" \
|
||||
--ttl-hours "$TTL" \
|
||||
--url-template "$URL_TEMPLATE" \
|
||||
--evict-oldest > .tmp/slot-allocation.json
|
||||
|
||||
SLOT="$(python -c "import json;d=json.load(open('.tmp/slot-allocation.json', encoding='utf-8'));print(d.get('allocation',{}).get('slot',''))")"
|
||||
URL="$(python -c "import json;d=json.load(open('.tmp/slot-allocation.json', encoding='utf-8'));print(d.get('allocation',{}).get('url',''))")"
|
||||
echo "slot=$SLOT" >> "$GITHUB_OUTPUT"
|
||||
echo "url=$URL" >> "$GITHUB_OUTPUT"
|
||||
cat .tmp/slot-allocation.json
|
||||
|
||||
- name: Deploy by scope
|
||||
shell: bash
|
||||
run: |
|
||||
set -euo pipefail
|
||||
SCOPE="${{ steps.scope.outputs.scope }}"
|
||||
run_or_echo () {
|
||||
local cmd="$1"
|
||||
local fallback="$2"
|
||||
if [ -n "$cmd" ]; then
|
||||
bash -lc "$cmd"
|
||||
else
|
||||
echo "$fallback"
|
||||
fi
|
||||
}
|
||||
|
||||
case "$SCOPE" in
|
||||
skip)
|
||||
echo "Scope=skip: docs/tests-only or no changes, deployment skipped."
|
||||
;;
|
||||
client_only)
|
||||
run_or_echo "${CLIENT_DEPLOY_CMD:-}" "Scope=client_only: set repo var CLIENT_DEPLOY_CMD."
|
||||
;;
|
||||
server_only)
|
||||
run_or_echo "${SERVER_DEPLOY_CMD:-}" "Scope=server_only: set repo var SERVER_DEPLOY_CMD."
|
||||
;;
|
||||
full_stack)
|
||||
run_or_echo "${FULL_STACK_DEPLOY_CMD:-}" "Scope=full_stack: set repo var FULL_STACK_DEPLOY_CMD."
|
||||
;;
|
||||
infra_only)
|
||||
run_or_echo "${INFRA_APPLY_CMD:-}" "Scope=infra_only: set repo var INFRA_APPLY_CMD."
|
||||
;;
|
||||
*)
|
||||
echo "Unknown scope: $SCOPE"
|
||||
exit 1
|
||||
;;
|
||||
esac
|
||||
|
||||
- name: Persist preview slot state
|
||||
shell: bash
|
||||
run: |
|
||||
if [ ! -f "$PREVIEW_STATE_FILE" ]; then
|
||||
exit 0
|
||||
fi
|
||||
if [ -z "$(git status --porcelain -- "$PREVIEW_STATE_FILE")" ]; then
|
||||
exit 0
|
||||
fi
|
||||
|
||||
git config user.name "gitea-actions"
|
||||
git config user.email "gitea-actions@local"
|
||||
git add "$PREVIEW_STATE_FILE"
|
||||
git commit -m "chore: update preview slot state [skip ci]" || true
|
||||
git push || true
|
||||
|
||||
- name: Summary
|
||||
shell: bash
|
||||
run: |
|
||||
echo "branch: ${{ steps.target.outputs.branch }}"
|
||||
echo "issue: ${{ steps.target.outputs.issue }}"
|
||||
echo "scope: ${{ steps.scope.outputs.scope }}"
|
||||
echo "slot: ${{ steps.slot.outputs.slot }}"
|
||||
echo "url: ${{ steps.slot.outputs.url }}"
|
||||
83
.gitea/workflows/preview-slot-reclaim.yml
Normal file
83
.gitea/workflows/preview-slot-reclaim.yml
Normal file
@@ -0,0 +1,83 @@
|
||||
name: preview-slot-reclaim
|
||||
|
||||
on:
|
||||
schedule:
|
||||
- cron: "15 * * * *"
|
||||
workflow_dispatch:
|
||||
inputs:
|
||||
repo:
|
||||
description: "owner/repo (optional)"
|
||||
required: false
|
||||
type: string
|
||||
branch:
|
||||
description: "Branch to release (optional)"
|
||||
required: false
|
||||
type: string
|
||||
issue:
|
||||
description: "Issue number to release (optional)"
|
||||
required: false
|
||||
type: string
|
||||
|
||||
jobs:
|
||||
reclaim:
|
||||
runs-on: ubuntu-latest
|
||||
env:
|
||||
PREVIEW_SLOTS: ${{ vars.PREVIEW_SLOTS }}
|
||||
PREVIEW_STATE_FILE: .tmp/preview-slots.json
|
||||
steps:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Setup Python
|
||||
uses: actions/setup-python@v5
|
||||
with:
|
||||
python-version: "3.11"
|
||||
|
||||
- name: Release target or prune expired
|
||||
shell: bash
|
||||
run: |
|
||||
SLOTS="${PREVIEW_SLOTS:-preview-a,preview-b}"
|
||||
REPO_INPUT="${{ inputs.repo }}"
|
||||
BRANCH_INPUT="${{ inputs.branch }}"
|
||||
ISSUE_INPUT="${{ inputs.issue }}"
|
||||
|
||||
mkdir -p .tmp
|
||||
|
||||
if [ -n "$REPO_INPUT" ] || [ -n "$BRANCH_INPUT" ] || [ -n "$ISSUE_INPUT" ]; then
|
||||
ISSUE_FLAG=""
|
||||
if [ -n "$ISSUE_INPUT" ]; then
|
||||
ISSUE_FLAG="--issue $ISSUE_INPUT"
|
||||
fi
|
||||
python skills/gitea-issue-devops-agent/scripts/preview_slot_allocator.py \
|
||||
--state-file "$PREVIEW_STATE_FILE" \
|
||||
--slots "$SLOTS" \
|
||||
--repo "$REPO_INPUT" \
|
||||
--branch "$BRANCH_INPUT" \
|
||||
$ISSUE_FLAG \
|
||||
--release > .tmp/slot-reclaim.json
|
||||
else
|
||||
python skills/gitea-issue-devops-agent/scripts/preview_slot_allocator.py \
|
||||
--state-file "$PREVIEW_STATE_FILE" \
|
||||
--slots "$SLOTS" \
|
||||
--list > .tmp/slot-reclaim.json
|
||||
fi
|
||||
|
||||
cat .tmp/slot-reclaim.json
|
||||
|
||||
- name: Persist state
|
||||
shell: bash
|
||||
run: |
|
||||
if [ ! -f "$PREVIEW_STATE_FILE" ]; then
|
||||
exit 0
|
||||
fi
|
||||
if [ -z "$(git status --porcelain -- "$PREVIEW_STATE_FILE")" ]; then
|
||||
exit 0
|
||||
fi
|
||||
|
||||
git config user.name "gitea-actions"
|
||||
git config user.email "gitea-actions@local"
|
||||
git add "$PREVIEW_STATE_FILE"
|
||||
git commit -m "chore: reclaim preview slots [skip ci]" || true
|
||||
git push || true
|
||||
Reference in New Issue
Block a user