|
| 1 | +#!/usr/bin/env sh |
| 2 | + |
| 3 | +set -eu |
| 4 | + |
| 5 | +usage() { |
| 6 | + cat <<'USAGE' |
| 7 | +在仓库内直接创建 GitHub Issue。 |
| 8 | +
|
| 9 | +用法: |
| 10 | + ./scripts/create_issue.sh --type <proposal|architecture|implementation> --title <标题> [选项] |
| 11 | +
|
| 12 | +选项: |
| 13 | + --repo <owner/repo> 目标仓库,默认自动检测当前仓库 |
| 14 | + --body-file <path> 指定 issue 正文文件 |
| 15 | + --labels <a,b,c> 逗号分隔的标签列表(可选) |
| 16 | + --type <type> issue 类型:proposal|architecture|implementation |
| 17 | + --title <title> issue 标题(不含类型前缀) |
| 18 | + -h, --help 显示帮助 |
| 19 | +
|
| 20 | +示例: |
| 21 | + ./scripts/create_issue.sh --type proposal --title "新增会话恢复策略" |
| 22 | + ./scripts/create_issue.sh --type implementation --title "修复 streaming 中断持久化" --labels "bug,priority-high" |
| 23 | +USAGE |
| 24 | +} |
| 25 | + |
| 26 | +require_cmd() { |
| 27 | + if ! command -v "$1" >/dev/null 2>&1; then |
| 28 | + echo "缺少命令: $1" >&2 |
| 29 | + exit 1 |
| 30 | + fi |
| 31 | +} |
| 32 | + |
| 33 | +default_repo() { |
| 34 | + gh repo view --json nameWithOwner -q .nameWithOwner 2>/dev/null || true |
| 35 | +} |
| 36 | + |
| 37 | +title_prefix() { |
| 38 | + case "$1" in |
| 39 | + proposal) echo "【提案】" ;; |
| 40 | + architecture) echo "【架构】" ;; |
| 41 | + implementation) echo "【实现】" ;; |
| 42 | + *) return 1 ;; |
| 43 | + esac |
| 44 | +} |
| 45 | + |
| 46 | +# trim_label 用于去除标签参数的首尾空白字符,避免传递无效标签值。 |
| 47 | +trim_label() { |
| 48 | + printf '%s' "$1" | sed 's/^[[:space:]]*//;s/[[:space:]]*$//' |
| 49 | +} |
| 50 | + |
| 51 | +create_body_file() { |
| 52 | + type="$1" |
| 53 | + out="$2" |
| 54 | + |
| 55 | + case "$type" in |
| 56 | + proposal) |
| 57 | + cat >"$out" <<'BODY' |
| 58 | +### 目标问题(Why) |
| 59 | +- 当前痛点: |
| 60 | +- 触发场景: |
| 61 | +
|
| 62 | +### 设计方案(How) |
| 63 | +- 核心设计: |
| 64 | +- 关键机制: |
| 65 | +- 边界与非目标: |
| 66 | +
|
| 67 | +### 落地清单(What) |
| 68 | +- [ ] |
| 69 | +- [ ] |
| 70 | +
|
| 71 | +### 验收标准(Done) |
| 72 | +- [ ] |
| 73 | +- [ ] |
| 74 | +
|
| 75 | +### 风险与回滚 |
| 76 | +- 风险: |
| 77 | +- 回滚方案: |
| 78 | +BODY |
| 79 | + ;; |
| 80 | + architecture) |
| 81 | + cat >"$out" <<'BODY' |
| 82 | +### 目标问题(Why) |
| 83 | +- 当前痛点: |
| 84 | +- 影响范围: |
| 85 | +
|
| 86 | +### 现状与边界 |
| 87 | +- TUI: |
| 88 | +- Runtime: |
| 89 | +- Provider/Tools: |
| 90 | +- Session/Context: |
| 91 | +
|
| 92 | +### 核心设计(How) |
| 93 | +- 核心设计: |
| 94 | +- 数据流/事件流: |
| 95 | +- 关键取舍: |
| 96 | +
|
| 97 | +### 落地清单(What) |
| 98 | +- [ ] |
| 99 | +- [ ] |
| 100 | +
|
| 101 | +### 验收标准(Done) |
| 102 | +- [ ] |
| 103 | +- [ ] |
| 104 | +
|
| 105 | +### 风险与回滚 |
| 106 | +- 风险: |
| 107 | +- 回滚方案: |
| 108 | +BODY |
| 109 | + ;; |
| 110 | + implementation) |
| 111 | + cat >"$out" <<'BODY' |
| 112 | +### 关联 RFC / 架构 |
| 113 | +- 提案/架构 issue: |
| 114 | +- 当前问题: |
| 115 | +
|
| 116 | +### 实现设计(How) |
| 117 | +- 关键改动点: |
| 118 | +- 影响模块: |
| 119 | +- 边界与非目标: |
| 120 | +
|
| 121 | +### 任务拆解 |
| 122 | +- [ ] |
| 123 | +- [ ] |
| 124 | +
|
| 125 | +### 测试与验证(Done) |
| 126 | +- [ ] 正常路径 |
| 127 | +- [ ] 边界条件 |
| 128 | +- [ ] 异常分支 |
| 129 | +
|
| 130 | +### 风险与回滚 |
| 131 | +- 风险: |
| 132 | +- 回滚方案: |
| 133 | +BODY |
| 134 | + ;; |
| 135 | + *) |
| 136 | + echo "不支持的类型: $type" >&2 |
| 137 | + exit 1 |
| 138 | + ;; |
| 139 | + esac |
| 140 | +} |
| 141 | + |
| 142 | +REPO="" |
| 143 | +BODY_FILE="" |
| 144 | +LABELS="" |
| 145 | +TYPE="" |
| 146 | +TITLE="" |
| 147 | + |
| 148 | +while [ "$#" -gt 0 ]; do |
| 149 | + case "$1" in |
| 150 | + --repo) |
| 151 | + REPO="${2:-}" |
| 152 | + shift 2 |
| 153 | + ;; |
| 154 | + --body-file) |
| 155 | + BODY_FILE="${2:-}" |
| 156 | + shift 2 |
| 157 | + ;; |
| 158 | + --labels) |
| 159 | + LABELS="${2:-}" |
| 160 | + shift 2 |
| 161 | + ;; |
| 162 | + --type) |
| 163 | + TYPE="${2:-}" |
| 164 | + shift 2 |
| 165 | + ;; |
| 166 | + --title) |
| 167 | + TITLE="${2:-}" |
| 168 | + shift 2 |
| 169 | + ;; |
| 170 | + -h|--help) |
| 171 | + usage |
| 172 | + exit 0 |
| 173 | + ;; |
| 174 | + *) |
| 175 | + echo "未知参数: $1" >&2 |
| 176 | + usage |
| 177 | + exit 1 |
| 178 | + ;; |
| 179 | + esac |
| 180 | +done |
| 181 | + |
| 182 | +require_cmd gh |
| 183 | + |
| 184 | +if [ -z "$TYPE" ] || [ -z "$TITLE" ]; then |
| 185 | + echo "--type 和 --title 为必填参数" >&2 |
| 186 | + usage |
| 187 | + exit 1 |
| 188 | +fi |
| 189 | + |
| 190 | +if [ -z "$REPO" ]; then |
| 191 | + REPO="$(default_repo)" |
| 192 | +fi |
| 193 | +if [ -z "$REPO" ]; then |
| 194 | + echo "无法自动识别仓库,请通过 --repo 显式传入 owner/repo" >&2 |
| 195 | + exit 1 |
| 196 | +fi |
| 197 | + |
| 198 | +PREFIX="$(title_prefix "$TYPE" || true)" |
| 199 | +if [ -z "$PREFIX" ]; then |
| 200 | + echo "--type 仅支持: proposal | architecture | implementation" >&2 |
| 201 | + exit 1 |
| 202 | +fi |
| 203 | + |
| 204 | +FINAL_TITLE="$PREFIX $TITLE" |
| 205 | +TEMP_BODY="" |
| 206 | +if [ -n "$BODY_FILE" ]; then |
| 207 | + if [ ! -f "$BODY_FILE" ]; then |
| 208 | + echo "--body-file 指向的文件不存在: $BODY_FILE" >&2 |
| 209 | + exit 1 |
| 210 | + fi |
| 211 | +else |
| 212 | + TEMP_BODY="$(mktemp -t neocode-issue-body-XXXXXX.md)" |
| 213 | + BODY_FILE="$TEMP_BODY" |
| 214 | + create_body_file "$TYPE" "$BODY_FILE" |
| 215 | +fi |
| 216 | + |
| 217 | +cleanup() { |
| 218 | + if [ -n "$TEMP_BODY" ] && [ -f "$TEMP_BODY" ]; then |
| 219 | + rm -f "$TEMP_BODY" |
| 220 | + fi |
| 221 | +} |
| 222 | +trap cleanup EXIT INT TERM |
| 223 | + |
| 224 | +set -- issue create --repo "$REPO" --title "$FINAL_TITLE" --body-file "$BODY_FILE" |
| 225 | +if [ -n "$LABELS" ]; then |
| 226 | + OLD_IFS=$IFS |
| 227 | + IFS=',' |
| 228 | + for label in $LABELS; do |
| 229 | + trimmed_label="$(trim_label "$label")" |
| 230 | + if [ -n "$trimmed_label" ]; then |
| 231 | + set -- "$@" --label "$trimmed_label" |
| 232 | + fi |
| 233 | + done |
| 234 | + IFS=$OLD_IFS |
| 235 | +fi |
| 236 | + |
| 237 | +ISSUE_URL="$(gh "$@")" |
| 238 | +echo "Issue created: $ISSUE_URL" |
0 commit comments