Files
bl/.woodpecker/my-first-workflow.yaml

240 lines
9.1 KiB
Go
Raw Normal View History

2026-01-27 06:10:17 +00:00
# 触发条件仅pushmanual手动触发分支限定main修正格式错误
2026-01-26 10:14:44 +08:00
when:
2026-01-27 06:10:17 +00:00
- event: push
- event: manual
branch: main
2026-01-26 11:07:37 +08:00
skip_clone: true
2026-01-26 14:06:51 +08:00
2026-01-27 00:02:38 +08:00
# 全局配置变量替换占位符即可使用
variables:
SCREEN_NAME: "logic_service"
REMOTE_EXE_DIR: "/opt/logic"
JSON_CONFIG_URL: "https://你的JSON配置地址.com/deploy.json"
LOG_PATH: "$HOME/run.log"
2026-01-27 06:10:17 +00:00
# 流水线核心步骤理顺依赖链确保build依赖clone结果
2026-01-27 00:02:38 +08:00
steps:
2026-01-27 06:10:17 +00:00
# ========== 1. 替代clone拉取代码核心依赖 ==========
prepare:
image: debian:bookworm
2026-01-27 06:10:17 +00:00
environment:
from_secret:
WOODPECKER_SSH_KEY: WOODPECKER_SSH_KEY
commands:
2026-01-27 06:10:17 +00:00
- apt update
2026-01-27 00:57:40 +08:00
# 安装完整依赖解决SSH/libcrypto问题
2026-01-27 06:10:17 +00:00
- apt install -y ca-certificates curl git openssh-client openssl libssl-dev
# 清理旧SSH文件严格配置权限
2026-01-27 00:57:40 +08:00
- rm -rf /root/.ssh/*
- mkdir -p /root/.ssh && chmod 700 /root/.ssh
2026-01-27 06:10:17 +00:00
# 写入并清理SSH密钥兼容RSA/ED25519保留原始换行
# 替换原echo "$WOODPECKER_SSH_KEY"那行其余ssh配置不变
# ========== 关键修改SSH密钥写入逻辑核心修复 ==========
# 替换原echo命令用cat+EOF保留密钥原始换行避免格式损坏
2026-01-27 03:05:37 +00:00
- |
2026-01-27 06:17:13 +00:00
2026-01-27 03:05:37 +00:00
if [ -n "$WOODPECKER_SSH_KEY" ]; then
2026-01-27 06:17:13 +00:00
cat > /root/.ssh/id_ed25519 << EOF
2026-01-27 06:10:17 +00:00
$WOODPECKER_SSH_KEY
EOF
2026-01-27 06:17:13 +00:00
chmod 600 /root/.ssh/id_ed25519
2026-01-27 03:05:37 +00:00
echo "✅ ED25519密钥写入完成"
2026-01-27 06:17:13 +00:00
else
echo "⚠️ WOODPECKER_SSH_KEY变量为空未写入SSH密钥"
fi
2026-01-27 06:10:17 +00:00
- chmod 600 /root/.ssh/id_ed25519
2026-01-27 01:01:23 +08:00
# 添加GitHub主机密钥完整覆盖避免重复
- ssh-keyscan -H github.com > /root/.ssh/known_hosts
- chmod 600 /root/.ssh/known_hosts
2026-01-27 06:10:17 +00:00
# 检查私钥是否能被解析关键测试
2026-01-27 02:48:52 +00:00
- ssh-keygen -lf /root/.ssh/id_ed25519
2026-01-27 01:01:23 +08:00
- echo "🔍 尝试SSH连接GitHub详细日志..."
2026-01-27 06:10:17 +00:00
# 重新执行命令取消输出屏蔽查看具体错误
2026-01-27 03:46:44 +08:00
- ssh -vvv -i /root/.ssh/id_ed25519 -o StrictHostKeyChecking=accept-new git@github.com 2>&1
2026-01-27 00:57:40 +08:00
# 拉取代码拆分命令避免静默失败
- git init
- git remote add origin $CI_REPO_CLONE_SSH_URL
- git config core.sshCommand 'ssh -i /root/.ssh/id_ed25519'
2026-01-27 00:57:40 +08:00
- git fetch origin $CI_REPO_DEFAULT_BRANCH
- git checkout $CI_REPO_DEFAULT_BRANCH
- echo "✅ 代码拉取完成"
2026-01-26 14:06:51 +08:00
2026-01-27 06:10:17 +00:00
# ========== 2. 初始化Go环境依赖prepare代码拉取完成 ==========
prepare-go:
2026-01-26 14:19:12 +08:00
image: golang:1.25
2026-01-27 06:10:17 +00:00
depends_on: [prepare] # 明确依赖先拉代码再初始化Go
2026-01-26 14:19:12 +08:00
commands:
- go version
- go mod download -x
- go mod verify
2026-01-27 00:57:40 +08:00
- echo "✅ Go环境初始化完成"
2026-01-26 14:06:51 +08:00
2026-01-27 06:10:17 +00:00
# ========== 3. 生成版本号依赖prepare-goGo环境就绪 ==========
set-version:
2026-01-26 14:19:12 +08:00
image: golang:1.25
2026-01-27 06:10:17 +00:00
depends_on: [prepare-go] # 明确依赖Go环境就绪后生成版本号
2026-01-26 14:19:12 +08:00
commands:
2026-01-27 00:02:38 +08:00
- |
if [ -n "${CI_COMMIT_TAG}" ]; then
VERSION="${CI_COMMIT_TAG}"
else
2026-01-27 06:10:17 +00:00
VERSION="v$(git rev-parse --short=8 HEAD)" # 依赖prepare拉取的代码
2026-01-27 00:02:38 +08:00
fi
echo "BUILD_VERSION=${VERSION}" >> $CI_ENV_FILE
echo "✅ 生成版本号:${VERSION}"
2026-01-26 14:06:51 +08:00
2026-01-27 06:10:17 +00:00
# ========== 4. 编译Go服务核心依赖prepare+prepare-go+set-version ==========
build_logic:
2026-01-26 14:19:12 +08:00
image: golang:1.25
2026-01-27 06:10:17 +00:00
depends_on: [set-version] # 明确依赖版本号生成后再编译
2026-01-27 00:02:38 +08:00
environment:
CGO_ENABLED: 0
GO111MODULE: on
GOSUMDB: off
2026-01-26 14:19:12 +08:00
commands:
- mkdir -p build
2026-01-27 06:10:17 +00:00
- BIN_NAME="logic_${BUILD_VERSION}" # 依赖set-version的BUILD_VERSION
2026-01-26 14:19:12 +08:00
- echo "BIN_NAME=${BIN_NAME}" >> $CI_ENV_FILE
2026-01-27 00:02:38 +08:00
- |
go build -v \
-p=4 \
-trimpath \
-buildvcs=false \
-ldflags "-s -w -buildid= -extldflags '-static' -X main.version=${BUILD_VERSION}" \
-o ./build/${BIN_NAME} \
2026-01-27 06:10:17 +00:00
./logic # 依赖prepare拉取的./logic代码目录
2026-01-27 00:02:38 +08:00
- |
if [ ! -f ./build/${BIN_NAME} ]; then
2026-01-26 14:26:57 +08:00
echo "❌ 编译失败:产物${BIN_NAME}不存在"
exit 1
fi
2026-01-26 14:19:12 +08:00
- ls -lh ./build/
- ./build/${BIN_NAME} -v || true
2026-01-26 14:26:57 +08:00
- echo "✅ Go服务编译完成"
2026-01-26 14:06:51 +08:00
2026-01-27 06:10:17 +00:00
# ========== 5. 拉取部署配置可并行依赖prepare确保代码拉取完成 ==========
fetch-deploy-config:
2026-01-26 14:19:12 +08:00
image: alpine:latest
2026-01-27 06:10:17 +00:00
depends_on: [prepare] # 确保代码拉取完成后再拉配置
2026-01-26 14:19:12 +08:00
commands:
2026-01-27 00:02:38 +08:00
- apk add --no-cache curl jq
2026-01-26 14:26:57 +08:00
- echo "🔧 拉取部署配置:${JSON_CONFIG_URL}"
2026-01-27 00:02:38 +08:00
- |
for i in 1 2 3; do
curl -sSL --connect-timeout 10 ${JSON_CONFIG_URL} -o /tmp/deploy-config.json && break
echo "⚠️ 第${i}次拉取失败,重试中..."
sleep 2
done
if [ ! -f /tmp/deploy-config.json ]; then
2026-01-27 06:10:17 +00:00
echo "❌ 拉取JSON失败"
2026-01-27 00:02:38 +08:00
exit 1
fi
2026-01-26 14:19:12 +08:00
- |
JSON_TYPE=$(jq -r 'type' /tmp/deploy-config.json)
if [ "$JSON_TYPE" = "array" ]; then
2026-01-27 06:10:17 +00:00
REMOTE_HOSTS=$(jq -r '.[].loginaddr' /tmp/deploy-config.json | tr '\n' ',' | sed 's/,$//')
REMOTE_USERS=$(jq -r '.[].user' /tmp/deploy-config.json | tr '\n' ',' | sed 's/,$//')
REMOTE_PASSWORDS=$(jq -r '.[].password' /tmp/deploy-config.json | tr '\n' ',' | sed 's/,$//')
REMOTE_ONLINE_IDS=$(jq -r '.[].online_id' /tmp/deploy-config.json | tr '\n' ',' | sed 's/,$//')
2026-01-26 14:19:12 +08:00
else
2026-01-27 06:10:17 +00:00
REMOTE_HOSTS=$(jq -r '.loginaddr' /tmp/deploy-config.json)
REMOTE_USERS=$(jq -r '.user' /tmp/deploy-config.json)
REMOTE_PASSWORDS=$(jq -r '.password' /tmp/deploy-config.json)
REMOTE_ONLINE_IDS=$(jq -r '.online_id' /tmp/deploy-config.json)
2026-01-26 14:19:12 +08:00
fi
2026-01-27 00:02:38 +08:00
- |
if [ -z "$REMOTE_HOSTS" ] || [ -z "$REMOTE_USERS" ]; then
echo "❌ 解析配置失败:服务器/用户名为空"
cat /tmp/deploy-config.json
exit 1
fi
2026-01-26 14:26:57 +08:00
- echo "✅ 配置解析完成 | 服务器:${REMOTE_HOSTS} | OnlineID${REMOTE_ONLINE_IDS}"
2026-01-26 14:19:12 +08:00
- echo "REMOTE_HOSTS=${REMOTE_HOSTS}" >> $CI_ENV_FILE
- echo "REMOTE_USERS=${REMOTE_USERS}" >> $CI_ENV_FILE
- echo "REMOTE_PASSWORDS=${REMOTE_PASSWORDS}" >> $CI_ENV_FILE
- echo "REMOTE_ONLINE_IDS=${REMOTE_ONLINE_IDS}" >> $CI_ENV_FILE
2026-01-26 14:06:51 +08:00
2026-01-27 06:10:17 +00:00
# ========== 6. SCP推送产物依赖编译+配置解析 ==========
scp-exe-to-servers:
2026-01-27 00:02:38 +08:00
image: appleboy/drone-scp:1.6.2
2026-01-26 14:19:12 +08:00
settings:
host: ${REMOTE_HOSTS}
username: ${REMOTE_USERS}
password: ${REMOTE_PASSWORDS}
source: ./build/${BIN_NAME}
target: ${REMOTE_EXE_DIR}/
strip_components: 1
skip_verify: true
2026-01-27 00:02:38 +08:00
timeout: 30s
2026-01-26 14:22:21 +08:00
depends_on:
- build_logic
2026-01-26 14:22:21 +08:00
- fetch-deploy-config
2026-01-26 14:17:42 +08:00
2026-01-27 06:10:17 +00:00
# ========== 7. 远程部署依赖SCP推送 ==========
deploy-to-servers:
2026-01-27 00:02:38 +08:00
image: appleboy/drone-ssh:1.7.0
2026-01-26 14:19:12 +08:00
settings:
host: ${REMOTE_HOSTS}
username: ${REMOTE_USERS}
password: ${REMOTE_PASSWORDS}
skip_verify: true
2026-01-27 00:02:38 +08:00
timeout: 60s
2026-01-26 14:19:12 +08:00
script:
- |
if command -v screen &> /dev/null; then
2026-01-27 06:10:17 +00:00
echo "✅ Screen已安装"
2026-01-26 14:19:12 +08:00
else
2026-01-27 06:10:17 +00:00
echo "🔧 安装Screen..."
if command -v apt &> /dev/null; then
apt update -y && apt install -y screen -qq
elif command -v yum &> /dev/null; then
yum install -y screen -q
elif command -v dnf &> /dev/null; then
dnf install -y screen -q
else
echo "❌ 不支持的包管理器,安装失败"
exit 1
fi
command -v screen || { echo "❌ Screen安装后检测失败"; exit 1; }
2026-01-26 14:19:12 +08:00
fi
- mkdir -p ${REMOTE_EXE_DIR}
2026-01-27 06:10:17 +00:00
- chmod +x ${REMOTE_EXE_DIR}/${BIN_NAME} || { echo "❌ 设置执行权限失败"; exit 1; }
2026-01-26 14:19:12 +08:00
- screen -S ${SCREEN_NAME} -X quit || true
- |
2026-01-26 14:26:57 +08:00
echo "🚀 启动程序:${REMOTE_EXE_DIR}/${BIN_NAME} -id=${REMOTE_ONLINE_IDS}"
screen -dmS ${SCREEN_NAME} bash -c "\"${REMOTE_EXE_DIR}/${BIN_NAME}\" -id=${REMOTE_ONLINE_IDS} | tee -a ${LOG_PATH}"
2026-01-26 14:19:12 +08:00
sleep 2
if screen -ls | grep -q "${SCREEN_NAME}"; then
2026-01-27 06:10:17 +00:00
echo "✅ 启动成功Screen会话"
screen -ls
2026-01-26 14:19:12 +08:00
else
2026-01-27 06:10:17 +00:00
echo "❌ 启动失败,无${SCREEN_NAME}会话"
screen -ls && exit 1
2026-01-26 14:19:12 +08:00
fi
2026-01-27 06:10:17 +00:00
depends_on: [scp-exe-to-servers]
2026-01-26 14:06:51 +08:00
2026-01-27 06:10:17 +00:00
# ========== 8. 打印部署汇总依赖部署完成 ==========
print-deploy-info:
2026-01-26 14:19:12 +08:00
image: alpine:latest
2026-01-27 00:57:40 +08:00
depends_on: [deploy-to-servers]
2026-01-26 14:19:12 +08:00
commands:
- echo "======================================"
2026-01-27 06:10:17 +00:00
- echo "🎉 全量部署完成!"
- echo "📌 版本号:${BUILD_VERSION}"
- echo "📌 触发方式:${CI_EVENT_NAME}"
- echo "📌 部署服务器:${REMOTE_HOSTS}"
- echo "📌 OnlineID${REMOTE_ONLINE_IDS}"
- echo "📌 Screen会话${SCREEN_NAME}"
- echo "📌 程序目录:${REMOTE_EXE_DIR}"
- echo "📌 日志路径:${LOG_PATH}"
- echo "📌 对应Commit${CI_COMMIT_SHA}"
2026-01-27 03:30:36 +08:00
- echo "======================================"