index.ts 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846
  1. import path from "path"
  2. import { App } from "../app/app"
  3. import { Identifier } from "../id/id"
  4. import { Storage } from "../storage/storage"
  5. import { Log } from "../util/log"
  6. import {
  7. generateText,
  8. LoadAPIKeyError,
  9. convertToCoreMessages,
  10. streamText,
  11. tool,
  12. type Tool as AITool,
  13. type LanguageModelUsage,
  14. type CoreMessage,
  15. type UIMessage,
  16. type ProviderMetadata,
  17. } from "ai"
  18. import { z, ZodSchema } from "zod"
  19. import { Decimal } from "decimal.js"
  20. import PROMPT_INITIALIZE from "../session/prompt/initialize.txt"
  21. import { Share } from "../share/share"
  22. import { Message } from "./message"
  23. import { Bus } from "../bus"
  24. import { Provider } from "../provider/provider"
  25. import { MCP } from "../mcp"
  26. import { NamedError } from "../util/error"
  27. import type { Tool } from "../tool/tool"
  28. import { SystemPrompt } from "./system"
  29. import { Flag } from "../flag/flag"
  30. import type { ModelsDev } from "../provider/models"
  31. import { GlobalConfig } from "../global/config"
  32. export namespace Session {
  33. const log = Log.create({ service: "session" })
  34. export const Info = z
  35. .object({
  36. id: Identifier.schema("session"),
  37. parentID: Identifier.schema("session").optional(),
  38. share: z
  39. .object({
  40. secret: z.string(),
  41. url: z.string(),
  42. })
  43. .optional(),
  44. title: z.string(),
  45. time: z.object({
  46. created: z.number(),
  47. updated: z.number(),
  48. }),
  49. })
  50. .openapi({
  51. ref: "session.info",
  52. })
  53. export type Info = z.output<typeof Info>
  54. export const Event = {
  55. Updated: Bus.event(
  56. "session.updated",
  57. z.object({
  58. info: Info,
  59. }),
  60. ),
  61. Error: Bus.event(
  62. "session.error",
  63. z.object({
  64. error: Message.Info.shape.metadata.shape.error,
  65. }),
  66. ),
  67. }
  68. const state = App.state("session", () => {
  69. const sessions = new Map<string, Info>()
  70. const messages = new Map<string, Message.Info[]>()
  71. return {
  72. sessions,
  73. messages,
  74. }
  75. })
  76. export async function create(parentID?: string) {
  77. const result: Info = {
  78. id: Identifier.descending("session"),
  79. parentID,
  80. title:
  81. (parentID ? "Child session - " : "New Session - ") +
  82. new Date().toISOString(),
  83. time: {
  84. created: Date.now(),
  85. updated: Date.now(),
  86. },
  87. }
  88. log.info("created", result)
  89. state().sessions.set(result.id, result)
  90. await Storage.writeJSON("session/info/" + result.id, result)
  91. const cfg = await GlobalConfig.get()
  92. if (!result.parentID && (Flag.OPENCODE_AUTO_SHARE || cfg.autoshare))
  93. share(result.id).then((share) => {
  94. update(result.id, (draft) => {
  95. draft.share = share
  96. })
  97. })
  98. Bus.publish(Event.Updated, {
  99. info: result,
  100. })
  101. return result
  102. }
  103. export async function get(id: string) {
  104. const result = state().sessions.get(id)
  105. if (result) {
  106. return result
  107. }
  108. const read = await Storage.readJSON<Info>("session/info/" + id)
  109. state().sessions.set(id, read)
  110. return read as Info
  111. }
  112. export async function share(id: string) {
  113. const session = await get(id)
  114. if (session.share) return session.share
  115. const share = await Share.create(id)
  116. await update(id, (draft) => {
  117. draft.share = share
  118. })
  119. for (const msg of await messages(id)) {
  120. await Share.sync("session/message/" + id + "/" + msg.id, msg)
  121. }
  122. return share
  123. }
  124. export async function update(id: string, editor: (session: Info) => void) {
  125. const { sessions } = state()
  126. const session = await get(id)
  127. if (!session) return
  128. editor(session)
  129. session.time.updated = Date.now()
  130. sessions.set(id, session)
  131. await Storage.writeJSON("session/info/" + id, session)
  132. Bus.publish(Event.Updated, {
  133. info: session,
  134. })
  135. return session
  136. }
  137. export async function messages(sessionID: string) {
  138. const result = [] as Message.Info[]
  139. const list = Storage.list("session/message/" + sessionID)
  140. for await (const p of list) {
  141. const read = await Storage.readJSON<Message.Info>(p)
  142. result.push(read)
  143. }
  144. result.sort((a, b) => (a.id > b.id ? 1 : -1))
  145. return result
  146. }
  147. export async function getMessage(sessionID: string, messageID: string) {
  148. return Storage.readJSON<Message.Info>(
  149. "session/message/" + sessionID + "/" + messageID,
  150. )
  151. }
  152. export async function* list() {
  153. for await (const item of Storage.list("session/info")) {
  154. const sessionID = path.basename(item, ".json")
  155. yield get(sessionID)
  156. }
  157. }
  158. export function abort(sessionID: string) {
  159. const controller = pending.get(sessionID)
  160. if (!controller) return false
  161. controller.abort()
  162. pending.delete(sessionID)
  163. return true
  164. }
  165. async function updateMessage(msg: Message.Info) {
  166. await Storage.writeJSON(
  167. "session/message/" + msg.metadata.sessionID + "/" + msg.id,
  168. msg,
  169. )
  170. Bus.publish(Message.Event.Updated, {
  171. info: msg,
  172. })
  173. }
  174. export async function chat(input: {
  175. sessionID: string
  176. providerID: string
  177. modelID: string
  178. parts: Message.Part[]
  179. system?: string[]
  180. tools?: Tool.Info[]
  181. }) {
  182. const l = log.clone().tag("session", input.sessionID)
  183. l.info("chatting")
  184. const model = await Provider.getModel(input.providerID, input.modelID)
  185. let msgs = await messages(input.sessionID)
  186. const previous = msgs.at(-1)
  187. // auto summarize if too long
  188. if (previous?.metadata.assistant) {
  189. const tokens =
  190. previous.metadata.assistant.tokens.input +
  191. previous.metadata.assistant.tokens.cache.read +
  192. previous.metadata.assistant.tokens.cache.write +
  193. previous.metadata.assistant.tokens.output
  194. if (
  195. model.info.limit.context &&
  196. tokens >
  197. (model.info.limit.context - (model.info.limit.output ?? 0)) * 0.9
  198. ) {
  199. await summarize({
  200. sessionID: input.sessionID,
  201. providerID: input.providerID,
  202. modelID: input.modelID,
  203. })
  204. return chat(input)
  205. }
  206. }
  207. using abort = lock(input.sessionID)
  208. const lastSummary = msgs.findLast(
  209. (msg) => msg.metadata.assistant?.summary === true,
  210. )
  211. if (lastSummary) msgs = msgs.filter((msg) => msg.id >= lastSummary.id)
  212. const app = App.info()
  213. const session = await get(input.sessionID)
  214. if (msgs.length === 0 && !session.parentID) {
  215. generateText({
  216. maxTokens: input.providerID === "google" ? 1024 : 20,
  217. messages: [
  218. ...SystemPrompt.title(input.providerID).map(
  219. (x): CoreMessage => ({
  220. role: "system",
  221. content: x,
  222. providerOptions: {
  223. ...(input.providerID === "anthropic"
  224. ? {
  225. anthropic: {
  226. cacheControl: { type: "ephemeral" },
  227. },
  228. }
  229. : {}),
  230. },
  231. }),
  232. ),
  233. ...convertToCoreMessages([
  234. {
  235. role: "user",
  236. content: "",
  237. parts: toParts(input.parts),
  238. },
  239. ]),
  240. ],
  241. model: model.language,
  242. })
  243. .then((result) => {
  244. if (result.text)
  245. return Session.update(input.sessionID, (draft) => {
  246. draft.title = result.text
  247. })
  248. })
  249. .catch(() => {})
  250. }
  251. const msg: Message.Info = {
  252. role: "user",
  253. id: Identifier.ascending("message"),
  254. parts: input.parts,
  255. metadata: {
  256. time: {
  257. created: Date.now(),
  258. },
  259. sessionID: input.sessionID,
  260. tool: {},
  261. },
  262. }
  263. await updateMessage(msg)
  264. msgs.push(msg)
  265. const system = input.system ?? SystemPrompt.provider(input.providerID)
  266. system.push(...(await SystemPrompt.environment()))
  267. system.push(...(await SystemPrompt.custom()))
  268. const next: Message.Info = {
  269. id: Identifier.ascending("message"),
  270. role: "assistant",
  271. parts: [],
  272. metadata: {
  273. assistant: {
  274. system,
  275. path: {
  276. cwd: app.path.cwd,
  277. root: app.path.root,
  278. },
  279. cost: 0,
  280. tokens: {
  281. input: 0,
  282. output: 0,
  283. reasoning: 0,
  284. cache: { read: 0, write: 0 },
  285. },
  286. modelID: input.modelID,
  287. providerID: input.providerID,
  288. },
  289. time: {
  290. created: Date.now(),
  291. },
  292. sessionID: input.sessionID,
  293. tool: {},
  294. },
  295. }
  296. await updateMessage(next)
  297. const tools: Record<string, AITool> = {}
  298. for (const item of await Provider.tools(input.providerID)) {
  299. tools[item.id.replaceAll(".", "_")] = tool({
  300. id: item.id as any,
  301. description: item.description,
  302. parameters: item.parameters as ZodSchema,
  303. async execute(args, opts) {
  304. const start = Date.now()
  305. try {
  306. const result = await item.execute(args, {
  307. sessionID: input.sessionID,
  308. abort: abort.signal,
  309. messageID: next.id,
  310. })
  311. next.metadata!.tool![opts.toolCallId] = {
  312. ...result.metadata,
  313. time: {
  314. start,
  315. end: Date.now(),
  316. },
  317. }
  318. await updateMessage(next)
  319. return result.output
  320. } catch (e: any) {
  321. next.metadata!.tool![opts.toolCallId] = {
  322. error: true,
  323. message: e.toString(),
  324. title: e.toString(),
  325. time: {
  326. start,
  327. end: Date.now(),
  328. },
  329. }
  330. await updateMessage(next)
  331. return e.toString()
  332. }
  333. },
  334. })
  335. }
  336. for (const [key, item] of Object.entries(await MCP.tools())) {
  337. const execute = item.execute
  338. if (!execute) continue
  339. item.execute = async (args, opts) => {
  340. const start = Date.now()
  341. try {
  342. const result = await execute(args, opts)
  343. next.metadata!.tool![opts.toolCallId] = {
  344. ...result.metadata,
  345. time: {
  346. start,
  347. end: Date.now(),
  348. },
  349. }
  350. await updateMessage(next)
  351. return result.content
  352. .filter((x: any) => x.type === "text")
  353. .map((x: any) => x.text)
  354. .join("\n\n")
  355. } catch (e: any) {
  356. next.metadata!.tool![opts.toolCallId] = {
  357. error: true,
  358. message: e.toString(),
  359. title: "mcp",
  360. time: {
  361. start,
  362. end: Date.now(),
  363. },
  364. }
  365. await updateMessage(next)
  366. return e.toString()
  367. }
  368. }
  369. tools[key] = item
  370. }
  371. let text: Message.TextPart | undefined
  372. await Bun.write(
  373. "/tmp/message.json",
  374. JSON.stringify(
  375. [
  376. ...system.map(
  377. (x): CoreMessage => ({
  378. role: "system",
  379. content: x,
  380. }),
  381. ),
  382. ...convertToCoreMessages(
  383. msgs.map(toUIMessage).filter((x) => x.parts.length > 0),
  384. ),
  385. ],
  386. null,
  387. 2,
  388. ),
  389. )
  390. const result = streamText({
  391. onStepFinish: async (step) => {
  392. log.info("step finish", { finishReason: step.finishReason })
  393. const assistant = next.metadata!.assistant!
  394. const usage = getUsage(model.info, step.usage, step.providerMetadata)
  395. assistant.cost += usage.cost
  396. assistant.tokens = usage.tokens
  397. await updateMessage(next)
  398. if (text) {
  399. Bus.publish(Message.Event.PartUpdated, {
  400. part: text,
  401. messageID: next.id,
  402. sessionID: next.metadata.sessionID,
  403. })
  404. }
  405. text = undefined
  406. },
  407. async onFinish(input) {
  408. log.info("message finish", {
  409. reason: input.finishReason,
  410. })
  411. const assistant = next.metadata!.assistant!
  412. const usage = getUsage(model.info, input.usage, input.providerMetadata)
  413. assistant.cost = usage.cost
  414. await updateMessage(next)
  415. },
  416. onError(err) {
  417. log.error("callback error", err)
  418. switch (true) {
  419. case LoadAPIKeyError.isInstance(err.error):
  420. next.metadata.error = new Provider.AuthError(
  421. {
  422. providerID: input.providerID,
  423. message: err.error.message,
  424. },
  425. { cause: err.error },
  426. ).toObject()
  427. break
  428. case err.error instanceof Error:
  429. next.metadata.error = new NamedError.Unknown(
  430. { message: err.error.toString() },
  431. { cause: err.error },
  432. ).toObject()
  433. break
  434. default:
  435. next.metadata.error = new NamedError.Unknown(
  436. { message: JSON.stringify(err.error) },
  437. { cause: err.error },
  438. )
  439. }
  440. Bus.publish(Event.Error, {
  441. error: next.metadata.error,
  442. })
  443. },
  444. // async prepareStep(step) {
  445. // next.parts.push({
  446. // type: "step-start",
  447. // })
  448. // await updateMessage(next)
  449. // return step
  450. // },
  451. toolCallStreaming: true,
  452. abortSignal: abort.signal,
  453. maxSteps: 1000,
  454. messages: [
  455. ...system.map(
  456. (x, index): CoreMessage => ({
  457. role: "system",
  458. content: x,
  459. providerOptions: {
  460. ...(input.providerID === "anthropic" && index < 4
  461. ? {
  462. anthropic: {
  463. cacheControl: { type: "ephemeral" },
  464. },
  465. }
  466. : {}),
  467. },
  468. }),
  469. ),
  470. ...convertToCoreMessages(
  471. msgs.map(toUIMessage).filter((x) => x.parts.length > 0),
  472. ),
  473. ],
  474. temperature: model.info.id === "codex-mini-latest" ? undefined : 0,
  475. tools: {
  476. ...tools,
  477. },
  478. model: model.language,
  479. })
  480. try {
  481. for await (const value of result.fullStream) {
  482. l.info("part", {
  483. type: value.type,
  484. })
  485. switch (value.type) {
  486. case "step-start":
  487. next.parts.push({
  488. type: "step-start",
  489. })
  490. break
  491. case "text-delta":
  492. if (!text) {
  493. text = {
  494. type: "text",
  495. text: value.textDelta,
  496. }
  497. next.parts.push(text)
  498. break
  499. } else text.text += value.textDelta
  500. break
  501. case "tool-call": {
  502. const [match] = next.parts.flatMap((p) =>
  503. p.type === "tool-invocation" &&
  504. p.toolInvocation.toolCallId === value.toolCallId
  505. ? [p]
  506. : [],
  507. )
  508. if (!match) break
  509. match.toolInvocation.args = value.args
  510. match.toolInvocation.state = "call"
  511. Bus.publish(Message.Event.PartUpdated, {
  512. part: match,
  513. messageID: next.id,
  514. sessionID: next.metadata.sessionID,
  515. })
  516. break
  517. }
  518. case "tool-call-streaming-start":
  519. next.parts.push({
  520. type: "tool-invocation",
  521. toolInvocation: {
  522. state: "partial-call",
  523. toolName: value.toolName,
  524. toolCallId: value.toolCallId,
  525. args: {},
  526. },
  527. })
  528. Bus.publish(Message.Event.PartUpdated, {
  529. part: next.parts[next.parts.length - 1],
  530. messageID: next.id,
  531. sessionID: next.metadata.sessionID,
  532. })
  533. break
  534. case "tool-call-delta":
  535. break
  536. // for some reason ai sdk claims to not send this part but it does
  537. // @ts-expect-error
  538. case "tool-result":
  539. const match = next.parts.find(
  540. (p) =>
  541. p.type === "tool-invocation" &&
  542. // @ts-expect-error
  543. p.toolInvocation.toolCallId === value.toolCallId,
  544. )
  545. if (match && match.type === "tool-invocation") {
  546. match.toolInvocation = {
  547. // @ts-expect-error
  548. args: value.args,
  549. // @ts-expect-error
  550. toolCallId: value.toolCallId,
  551. // @ts-expect-error
  552. toolName: value.toolName,
  553. state: "result",
  554. // @ts-expect-error
  555. result: value.result as string,
  556. }
  557. Bus.publish(Message.Event.PartUpdated, {
  558. part: match,
  559. messageID: next.id,
  560. sessionID: next.metadata.sessionID,
  561. })
  562. }
  563. break
  564. default:
  565. l.info("unhandled", {
  566. type: value.type,
  567. })
  568. }
  569. await updateMessage(next)
  570. }
  571. } catch (e: any) {
  572. log.error("stream error", {
  573. error: e,
  574. })
  575. switch (true) {
  576. case LoadAPIKeyError.isInstance(e):
  577. next.metadata.error = new Provider.AuthError(
  578. {
  579. providerID: input.providerID,
  580. message: e.message,
  581. },
  582. { cause: e },
  583. ).toObject()
  584. break
  585. case e instanceof Error:
  586. next.metadata.error = new NamedError.Unknown(
  587. { message: e.toString() },
  588. { cause: e },
  589. ).toObject()
  590. break
  591. default:
  592. next.metadata.error = new NamedError.Unknown(
  593. { message: JSON.stringify(e) },
  594. { cause: e },
  595. )
  596. }
  597. Bus.publish(Event.Error, {
  598. error: next.metadata.error,
  599. })
  600. }
  601. next.metadata!.time.completed = Date.now()
  602. for (const part of next.parts) {
  603. if (
  604. part.type === "tool-invocation" &&
  605. part.toolInvocation.state !== "result"
  606. ) {
  607. part.toolInvocation = {
  608. ...part.toolInvocation,
  609. state: "result",
  610. result: "request was aborted",
  611. }
  612. }
  613. }
  614. await updateMessage(next)
  615. return next
  616. }
  617. export async function summarize(input: {
  618. sessionID: string
  619. providerID: string
  620. modelID: string
  621. }) {
  622. using abort = lock(input.sessionID)
  623. const msgs = await messages(input.sessionID)
  624. const lastSummary = msgs.findLast(
  625. (msg) => msg.metadata.assistant?.summary === true,
  626. )?.id
  627. const filtered = msgs.filter((msg) => !lastSummary || msg.id >= lastSummary)
  628. const model = await Provider.getModel(input.providerID, input.modelID)
  629. const app = App.info()
  630. const system = SystemPrompt.summarize(input.providerID)
  631. const next: Message.Info = {
  632. id: Identifier.ascending("message"),
  633. role: "assistant",
  634. parts: [],
  635. metadata: {
  636. tool: {},
  637. sessionID: input.sessionID,
  638. assistant: {
  639. system,
  640. path: {
  641. cwd: app.path.cwd,
  642. root: app.path.root,
  643. },
  644. summary: true,
  645. cost: 0,
  646. modelID: input.modelID,
  647. providerID: input.providerID,
  648. tokens: {
  649. input: 0,
  650. output: 0,
  651. reasoning: 0,
  652. cache: { read: 0, write: 0 },
  653. },
  654. },
  655. time: {
  656. created: Date.now(),
  657. },
  658. },
  659. }
  660. await updateMessage(next)
  661. const result = await generateText({
  662. abortSignal: abort.signal,
  663. model: model.language,
  664. messages: [
  665. ...system.map(
  666. (x): CoreMessage => ({
  667. role: "system",
  668. content: x,
  669. }),
  670. ),
  671. ...convertToCoreMessages(filtered.map(toUIMessage)),
  672. {
  673. role: "user",
  674. content: [
  675. {
  676. type: "text",
  677. text: "Provide a detailed but concise summary of our conversation above. Focus on information that would be helpful for continuing the conversation, including what we did, what we're doing, which files we're working on, and what we're going to do next.",
  678. },
  679. ],
  680. },
  681. ],
  682. })
  683. next.parts.push({
  684. type: "text",
  685. text: result.text,
  686. })
  687. const assistant = next.metadata!.assistant!
  688. const usage = getUsage(model.info, result.usage, result.providerMetadata)
  689. assistant.cost = usage.cost
  690. assistant.tokens = usage.tokens
  691. await updateMessage(next)
  692. }
  693. const pending = new Map<string, AbortController>()
  694. function lock(sessionID: string) {
  695. log.info("locking", { sessionID })
  696. if (pending.has(sessionID)) throw new BusyError(sessionID)
  697. const controller = new AbortController()
  698. pending.set(sessionID, controller)
  699. return {
  700. signal: controller.signal,
  701. [Symbol.dispose]() {
  702. log.info("unlocking", { sessionID })
  703. pending.delete(sessionID)
  704. },
  705. }
  706. }
  707. function getUsage(
  708. model: ModelsDev.Model,
  709. usage: LanguageModelUsage,
  710. metadata?: ProviderMetadata,
  711. ) {
  712. const tokens = {
  713. input: usage.promptTokens ?? 0,
  714. output: usage.completionTokens ?? 0,
  715. reasoning: 0,
  716. cache: {
  717. write: (metadata?.["anthropic"]?.["cacheCreationInputTokens"] ??
  718. 0) as number,
  719. read: (metadata?.["anthropic"]?.["cacheReadInputTokens"] ??
  720. 0) as number,
  721. },
  722. }
  723. return {
  724. cost: new Decimal(0)
  725. .add(new Decimal(tokens.input).mul(model.cost.input).div(1_000_000))
  726. .add(new Decimal(tokens.output).mul(model.cost.output).div(1_000_000))
  727. .toNumber(),
  728. tokens,
  729. }
  730. }
  731. export class BusyError extends Error {
  732. constructor(public readonly sessionID: string) {
  733. super(`Session ${sessionID} is busy`)
  734. }
  735. }
  736. export async function initialize(input: {
  737. sessionID: string
  738. modelID: string
  739. providerID: string
  740. }) {
  741. const app = App.info()
  742. await Session.chat({
  743. sessionID: input.sessionID,
  744. providerID: input.providerID,
  745. modelID: input.modelID,
  746. parts: [
  747. {
  748. type: "text",
  749. text: PROMPT_INITIALIZE.replace("${path}", app.path.root),
  750. },
  751. ],
  752. })
  753. await App.initialize()
  754. }
  755. }
  756. function toUIMessage(msg: Message.Info): UIMessage {
  757. if (msg.role === "assistant") {
  758. return {
  759. id: msg.id,
  760. role: "assistant",
  761. content: "",
  762. parts: toParts(msg.parts),
  763. }
  764. }
  765. if (msg.role === "user") {
  766. return {
  767. id: msg.id,
  768. role: "user",
  769. content: "",
  770. parts: toParts(msg.parts),
  771. }
  772. }
  773. throw new Error("not implemented")
  774. }
  775. function toParts(parts: Message.Part[]): UIMessage["parts"] {
  776. const result: UIMessage["parts"] = []
  777. for (const part of parts) {
  778. switch (part.type) {
  779. case "text":
  780. result.push({ type: "text", text: part.text })
  781. break
  782. case "file":
  783. result.push({
  784. type: "file",
  785. data: part.url,
  786. mimeType: part.mediaType,
  787. })
  788. break
  789. case "tool-invocation":
  790. result.push({
  791. type: "tool-invocation",
  792. toolInvocation: part.toolInvocation,
  793. })
  794. break
  795. case "step-start":
  796. result.push({
  797. type: "step-start",
  798. })
  799. break
  800. default:
  801. break
  802. }
  803. }
  804. return result
  805. }