release.yml 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. name: Release
  2. on:
  3. push:
  4. branches:
  5. - "**"
  6. tags:
  7. - "v*.*.*"
  8. paths-ignore:
  9. - "**/*.md"
  10. - "**/*.yaml"
  11. pull_request:
  12. branches:
  13. - "**"
  14. paths-ignore:
  15. - "**/*.md"
  16. - "**/*.yaml"
  17. env:
  18. DOCKERHUB_REPO: ${{ secrets.DOCKERHUB_REPO != '' && secrets.DOCKERHUB_REPO || secrets.DOCKERHUB_USERNAME != '' && format('{0}/{1}', secrets.DOCKERHUB_USERNAME, 'aiproxy') || '' }}
  19. GHCR_REPO: ghcr.io/${{ github.repository }}
  20. ALIYUN_REGISTRY: ${{ secrets.ALIYUN_REGISTRY }}
  21. ALIYUN_REPO: ${{ secrets.ALIYUN_REPO != '' && secrets.ALIYUN_REPO || (secrets.ALIYUN_REGISTRY != '' && secrets.ALIYUN_USERNAME != '') && format('{0}/{1}/{2}', secrets.ALIYUN_REGISTRY, secrets.ALIYUN_USERNAME, 'aiproxy') || '' }}
  22. jobs:
  23. release-web:
  24. name: Release Web
  25. runs-on: ubuntu-24.04
  26. steps:
  27. - uses: actions/checkout@v5
  28. - name: Use Node.js 22
  29. uses: actions/setup-node@v6
  30. with:
  31. node-version: 22.x
  32. - name: Build
  33. working-directory: web
  34. run: |
  35. npm install -g pnpm
  36. pnpm install && pnpm run build
  37. - name: Upload Artifact
  38. uses: actions/upload-artifact@v4
  39. with:
  40. name: web
  41. path: web/dist
  42. release:
  43. name: Release AI Proxy
  44. runs-on: ubuntu-24.04
  45. needs: release-web
  46. permissions:
  47. contents: write
  48. strategy:
  49. fail-fast: false
  50. matrix:
  51. targets:
  52. - GOOS: linux
  53. GOARCH: arm64
  54. - GOOS: linux
  55. GOARCH: amd64
  56. - GOOS: darwin
  57. GOARCH: arm64
  58. - GOOS: darwin
  59. GOARCH: amd64
  60. - GOOS: windows
  61. GOARCH: amd64
  62. EXT: .exe
  63. - GOOS: windows
  64. GOARCH: arm64
  65. EXT: .exe
  66. steps:
  67. - name: Checkout
  68. uses: actions/checkout@v5
  69. - uses: actions/download-artifact@v5
  70. with:
  71. name: web
  72. path: core/public/dist
  73. - name: Setup Go
  74. uses: actions/setup-go@v6
  75. with:
  76. go-version-file: "core/go.mod"
  77. - name: Generate Swagger
  78. working-directory: core
  79. run: |
  80. go install github.com/swaggo/swag/cmd/swag@latest
  81. bash scripts/swag.sh
  82. - name: Build
  83. working-directory: core
  84. run: |
  85. export GOOS=${{ matrix.targets.GOOS }}
  86. export GOARCH=${{ matrix.targets.GOARCH }}
  87. go build -trimpath -ldflags "-s -w" -o aiproxy-${{ matrix.targets.GOOS }}-${{ matrix.targets.GOARCH }}${{ matrix.targets.EXT }}
  88. - name: Get release meta
  89. id: release_meta
  90. if: ${{ startsWith(github.ref, 'refs/tags/') }}
  91. working-directory: core
  92. run: |
  93. version=${GITHUB_REF/refs\/tags\/v/}
  94. echo "version: ${version}"
  95. prerelease=$(echo ${version} | grep -E 'rc|beta|alpha|pre|dev|snapshot|nightly' || true)
  96. release_name="Version ${version}"
  97. echo "release_name: ${release_name}"
  98. if [ -n "${prerelease}" ]; then
  99. prerelease=true
  100. release_name="${release_name} (Prerelease)"
  101. else
  102. prerelease=false
  103. fi
  104. tag_name="v${version}"
  105. echo "prerelease: ${prerelease}"
  106. echo "tag_name: ${tag_name}"
  107. echo "PRERELEASE=${prerelease}" >> $GITHUB_OUTPUT
  108. echo "RELEASE_NAME=${release_name}" >> $GITHUB_OUTPUT
  109. echo "TAG_NAME=${tag_name}" >> $GITHUB_OUTPUT
  110. - name: Release
  111. uses: softprops/action-gh-release@v2
  112. if: ${{ startsWith(github.ref, 'refs/tags/') }}
  113. with:
  114. token: ${{ secrets.GITHUB_TOKEN }}
  115. draft: false
  116. prerelease: ${{ steps.release_meta.outputs.PRERELEASE }}
  117. append_body: false
  118. fail_on_unmatched_files: true
  119. name: ${{ steps.release_meta.outputs.RELEASE_NAME }}
  120. tag_name: ${{ steps.release_meta.outputs.TAG_NAME }}
  121. files: |
  122. core/aiproxy-${{ matrix.targets.GOOS }}-${{ matrix.targets.GOARCH }}${{ matrix.targets.EXT }}
  123. build-docker-images:
  124. name: Build Docker Images
  125. permissions:
  126. packages: write
  127. strategy:
  128. matrix:
  129. include:
  130. - arch: amd64
  131. - arch: arm64
  132. runs-on: ubuntu-24.04-arm
  133. runs-on: ${{ matrix.runs-on || 'ubuntu-24.04' }}
  134. steps:
  135. - name: Login to GitHub Container Registry
  136. if: ${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' }}
  137. uses: docker/login-action@v3
  138. with:
  139. registry: ghcr.io
  140. username: ${{ github.actor }}
  141. password: ${{ secrets.GITHUB_TOKEN }}
  142. - name: Login to DockerHub
  143. uses: docker/login-action@v3
  144. if: ${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' && env.DOCKERHUB_REPO }}
  145. with:
  146. username: ${{ secrets.DOCKERHUB_USERNAME }}
  147. password: ${{ secrets.DOCKERHUB_TOKEN }}
  148. - name: Login to Aliyun Registry
  149. uses: docker/login-action@v3
  150. if: ${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' && env.ALIYUN_REGISTRY }}
  151. with:
  152. registry: ${{ env.ALIYUN_REGISTRY }}
  153. username: ${{ secrets.ALIYUN_USERNAME }}
  154. password: ${{ secrets.ALIYUN_PASSWORD }}
  155. - name: Checkout
  156. uses: actions/checkout@v5
  157. - name: Set up Docker Buildx
  158. uses: docker/setup-buildx-action@v3
  159. - name: Extract metadata (tags, labels) for Docker
  160. id: meta
  161. uses: docker/metadata-action@v5
  162. with:
  163. images: |
  164. ${{ env.DOCKERHUB_REPO }}
  165. ${{ env.GHCR_REPO }}
  166. ${{ env.ALIYUN_REPO }}
  167. - name: Build for ${{ matrix.arch }}
  168. id: build
  169. uses: docker/build-push-action@v6
  170. with:
  171. context: .
  172. labels: ${{ steps.meta.outputs.labels }}
  173. platforms: linux/${{ matrix.arch }}
  174. outputs: type=image,"name=${{ env.GHCR_REPO }}${{ env.DOCKERHUB_REPO && format(',{0}', env.DOCKERHUB_REPO) }}${{ env.ALIYUN_REPO && format(',{0}', env.ALIYUN_REPO) }}",name-canonical=true,push-by-digest=${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' }},push=${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' }}
  175. - name: Export digest
  176. working-directory: core
  177. run: |
  178. mkdir -p ${{ runner.temp }}/digests
  179. digest="${{ steps.build.outputs.digest }}"
  180. touch "${{ runner.temp }}/digests/${digest#sha256:}"
  181. - name: Upload digest
  182. uses: actions/upload-artifact@v4
  183. with:
  184. name: digests-${{ matrix.arch }}
  185. path: ${{ runner.temp }}/digests/*
  186. if-no-files-found: error
  187. retention-days: 1
  188. release-docker-images:
  189. name: Push Docker Images
  190. permissions:
  191. packages: write
  192. needs: build-docker-images
  193. runs-on: ubuntu-24.04
  194. if: ${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' }}
  195. steps:
  196. - name: Login to GitHub Container Registry
  197. uses: docker/login-action@v3
  198. with:
  199. registry: ghcr.io
  200. username: ${{ github.actor }}
  201. password: ${{ secrets.GITHUB_TOKEN }}
  202. - name: Login to DockerHub
  203. uses: docker/login-action@v3
  204. if: ${{ env.DOCKERHUB_REPO }}
  205. with:
  206. username: ${{ secrets.DOCKERHUB_USERNAME }}
  207. password: ${{ secrets.DOCKERHUB_TOKEN }}
  208. - name: Login to Aliyun Registry
  209. uses: docker/login-action@v3
  210. if: ${{ env.ALIYUN_REGISTRY }}
  211. with:
  212. registry: ${{ env.ALIYUN_REGISTRY }}
  213. username: ${{ secrets.ALIYUN_USERNAME }}
  214. password: ${{ secrets.ALIYUN_PASSWORD }}
  215. - name: Download digests
  216. uses: actions/download-artifact@v5
  217. with:
  218. path: ${{ runner.temp }}/digests
  219. pattern: digests-*
  220. merge-multiple: true
  221. - name: Set up Docker Buildx
  222. uses: docker/setup-buildx-action@v3
  223. - name: Extract metadata (tags, labels) for Docker
  224. id: meta
  225. uses: docker/metadata-action@v5
  226. with:
  227. images: |
  228. ${{ env.DOCKERHUB_REPO }}
  229. ${{ env.GHCR_REPO }}
  230. ${{ env.ALIYUN_REPO }}
  231. tags: |
  232. type=ref,event=branch
  233. type=ref,event=pr
  234. type=ref,event=tag
  235. type=semver,pattern={{version}}
  236. type=semver,pattern={{major}}.{{minor}}
  237. type=semver,pattern={{major}}
  238. type=sha
  239. - name: Create manifest list and push
  240. working-directory: ${{ runner.temp }}/digests
  241. run: |
  242. for TAG in $DOCKER_METADATA_OUTPUT_TAGS; do
  243. docker buildx imagetools create -t $TAG \
  244. $(printf '${{ env.GHCR_REPO }}@sha256:%s ' *)
  245. sleep 5
  246. done
  247. - name: Inspect image
  248. run: |
  249. docker buildx imagetools inspect ${{ env.GHCR_REPO }}:${{ steps.meta.outputs.version }}
  250. release-sealos-images:
  251. name: Push Sealos Images
  252. permissions:
  253. packages: write
  254. needs: release-docker-images
  255. runs-on: ubuntu-24.04
  256. if: ${{ github.event_name != 'pull_request' && github.actor != 'dependabot[bot]' }}
  257. steps:
  258. - name: Checkout
  259. uses: actions/checkout@v5
  260. - name: install cache images tools
  261. run: |
  262. sudo bash ./.github/scripts/install.sh
  263. - name: Set up QEMU
  264. uses: docker/setup-qemu-action@v3
  265. - uses: docker/setup-buildx-action@v3
  266. - name: Prepare cluster image list
  267. id: cluster_image_targets
  268. run: |
  269. set -euo pipefail
  270. images=("${GHCR_REPO}-cluster")
  271. if [ -n "${DOCKERHUB_REPO}" ]; then
  272. images+=("${DOCKERHUB_REPO}-cluster")
  273. fi
  274. if [ -n "${ALIYUN_REPO}" ]; then
  275. images+=("${ALIYUN_REPO}-cluster")
  276. fi
  277. {
  278. echo "images<<EOF"
  279. printf '%s\n' "${images[@]}"
  280. echo "EOF"
  281. csv=$(IFS=','; printf '%s' "${images[*]}")
  282. echo "names=${csv}"
  283. } >> "${GITHUB_OUTPUT}"
  284. - name: Extract metadata (tags, labels) for Docker
  285. id: meta
  286. uses: docker/metadata-action@v5
  287. with:
  288. images: ${{ steps.cluster_image_targets.outputs.images }}
  289. tags: |
  290. type=ref,event=branch
  291. type=ref,event=pr
  292. type=ref,event=tag
  293. type=semver,pattern={{version}}
  294. type=semver,pattern={{major}}.{{minor}}
  295. type=semver,pattern={{major}}
  296. type=sha
  297. - name: cache images
  298. working-directory: core/deploy
  299. run: |
  300. sudo sealos login -u "${{ github.repository_owner }}" -p "${{ secrets.GITHUB_TOKEN }}" ghcr.io
  301. sed -i "s#image: ghcr.io/labring/aiproxy:latest#image: ${{ env.GHCR_REPO }}:${{ steps.meta.outputs.version }}#g" charts/aiproxy/values.yaml
  302. sudo sealos registry save --registry-dir=registry_amd64 --arch amd64 .
  303. sudo sealos registry save --registry-dir=registry_arm64 --arch arm64 .
  304. - name: Login to GitHub Container Registry
  305. uses: docker/login-action@v3
  306. with:
  307. registry: ghcr.io
  308. username: ${{ github.actor }}
  309. password: ${{ secrets.GITHUB_TOKEN }}
  310. - name: Login to DockerHub
  311. uses: docker/login-action@v3
  312. if: ${{ env.DOCKERHUB_REPO }}
  313. with:
  314. username: ${{ secrets.DOCKERHUB_USERNAME }}
  315. password: ${{ secrets.DOCKERHUB_TOKEN }}
  316. - name: Login to Aliyun Registry
  317. uses: docker/login-action@v3
  318. if: ${{ env.ALIYUN_REGISTRY }}
  319. with:
  320. registry: ${{ env.ALIYUN_REGISTRY }}
  321. username: ${{ secrets.ALIYUN_USERNAME }}
  322. password: ${{ secrets.ALIYUN_PASSWORD }}
  323. - name: Build images
  324. uses: docker/build-push-action@v6
  325. with:
  326. context: ./core/deploy
  327. file: ./core/deploy/Kubefile
  328. labels: ${{ steps.meta.outputs.labels }}
  329. platforms: linux/amd64,linux/arm64
  330. push: ${{ github.event_name != 'pull_request' }}
  331. tags: ${{ steps.meta.outputs.tags }}