preload.ts 3.4 KB

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