session-messages.test.ts 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. import { afterEach, describe, expect, test } from "bun:test"
  2. import { Instance } from "../../src/project/instance"
  3. import { Server } from "../../src/server/server"
  4. import { Session } from "../../src/session"
  5. import { MessageV2 } from "../../src/session/message-v2"
  6. import { MessageID, PartID, type SessionID } from "../../src/session/schema"
  7. import { Log } from "../../src/util/log"
  8. import { tmpdir } from "../fixture/fixture"
  9. Log.init({ print: false })
  10. afterEach(async () => {
  11. await Instance.disposeAll()
  12. })
  13. async function withoutWatcher<T>(fn: () => Promise<T>) {
  14. if (process.platform !== "win32") return fn()
  15. const prev = process.env.KILO_EXPERIMENTAL_DISABLE_FILEWATCHER
  16. process.env.KILO_EXPERIMENTAL_DISABLE_FILEWATCHER = "true"
  17. try {
  18. return await fn()
  19. } finally {
  20. if (prev === undefined) delete process.env.KILO_EXPERIMENTAL_DISABLE_FILEWATCHER
  21. else process.env.KILO_EXPERIMENTAL_DISABLE_FILEWATCHER = prev
  22. }
  23. }
  24. async function fill(sessionID: SessionID, count: number, time = (i: number) => Date.now() + i) {
  25. const ids = [] as MessageID[]
  26. for (let i = 0; i < count; i++) {
  27. const id = MessageID.ascending()
  28. ids.push(id)
  29. await Session.updateMessage({
  30. id,
  31. sessionID,
  32. role: "user",
  33. time: { created: time(i) },
  34. agent: "test",
  35. model: { providerID: "test", modelID: "test" },
  36. tools: {},
  37. mode: "",
  38. } as unknown as MessageV2.Info)
  39. await Session.updatePart({
  40. id: PartID.ascending(),
  41. sessionID,
  42. messageID: id,
  43. type: "text",
  44. text: `m${i}`,
  45. })
  46. }
  47. return ids
  48. }
  49. describe("session messages endpoint", () => {
  50. test("returns cursor headers for older pages", async () => {
  51. await using tmp = await tmpdir({ git: true })
  52. await withoutWatcher(() =>
  53. Instance.provide({
  54. directory: tmp.path,
  55. fn: async () => {
  56. const session = await Session.create({})
  57. const ids = await fill(session.id, 5)
  58. const app = Server.Default().app
  59. const a = await app.request(`/session/${session.id}/message?limit=2`)
  60. expect(a.status).toBe(200)
  61. const aBody = (await a.json()) as MessageV2.WithParts[]
  62. expect(aBody.map((item) => item.info.id)).toEqual(ids.slice(-2))
  63. const cursor = a.headers.get("x-next-cursor")
  64. expect(cursor).toBeTruthy()
  65. expect(a.headers.get("link")).toContain('rel="next"')
  66. const b = await app.request(`/session/${session.id}/message?limit=2&before=${encodeURIComponent(cursor!)}`)
  67. expect(b.status).toBe(200)
  68. const bBody = (await b.json()) as MessageV2.WithParts[]
  69. expect(bBody.map((item) => item.info.id)).toEqual(ids.slice(-4, -2))
  70. await Session.remove(session.id)
  71. },
  72. }),
  73. )
  74. })
  75. test("keeps full-history responses when limit is omitted", async () => {
  76. await using tmp = await tmpdir({ git: true })
  77. await withoutWatcher(() =>
  78. Instance.provide({
  79. directory: tmp.path,
  80. fn: async () => {
  81. const session = await Session.create({})
  82. const ids = await fill(session.id, 3)
  83. const app = Server.Default().app
  84. const res = await app.request(`/session/${session.id}/message`)
  85. expect(res.status).toBe(200)
  86. const body = (await res.json()) as MessageV2.WithParts[]
  87. expect(body.map((item) => item.info.id)).toEqual(ids)
  88. await Session.remove(session.id)
  89. },
  90. }),
  91. )
  92. })
  93. test("rejects invalid cursors and missing sessions", async () => {
  94. await using tmp = await tmpdir({ git: true })
  95. await withoutWatcher(() =>
  96. Instance.provide({
  97. directory: tmp.path,
  98. fn: async () => {
  99. const session = await Session.create({})
  100. const app = Server.Default().app
  101. const bad = await app.request(`/session/${session.id}/message?limit=2&before=bad`)
  102. expect(bad.status).toBe(400)
  103. const miss = await app.request(`/session/ses_missing/message?limit=2`)
  104. expect(miss.status).toBe(404)
  105. await Session.remove(session.id)
  106. },
  107. }),
  108. )
  109. })
  110. test("does not truncate large legacy limit requests", async () => {
  111. await using tmp = await tmpdir({ git: true })
  112. await withoutWatcher(() =>
  113. Instance.provide({
  114. directory: tmp.path,
  115. fn: async () => {
  116. const session = await Session.create({})
  117. await fill(session.id, 520)
  118. const app = Server.Default().app
  119. const res = await app.request(`/session/${session.id}/message?limit=510`)
  120. expect(res.status).toBe(200)
  121. const body = (await res.json()) as MessageV2.WithParts[]
  122. expect(body).toHaveLength(510)
  123. await Session.remove(session.id)
  124. },
  125. }),
  126. )
  127. })
  128. })
  129. describe("session.prompt_async error handling", () => {
  130. test("prompt_async route has error handler for detached prompt call", async () => {
  131. const src = await Bun.file(new URL("../../src/server/routes/session.ts", import.meta.url)).text()
  132. const start = src.indexOf('"/:sessionID/prompt_async"')
  133. const end = src.indexOf('"/:sessionID/command"', start)
  134. expect(start).toBeGreaterThan(-1)
  135. expect(end).toBeGreaterThan(start)
  136. const route = src.slice(start, end)
  137. expect(route).toContain(".catch(")
  138. expect(route).toContain("Bus.publish(Session.Event.Error")
  139. })
  140. })