publish.yml 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447
  1. name: publish
  2. run-name: "${{ format('release {0}', inputs.bump) }}"
  3. on:
  4. push:
  5. branches:
  6. - ci
  7. - dev
  8. - beta
  9. - snapshot-*
  10. workflow_dispatch:
  11. inputs:
  12. bump:
  13. description: "Bump major, minor, or patch"
  14. required: false
  15. type: choice
  16. options:
  17. - major
  18. - minor
  19. - patch
  20. version:
  21. description: "Override version (optional)"
  22. required: false
  23. type: string
  24. concurrency: ${{ github.workflow }}-${{ github.ref }}-${{ inputs.version || inputs.bump }}
  25. permissions:
  26. id-token: write
  27. contents: write
  28. packages: write
  29. jobs:
  30. version:
  31. runs-on: blacksmith-4vcpu-ubuntu-2404
  32. if: github.repository == 'anomalyco/opencode'
  33. steps:
  34. - uses: actions/checkout@v3
  35. with:
  36. fetch-depth: 0
  37. - uses: ./.github/actions/setup-bun
  38. - name: Setup git committer
  39. id: committer
  40. uses: ./.github/actions/setup-git-committer
  41. with:
  42. opencode-app-id: ${{ vars.OPENCODE_APP_ID }}
  43. opencode-app-secret: ${{ secrets.OPENCODE_APP_SECRET }}
  44. - name: Install OpenCode
  45. if: inputs.bump || inputs.version
  46. run: bun i -g opencode-ai
  47. - id: version
  48. run: |
  49. ./script/version.ts
  50. env:
  51. GH_TOKEN: ${{ steps.committer.outputs.token }}
  52. OPENCODE_BUMP: ${{ inputs.bump }}
  53. OPENCODE_VERSION: ${{ inputs.version }}
  54. OPENCODE_API_KEY: ${{ secrets.OPENCODE_API_KEY }}
  55. GH_REPO: ${{ (github.ref_name == 'beta' && 'anomalyco/opencode-beta') || github.repository }}
  56. outputs:
  57. version: ${{ steps.version.outputs.version }}
  58. release: ${{ steps.version.outputs.release }}
  59. tag: ${{ steps.version.outputs.tag }}
  60. repo: ${{ steps.version.outputs.repo }}
  61. build-cli:
  62. needs: version
  63. runs-on: blacksmith-4vcpu-ubuntu-2404
  64. if: github.repository == 'anomalyco/opencode'
  65. steps:
  66. - uses: actions/checkout@v3
  67. with:
  68. fetch-tags: true
  69. - uses: ./.github/actions/setup-bun
  70. - name: Setup git committer
  71. id: committer
  72. uses: ./.github/actions/setup-git-committer
  73. with:
  74. opencode-app-id: ${{ vars.OPENCODE_APP_ID }}
  75. opencode-app-secret: ${{ secrets.OPENCODE_APP_SECRET }}
  76. - name: Build
  77. id: build
  78. run: |
  79. ./packages/opencode/script/build.ts
  80. env:
  81. OPENCODE_VERSION: ${{ needs.version.outputs.version }}
  82. OPENCODE_RELEASE: ${{ needs.version.outputs.release }}
  83. GH_REPO: ${{ needs.version.outputs.repo }}
  84. GH_TOKEN: ${{ steps.committer.outputs.token }}
  85. - uses: actions/upload-artifact@v4
  86. with:
  87. name: opencode-cli
  88. path: packages/opencode/dist
  89. outputs:
  90. version: ${{ needs.version.outputs.version }}
  91. build-tauri:
  92. needs:
  93. - build-cli
  94. - version
  95. continue-on-error: false
  96. strategy:
  97. fail-fast: false
  98. matrix:
  99. settings:
  100. - host: macos-latest
  101. target: x86_64-apple-darwin
  102. - host: macos-latest
  103. target: aarch64-apple-darwin
  104. # github-hosted: blacksmith lacks ARM64 MSVC cross-compilation toolchain
  105. - host: windows-2025
  106. target: aarch64-pc-windows-msvc
  107. - host: blacksmith-4vcpu-windows-2025
  108. target: x86_64-pc-windows-msvc
  109. - host: blacksmith-4vcpu-ubuntu-2404
  110. target: x86_64-unknown-linux-gnu
  111. - host: blacksmith-8vcpu-ubuntu-2404-arm
  112. target: aarch64-unknown-linux-gnu
  113. runs-on: ${{ matrix.settings.host }}
  114. steps:
  115. - uses: actions/checkout@v3
  116. with:
  117. fetch-tags: true
  118. - uses: apple-actions/import-codesign-certs@v2
  119. if: ${{ runner.os == 'macOS' }}
  120. with:
  121. keychain: build
  122. p12-file-base64: ${{ secrets.APPLE_CERTIFICATE }}
  123. p12-password: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }}
  124. - name: Verify Certificate
  125. if: ${{ runner.os == 'macOS' }}
  126. run: |
  127. CERT_INFO=$(security find-identity -v -p codesigning build.keychain | grep "Developer ID Application")
  128. CERT_ID=$(echo "$CERT_INFO" | awk -F'"' '{print $2}')
  129. echo "CERT_ID=$CERT_ID" >> $GITHUB_ENV
  130. echo "Certificate imported."
  131. - name: Setup Apple API Key
  132. if: ${{ runner.os == 'macOS' }}
  133. run: |
  134. echo "${{ secrets.APPLE_API_KEY_PATH }}" > $RUNNER_TEMP/apple-api-key.p8
  135. - uses: ./.github/actions/setup-bun
  136. - uses: actions/setup-node@v4
  137. with:
  138. node-version: "24"
  139. - name: Cache apt packages
  140. if: contains(matrix.settings.host, 'ubuntu')
  141. uses: actions/cache@v4
  142. with:
  143. path: ~/apt-cache
  144. key: ${{ runner.os }}-${{ matrix.settings.target }}-apt-${{ hashFiles('.github/workflows/publish.yml') }}
  145. restore-keys: |
  146. ${{ runner.os }}-${{ matrix.settings.target }}-apt-
  147. - name: install dependencies (ubuntu only)
  148. if: contains(matrix.settings.host, 'ubuntu')
  149. run: |
  150. mkdir -p ~/apt-cache && chmod -R a+rw ~/apt-cache
  151. sudo apt-get update
  152. sudo apt-get install -y --no-install-recommends -o dir::cache::archives="$HOME/apt-cache" libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf
  153. sudo chmod -R a+rw ~/apt-cache
  154. - name: install Rust stable
  155. uses: dtolnay/rust-toolchain@stable
  156. with:
  157. targets: ${{ matrix.settings.target }}
  158. - uses: Swatinem/rust-cache@v2
  159. with:
  160. workspaces: packages/desktop/src-tauri
  161. shared-key: ${{ matrix.settings.target }}
  162. - name: Prepare
  163. run: |
  164. cd packages/desktop
  165. bun ./scripts/prepare.ts
  166. env:
  167. OPENCODE_VERSION: ${{ needs.version.outputs.version }}
  168. GITHUB_TOKEN: ${{ steps.committer.outputs.token }}
  169. RUST_TARGET: ${{ matrix.settings.target }}
  170. GH_TOKEN: ${{ github.token }}
  171. GITHUB_RUN_ID: ${{ github.run_id }}
  172. - name: Resolve tauri portable SHA
  173. if: contains(matrix.settings.host, 'ubuntu')
  174. run: echo "TAURI_PORTABLE_SHA=$(git ls-remote https://github.com/tauri-apps/tauri.git refs/heads/feat/truly-portable-appimage | cut -f1)" >> "$GITHUB_ENV"
  175. # Fixes AppImage build issues, can be removed when https://github.com/tauri-apps/tauri/pull/12491 is released
  176. - name: Install tauri-cli from portable appimage branch
  177. uses: taiki-e/cache-cargo-install-action@v3
  178. if: contains(matrix.settings.host, 'ubuntu')
  179. with:
  180. tool: tauri-cli
  181. git: https://github.com/tauri-apps/tauri
  182. # branch: feat/truly-portable-appimage
  183. rev: ${{ env.TAURI_PORTABLE_SHA }}
  184. - name: Show tauri-cli version
  185. if: contains(matrix.settings.host, 'ubuntu')
  186. run: cargo tauri --version
  187. - name: Setup git committer
  188. id: committer
  189. uses: ./.github/actions/setup-git-committer
  190. with:
  191. opencode-app-id: ${{ vars.OPENCODE_APP_ID }}
  192. opencode-app-secret: ${{ secrets.OPENCODE_APP_SECRET }}
  193. - name: Build and upload artifacts
  194. uses: tauri-apps/tauri-action@390cbe447412ced1303d35abe75287949e43437a
  195. timeout-minutes: 60
  196. with:
  197. projectPath: packages/desktop
  198. uploadWorkflowArtifacts: true
  199. tauriScript: ${{ (contains(matrix.settings.host, 'ubuntu') && 'cargo tauri') || '' }}
  200. args: --target ${{ matrix.settings.target }} --config ${{ (github.ref_name == 'beta' && './src-tauri/tauri.beta.conf.json') || './src-tauri/tauri.prod.conf.json' }} --verbose
  201. updaterJsonPreferNsis: true
  202. releaseId: ${{ needs.version.outputs.release }}
  203. tagName: ${{ needs.version.outputs.tag }}
  204. releaseDraft: true
  205. releaseAssetNamePattern: opencode-desktop-[platform]-[arch][ext]
  206. repo: ${{ (github.ref_name == 'beta' && 'opencode-beta') || '' }}
  207. releaseCommitish: ${{ github.sha }}
  208. env:
  209. GITHUB_TOKEN: ${{ steps.committer.outputs.token }}
  210. TAURI_BUNDLER_NEW_APPIMAGE_FORMAT: true
  211. TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY }}
  212. TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY_PASSWORD }}
  213. APPLE_CERTIFICATE: ${{ secrets.APPLE_CERTIFICATE }}
  214. APPLE_CERTIFICATE_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }}
  215. APPLE_SIGNING_IDENTITY: ${{ env.CERT_ID }}
  216. APPLE_API_ISSUER: ${{ secrets.APPLE_API_ISSUER }}
  217. APPLE_API_KEY: ${{ secrets.APPLE_API_KEY }}
  218. APPLE_API_KEY_PATH: ${{ runner.temp }}/apple-api-key.p8
  219. build-electron:
  220. needs:
  221. - build-cli
  222. - version
  223. continue-on-error: false
  224. strategy:
  225. fail-fast: false
  226. matrix:
  227. settings:
  228. - host: macos-latest
  229. target: x86_64-apple-darwin
  230. platform_flag: --mac --x64
  231. - host: macos-latest
  232. target: aarch64-apple-darwin
  233. platform_flag: --mac --arm64
  234. # github-hosted: blacksmith lacks ARM64 MSVC cross-compilation toolchain
  235. - host: "windows-2025"
  236. target: aarch64-pc-windows-msvc
  237. platform_flag: --win --arm64
  238. - host: "blacksmith-4vcpu-windows-2025"
  239. target: x86_64-pc-windows-msvc
  240. platform_flag: --win
  241. - host: "blacksmith-4vcpu-ubuntu-2404"
  242. target: x86_64-unknown-linux-gnu
  243. platform_flag: --linux
  244. - host: "blacksmith-4vcpu-ubuntu-2404"
  245. target: aarch64-unknown-linux-gnu
  246. platform_flag: --linux
  247. runs-on: ${{ matrix.settings.host }}
  248. # if: github.ref_name == 'beta'
  249. steps:
  250. - uses: actions/checkout@v3
  251. - uses: apple-actions/import-codesign-certs@v2
  252. if: runner.os == 'macOS'
  253. with:
  254. keychain: build
  255. p12-file-base64: ${{ secrets.APPLE_CERTIFICATE }}
  256. p12-password: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }}
  257. - name: Setup Apple API Key
  258. if: runner.os == 'macOS'
  259. run: echo "${{ secrets.APPLE_API_KEY_PATH }}" > $RUNNER_TEMP/apple-api-key.p8
  260. - uses: ./.github/actions/setup-bun
  261. - uses: actions/setup-node@v4
  262. with:
  263. node-version: "24"
  264. - name: Cache apt packages
  265. if: contains(matrix.settings.host, 'ubuntu')
  266. uses: actions/cache@v4
  267. with:
  268. path: ~/apt-cache
  269. key: ${{ runner.os }}-${{ matrix.settings.target }}-apt-electron-${{ hashFiles('.github/workflows/publish.yml') }}
  270. restore-keys: |
  271. ${{ runner.os }}-${{ matrix.settings.target }}-apt-electron-
  272. - name: Install dependencies (ubuntu only)
  273. if: contains(matrix.settings.host, 'ubuntu')
  274. run: |
  275. mkdir -p ~/apt-cache && chmod -R a+rw ~/apt-cache
  276. sudo apt-get update
  277. sudo apt-get install -y --no-install-recommends -o dir::cache::archives="$HOME/apt-cache" rpm
  278. sudo chmod -R a+rw ~/apt-cache
  279. - name: Setup git committer
  280. id: committer
  281. uses: ./.github/actions/setup-git-committer
  282. with:
  283. opencode-app-id: ${{ vars.OPENCODE_APP_ID }}
  284. opencode-app-secret: ${{ secrets.OPENCODE_APP_SECRET }}
  285. - name: Prepare
  286. run: bun ./scripts/prepare.ts
  287. working-directory: packages/desktop-electron
  288. env:
  289. OPENCODE_VERSION: ${{ needs.version.outputs.version }}
  290. OPENCODE_CHANNEL: ${{ (github.ref_name == 'beta' && 'beta') || 'prod' }}
  291. RUST_TARGET: ${{ matrix.settings.target }}
  292. GH_TOKEN: ${{ github.token }}
  293. GITHUB_RUN_ID: ${{ github.run_id }}
  294. - name: Build
  295. run: bun run build
  296. working-directory: packages/desktop-electron
  297. env:
  298. OPENCODE_CHANNEL: ${{ (github.ref_name == 'beta' && 'beta') || 'prod' }}
  299. - name: Package and publish
  300. if: needs.version.outputs.release
  301. run: npx electron-builder ${{ matrix.settings.platform_flag }} --publish always --config electron-builder.config.ts
  302. working-directory: packages/desktop-electron
  303. timeout-minutes: 60
  304. env:
  305. OPENCODE_CHANNEL: ${{ (github.ref_name == 'beta' && 'beta') || 'prod' }}
  306. GH_TOKEN: ${{ steps.committer.outputs.token }}
  307. CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }}
  308. CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }}
  309. APPLE_API_KEY: ${{ runner.temp }}/apple-api-key.p8
  310. APPLE_API_KEY_ID: ${{ secrets.APPLE_API_KEY }}
  311. APPLE_API_ISSUER: ${{ secrets.APPLE_API_ISSUER }}
  312. - name: Package (no publish)
  313. if: ${{ !needs.version.outputs.release }}
  314. run: npx electron-builder ${{ matrix.settings.platform_flag }} --publish never --config electron-builder.config.ts
  315. working-directory: packages/desktop-electron
  316. timeout-minutes: 60
  317. env:
  318. OPENCODE_CHANNEL: ${{ (github.ref_name == 'beta' && 'beta') || 'prod' }}
  319. - uses: actions/upload-artifact@v4
  320. with:
  321. name: opencode-electron-${{ matrix.settings.target }}
  322. path: packages/desktop-electron/dist/*
  323. - uses: actions/upload-artifact@v4
  324. if: needs.version.outputs.release
  325. with:
  326. name: latest-yml-${{ matrix.settings.target }}
  327. path: packages/desktop-electron/dist/latest*.yml
  328. publish:
  329. needs:
  330. - version
  331. - build-cli
  332. - build-tauri
  333. - build-electron
  334. runs-on: blacksmith-4vcpu-ubuntu-2404
  335. steps:
  336. - uses: actions/checkout@v3
  337. - uses: ./.github/actions/setup-bun
  338. - name: Login to GitHub Container Registry
  339. uses: docker/login-action@v3
  340. with:
  341. registry: ghcr.io
  342. username: ${{ github.repository_owner }}
  343. password: ${{ secrets.GITHUB_TOKEN }}
  344. - name: Set up QEMU
  345. uses: docker/setup-qemu-action@v3
  346. - name: Set up Docker Buildx
  347. uses: docker/setup-buildx-action@v3
  348. - uses: actions/setup-node@v4
  349. with:
  350. node-version: "24"
  351. registry-url: "https://registry.npmjs.org"
  352. - name: Setup git committer
  353. id: committer
  354. uses: ./.github/actions/setup-git-committer
  355. with:
  356. opencode-app-id: ${{ vars.OPENCODE_APP_ID }}
  357. opencode-app-secret: ${{ secrets.OPENCODE_APP_SECRET }}
  358. - uses: actions/download-artifact@v4
  359. with:
  360. name: opencode-cli
  361. path: packages/opencode/dist
  362. - uses: actions/download-artifact@v4
  363. if: needs.version.outputs.release
  364. with:
  365. pattern: latest-yml-*
  366. path: /tmp/latest-yml
  367. - name: Cache apt packages (AUR)
  368. uses: actions/cache@v4
  369. with:
  370. path: /var/cache/apt/archives
  371. key: ${{ runner.os }}-apt-aur-${{ hashFiles('.github/workflows/publish.yml') }}
  372. restore-keys: |
  373. ${{ runner.os }}-apt-aur-
  374. - name: Setup SSH for AUR
  375. run: |
  376. sudo apt-get update
  377. sudo apt-get install -y pacman-package-manager
  378. mkdir -p ~/.ssh
  379. echo "${{ secrets.AUR_KEY }}" > ~/.ssh/id_rsa
  380. chmod 600 ~/.ssh/id_rsa
  381. git config --global user.email "[email protected]"
  382. git config --global user.name "opencode"
  383. ssh-keyscan -H aur.archlinux.org >> ~/.ssh/known_hosts || true
  384. - run: ./script/publish.ts
  385. env:
  386. OPENCODE_VERSION: ${{ needs.version.outputs.version }}
  387. OPENCODE_RELEASE: ${{ needs.version.outputs.release }}
  388. AUR_KEY: ${{ secrets.AUR_KEY }}
  389. GITHUB_TOKEN: ${{ steps.committer.outputs.token }}
  390. GH_REPO: ${{ needs.version.outputs.repo }}
  391. NPM_CONFIG_PROVENANCE: false
  392. LATEST_YML_DIR: /tmp/latest-yml