preload.ts 3.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. // IMPORTANT: Set env vars BEFORE any imports from src/ directory
  2. // xdg-basedir reads env vars at import time, so we must set these first
  3. import os from "os"
  4. import path from "path"
  5. import fs from "fs/promises"
  6. import { afterAll } from "bun:test"
  7. // Set XDG env vars FIRST, before any src/ imports
  8. const dir = path.join(os.tmpdir(), "opencode-test-data-" + process.pid)
  9. await fs.mkdir(dir, { recursive: true })
  10. afterAll(async () => {
  11. const { Database } = await import("../src/storage/db")
  12. Database.close()
  13. const busy = (error: unknown) =>
  14. typeof error === "object" && error !== null && "code" in error && error.code === "EBUSY"
  15. const rm = async (left: number): Promise<void> => {
  16. Bun.gc(true)
  17. await Bun.sleep(100)
  18. return fs.rm(dir, { recursive: true, force: true }).catch((error) => {
  19. if (!busy(error)) throw error
  20. if (left <= 1) throw error
  21. return rm(left - 1)
  22. })
  23. }
  24. // Windows can keep SQLite WAL handles alive until GC finalizers run, so we
  25. // force GC and retry teardown to avoid flaky EBUSY in test cleanup.
  26. await rm(30)
  27. })
  28. process.env["XDG_DATA_HOME"] = path.join(dir, "share")
  29. process.env["XDG_CACHE_HOME"] = path.join(dir, "cache")
  30. process.env["XDG_CONFIG_HOME"] = path.join(dir, "config")
  31. process.env["XDG_STATE_HOME"] = path.join(dir, "state")
  32. process.env["OPENCODE_MODELS_PATH"] = path.join(import.meta.dir, "tool", "fixtures", "models-api.json")
  33. // Set test home directory to isolate tests from user's actual home directory
  34. // This prevents tests from picking up real user configs/skills from ~/.claude/skills
  35. const testHome = path.join(dir, "home")
  36. await fs.mkdir(testHome, { recursive: true })
  37. process.env["OPENCODE_TEST_HOME"] = testHome
  38. // Set test managed config directory to isolate tests from system managed settings
  39. const testManagedConfigDir = path.join(dir, "managed")
  40. process.env["OPENCODE_TEST_MANAGED_CONFIG_DIR"] = testManagedConfigDir
  41. // Write the cache version file to prevent global/index.ts from clearing the cache
  42. const cacheDir = path.join(dir, "cache", "opencode")
  43. await fs.mkdir(cacheDir, { recursive: true })
  44. await fs.writeFile(path.join(cacheDir, "version"), "14")
  45. // Clear provider env vars to ensure clean test state
  46. delete process.env["ANTHROPIC_API_KEY"]
  47. delete process.env["OPENAI_API_KEY"]
  48. delete process.env["GOOGLE_API_KEY"]
  49. delete process.env["GOOGLE_GENERATIVE_AI_API_KEY"]
  50. delete process.env["AZURE_OPENAI_API_KEY"]
  51. delete process.env["AWS_ACCESS_KEY_ID"]
  52. delete process.env["AWS_PROFILE"]
  53. delete process.env["AWS_REGION"]
  54. delete process.env["AWS_BEARER_TOKEN_BEDROCK"]
  55. delete process.env["OPENROUTER_API_KEY"]
  56. delete process.env["GROQ_API_KEY"]
  57. delete process.env["MISTRAL_API_KEY"]
  58. delete process.env["PERPLEXITY_API_KEY"]
  59. delete process.env["TOGETHER_API_KEY"]
  60. delete process.env["XAI_API_KEY"]
  61. delete process.env["DEEPSEEK_API_KEY"]
  62. delete process.env["FIREWORKS_API_KEY"]
  63. delete process.env["CEREBRAS_API_KEY"]
  64. delete process.env["SAMBANOVA_API_KEY"]
  65. // Now safe to import from src/
  66. const { Log } = await import("../src/util/log")
  67. Log.init({
  68. print: false,
  69. dev: true,
  70. level: "DEBUG",
  71. })