publish.ts 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. #!/usr/bin/env bun
  2. import { Script } from "@opencode-ai/script"
  3. import { $ } from "bun"
  4. import { fileURLToPath } from "url"
  5. console.log("=== publishing ===\n")
  6. const tag = `v${Script.version}`
  7. const pkgjsons = await Array.fromAsync(
  8. new Bun.Glob("**/package.json").scan({
  9. absolute: true,
  10. }),
  11. ).then((arr) => arr.filter((x) => !x.includes("node_modules") && !x.includes("dist")))
  12. const extensionToml = fileURLToPath(new URL("../packages/extensions/zed/extension.toml", import.meta.url))
  13. const publishPackageJsons = ["packages/plugin/package.json", "packages/sdk/js/package.json"]
  14. async function hasChanges() {
  15. return (await $`git diff --quiet && git diff --cached --quiet`.nothrow()).exitCode !== 0
  16. }
  17. async function hasPublishPackageJsonChanges() {
  18. if ((await $`git diff --quiet -- ${publishPackageJsons}`.nothrow()).exitCode !== 0) return true
  19. return (await $`git diff --cached --quiet -- ${publishPackageJsons}`.nothrow()).exitCode !== 0
  20. }
  21. async function logPublishPackageJsonChanges() {
  22. await $`git status --short -- ${publishPackageJsons}`
  23. await $`git diff -- ${publishPackageJsons}`
  24. await $`git diff --cached -- ${publishPackageJsons}`
  25. }
  26. async function releaseTagExists() {
  27. return (await $`git rev-parse -q --verify refs/tags/${tag}`.nothrow()).exitCode === 0
  28. }
  29. async function prepareReleaseFiles() {
  30. for (const file of pkgjsons) {
  31. let pkg = await Bun.file(file).text()
  32. pkg = pkg.replaceAll(/"version": "[^"]+"/g, `"version": "${Script.version}"`)
  33. console.log("updated:", file)
  34. await Bun.file(file).write(pkg)
  35. }
  36. let toml = await Bun.file(extensionToml).text()
  37. toml = toml.replace(/^version = "[^"]+"/m, `version = "${Script.version}"`)
  38. toml = toml.replaceAll(/releases\/download\/v[^/]+\//g, `releases/download/v${Script.version}/`)
  39. console.log("updated:", extensionToml)
  40. await Bun.file(extensionToml).write(toml)
  41. await $`bun install`
  42. await $`./packages/sdk/js/script/build.ts`
  43. }
  44. if (Script.release && !Script.preview) {
  45. await $`git fetch origin --tags`
  46. await $`git switch --detach`
  47. }
  48. await prepareReleaseFiles()
  49. if (Script.release && !Script.preview) {
  50. if (await releaseTagExists()) {
  51. console.log(`release tag ${tag} already exists, skipping tag creation`)
  52. } else {
  53. await $`git commit -am "release: ${tag}"`
  54. await $`git tag ${tag}`
  55. await $`git push origin refs/tags/${tag} --no-verify`
  56. await new Promise((resolve) => setTimeout(resolve, 5_000))
  57. }
  58. }
  59. console.log("\n=== cli ===\n")
  60. await import(`../packages/opencode/script/publish.ts`)
  61. console.log("\n=== sdk ===\n")
  62. await import(`../packages/sdk/js/script/publish.ts`)
  63. console.log("\n=== plugin ===\n")
  64. await import(`../packages/plugin/script/publish.ts`)
  65. if (Script.release) {
  66. await import(`../packages/desktop/scripts/finalize-latest-json.ts`)
  67. await import(`../packages/desktop-electron/scripts/finalize-latest-yml.ts`)
  68. }
  69. if (Script.release && !Script.preview) {
  70. await $`git fetch origin`
  71. if (await hasPublishPackageJsonChanges()) {
  72. console.error("publish scripts left package.json changes before syncing dev")
  73. await logPublishPackageJsonChanges()
  74. throw new Error("packages/plugin/package.json or packages/sdk/js/package.json changed during publish")
  75. }
  76. await $`git checkout -B dev origin/dev`
  77. await prepareReleaseFiles()
  78. if (await hasChanges()) {
  79. await $`git commit -am "sync release versions for v${Script.version}"`
  80. await $`git push origin HEAD:dev --no-verify`
  81. } else {
  82. console.log(`dev already synced for ${tag}`)
  83. }
  84. }
  85. if (Script.release) {
  86. await $`gh release edit ${tag} --draft=false --repo ${process.env.GH_REPO}`
  87. }
  88. const dir = fileURLToPath(new URL("..", import.meta.url))
  89. process.chdir(dir)