e2e-local.ts 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. import fs from "node:fs/promises"
  2. import net from "node:net"
  3. import os from "node:os"
  4. import path from "node:path"
  5. async function freePort() {
  6. return await new Promise<number>((resolve, reject) => {
  7. const server = net.createServer()
  8. server.once("error", reject)
  9. server.listen(0, () => {
  10. const address = server.address()
  11. if (!address || typeof address === "string") {
  12. server.close(() => reject(new Error("Failed to acquire a free port")))
  13. return
  14. }
  15. server.close((err) => {
  16. if (err) {
  17. reject(err)
  18. return
  19. }
  20. resolve(address.port)
  21. })
  22. })
  23. })
  24. }
  25. async function waitForHealth(url: string) {
  26. const timeout = Date.now() + 120_000
  27. const errors: string[] = []
  28. while (Date.now() < timeout) {
  29. const result = await fetch(url)
  30. .then((r) => ({ ok: r.ok, error: undefined }))
  31. .catch((error) => ({
  32. ok: false,
  33. error: error instanceof Error ? error.message : String(error),
  34. }))
  35. if (result.ok) return
  36. if (result.error) errors.push(result.error)
  37. await new Promise((r) => setTimeout(r, 250))
  38. }
  39. const last = errors.length ? ` (last error: ${errors[errors.length - 1]})` : ""
  40. throw new Error(`Timed out waiting for server health: ${url}${last}`)
  41. }
  42. const appDir = process.cwd()
  43. const repoDir = path.resolve(appDir, "../..")
  44. const opencodeDir = path.join(repoDir, "packages", "opencode")
  45. const extraArgs = (() => {
  46. const args = process.argv.slice(2)
  47. if (args[0] === "--") return args.slice(1)
  48. return args
  49. })()
  50. const [serverPort, webPort] = await Promise.all([freePort(), freePort()])
  51. const sandbox = await fs.mkdtemp(path.join(os.tmpdir(), "opencode-e2e-"))
  52. const keepSandbox = process.env.OPENCODE_E2E_KEEP_SANDBOX === "1"
  53. const serverEnv = {
  54. ...process.env,
  55. OPENCODE_DISABLE_SHARE: process.env.OPENCODE_DISABLE_SHARE ?? "true",
  56. OPENCODE_DISABLE_LSP_DOWNLOAD: "true",
  57. OPENCODE_DISABLE_DEFAULT_PLUGINS: "true",
  58. OPENCODE_EXPERIMENTAL_DISABLE_FILEWATCHER: "true",
  59. OPENCODE_TEST_HOME: path.join(sandbox, "home"),
  60. XDG_DATA_HOME: path.join(sandbox, "share"),
  61. XDG_CACHE_HOME: path.join(sandbox, "cache"),
  62. XDG_CONFIG_HOME: path.join(sandbox, "config"),
  63. XDG_STATE_HOME: path.join(sandbox, "state"),
  64. OPENCODE_E2E_PROJECT_DIR: repoDir,
  65. OPENCODE_E2E_SESSION_TITLE: "E2E Session",
  66. OPENCODE_E2E_MESSAGE: "Seeded for UI e2e",
  67. OPENCODE_E2E_MODEL: "opencode/gpt-5-nano",
  68. OPENCODE_CLIENT: "app",
  69. } satisfies Record<string, string>
  70. const runnerEnv = {
  71. ...serverEnv,
  72. PLAYWRIGHT_SERVER_HOST: "127.0.0.1",
  73. PLAYWRIGHT_SERVER_PORT: String(serverPort),
  74. VITE_OPENCODE_SERVER_HOST: "127.0.0.1",
  75. VITE_OPENCODE_SERVER_PORT: String(serverPort),
  76. PLAYWRIGHT_PORT: String(webPort),
  77. } satisfies Record<string, string>
  78. let seed: ReturnType<typeof Bun.spawn> | undefined
  79. let runner: ReturnType<typeof Bun.spawn> | undefined
  80. let server: { stop: () => Promise<void> | void } | undefined
  81. let inst: { Instance: { disposeAll: () => Promise<void> | void } } | undefined
  82. let cleaned = false
  83. let internalError = false
  84. const cleanup = async () => {
  85. if (cleaned) return
  86. cleaned = true
  87. if (seed && seed.exitCode === null) seed.kill("SIGTERM")
  88. if (runner && runner.exitCode === null) runner.kill("SIGTERM")
  89. const jobs = [
  90. inst?.Instance.disposeAll(),
  91. server?.stop(),
  92. keepSandbox ? undefined : fs.rm(sandbox, { recursive: true, force: true }),
  93. ].filter(Boolean)
  94. await Promise.allSettled(jobs)
  95. }
  96. const shutdown = (code: number, reason: string) => {
  97. process.exitCode = code
  98. void cleanup().finally(() => {
  99. console.error(`e2e-local shutdown: ${reason}`)
  100. process.exit(code)
  101. })
  102. }
  103. const reportInternalError = (reason: string, error: unknown) => {
  104. internalError = true
  105. console.error(`e2e-local internal error: ${reason}`)
  106. console.error(error)
  107. }
  108. process.once("SIGINT", () => shutdown(130, "SIGINT"))
  109. process.once("SIGTERM", () => shutdown(143, "SIGTERM"))
  110. process.once("SIGHUP", () => shutdown(129, "SIGHUP"))
  111. process.once("uncaughtException", (error) => {
  112. reportInternalError("uncaughtException", error)
  113. })
  114. process.once("unhandledRejection", (error) => {
  115. reportInternalError("unhandledRejection", error)
  116. })
  117. let code = 1
  118. try {
  119. seed = Bun.spawn(["bun", "script/seed-e2e.ts"], {
  120. cwd: opencodeDir,
  121. env: serverEnv,
  122. stdout: "inherit",
  123. stderr: "inherit",
  124. })
  125. const seedExit = await seed.exited
  126. if (seedExit !== 0) {
  127. code = seedExit
  128. } else {
  129. Object.assign(process.env, serverEnv)
  130. process.env.AGENT = "1"
  131. process.env.OPENCODE = "1"
  132. const log = await import("../../opencode/src/util/log")
  133. const install = await import("../../opencode/src/installation")
  134. await log.Log.init({
  135. print: true,
  136. dev: install.Installation.isLocal(),
  137. level: "WARN",
  138. })
  139. const servermod = await import("../../opencode/src/server/server")
  140. inst = await import("../../opencode/src/project/instance")
  141. server = servermod.Server.listen({ port: serverPort, hostname: "127.0.0.1" })
  142. console.log(`opencode server listening on http://127.0.0.1:${serverPort}`)
  143. await waitForHealth(`http://127.0.0.1:${serverPort}/global/health`)
  144. runner = Bun.spawn(["bun", "test:e2e", ...extraArgs], {
  145. cwd: appDir,
  146. env: runnerEnv,
  147. stdout: "inherit",
  148. stderr: "inherit",
  149. })
  150. code = await runner.exited
  151. }
  152. } catch (error) {
  153. console.error(error)
  154. code = 1
  155. } finally {
  156. await cleanup()
  157. }
  158. if (code === 0 && internalError) code = 1
  159. process.exit(code)