From 3fb57044d120618c92214b734d1fc831955e39a2 Mon Sep 17 00:00:00 2001 From: Adam <2363879+adamdotdevin@users.noreply.github.com> Date: Sun, 23 Nov 2025 19:28:48 -0600 Subject: [PATCH] wip(share): more styling --- .opencode/opencode.jsonc | 10 ++++--- packages/enterprise/src/core/share.ts | 9 +++++- .../src/routes/share/[sessionID].tsx | 27 +++++++++++------- packages/opencode/src/share/share-next.ts | 28 +++++++++++++++++++ 4 files changed, 59 insertions(+), 15 deletions(-) diff --git a/.opencode/opencode.jsonc b/.opencode/opencode.jsonc index dd5a4c750..aadc02408 100644 --- a/.opencode/opencode.jsonc +++ b/.opencode/opencode.jsonc @@ -1,9 +1,11 @@ { "$schema": "https://opencode.ai/config.json", - "plugin": ["opencode-openai-codex-auth"], - // "enterprise": { - // "url": "http://localhost:3000", - // }, + "plugin": [ + "opencode-openai-codex-auth" + ], + "enterprise": { + "url": "http://localhost:3000", + }, "provider": { "opencode": { "options": { diff --git a/packages/enterprise/src/core/share.ts b/packages/enterprise/src/core/share.ts index a7dfbfdcc..c24077ec3 100644 --- a/packages/enterprise/src/core/share.ts +++ b/packages/enterprise/src/core/share.ts @@ -1,4 +1,4 @@ -import { FileDiff, Message, Part, Session, SessionStatus } from "@opencode-ai/sdk" +import { FileDiff, Message, Model, Part, Session, SessionStatus } from "@opencode-ai/sdk" import { fn } from "@opencode-ai/util/fn" import { iife } from "@opencode-ai/util/iife" import z from "zod" @@ -32,6 +32,10 @@ export namespace Share { type: z.literal("session_status"), data: z.custom(), }), + z.object({ + type: z.literal("model"), + data: z.custom(), + }), ]) export type Data = z.infer @@ -111,6 +115,9 @@ export namespace Share { case "session_status": await Storage.write(["share_data", input.share.id, "session_status"], item.data) break + case "model": + await Storage.write(["share_data", input.share.id, "model"], item.data) + break } }), ) diff --git a/packages/enterprise/src/routes/share/[sessionID].tsx b/packages/enterprise/src/routes/share/[sessionID].tsx index 0ff0482f1..a2365aa1d 100644 --- a/packages/enterprise/src/routes/share/[sessionID].tsx +++ b/packages/enterprise/src/routes/share/[sessionID].tsx @@ -1,7 +1,7 @@ -import { FileDiff, Message, Part, Session, SessionStatus, UserMessage } from "@opencode-ai/sdk" +import { FileDiff, Message, Model, Part, Session, SessionStatus, UserMessage } from "@opencode-ai/sdk" import { SessionTurn } from "@opencode-ai/ui/session-turn" import { SessionReview } from "@opencode-ai/ui/session-review" -import { DataProvider, useData } from "@opencode-ai/ui/context" +import { DataProvider } from "@opencode-ai/ui/context" import { createAsync, query, RouteDefinition, useParams } from "@solidjs/router" import { createMemo, Show } from "solid-js" import { Share } from "~/core/share" @@ -29,6 +29,9 @@ const getData = query(async (sessionID) => { part: { [messageID: string]: Part[] } + model: { + [sessionID: string]: Model[] + } } = { session: [], session_diff: { @@ -41,6 +44,7 @@ const getData = query(async (sessionID) => { }, message: {}, part: {}, + model: {}, } for (const item of data) { switch (item.type) { @@ -61,6 +65,9 @@ const getData = query(async (sessionID) => { result.part[item.data.messageID] = result.part[item.data.messageID] ?? [] result.part[item.data.messageID].push(item.data) break + case "model": + result.model[sessionID] = item.data + break } } return result @@ -82,15 +89,14 @@ export default function () { {(data) => ( {iife(() => { - const data = useData() const [store, setStore] = createStore({ messageId: undefined as string | undefined, }) - const match = createMemo(() => Binary.search(data.session, params.sessionID!, (s) => s.id)) + const match = createMemo(() => Binary.search(data().session, params.sessionID!, (s) => s.id)) if (!match().found) throw new Error(`Session ${params.sessionID} not found`) - const info = createMemo(() => data.session[match().index]) + const info = createMemo(() => data().session[match().index]) const messages = createMemo(() => - params.sessionID ? (data.message[params.sessionID]?.filter((m) => m.role === "user") ?? []) : [], + params.sessionID ? (data().message[params.sessionID]?.filter((m) => m.role === "user") ?? []) : [], ) const firstUserMessage = createMemo(() => messages().at(0)) const activeMessage = createMemo( @@ -104,8 +110,9 @@ export default function () { } } const provider = createMemo(() => activeMessage()?.model?.providerID) - const model = createMemo(() => activeMessage()?.model?.modelID) - const diffs = createMemo(() => data.session_diff[params.sessionID!] ?? []) + const modelID = createMemo(() => activeMessage()?.model?.modelID) + const model = createMemo(() => data().model[params.sessionID!]?.find((m) => m.id === modelID())) + const diffs = createMemo(() => data().session_diff[params.sessionID!] ?? []) return (
@@ -152,7 +159,7 @@ export default function () { src={`https://models.dev/logos/${provider()}.svg`} class="size-4 shrink-0 dark:invert" /> -
{model()}
+
{model()?.name ?? modelID()}
{DateTime.fromMillis(info().time.created).toFormat("dd MMM yyyy, HH:mm")} @@ -163,7 +170,7 @@ export default function () {
1}> m.info, + ), + ], + }, + ]) + } }) Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => { await sync(evt.properties.part.sessionID, [ @@ -90,6 +104,10 @@ export namespace ShareNext { type: "session_diff" data: SDK.FileDiff[] } + | { + type: "model" + data: ModelsDev.Model[] + } async function sync(sessionID: string, data: Data[]) { const url = await Config.get().then((x) => x.enterprise!.url) @@ -129,6 +147,12 @@ export namespace ShareNext { const session = await Session.get(sessionID) const diffs = await Session.diff(sessionID) const messages = await Array.fromAsync(MessageV2.stream(sessionID)) + const models = await Promise.all( + messages + .filter((m) => m.info.role === "user") + .map((m) => (m.info as SDK.UserMessage).model) + .map((m) => Provider.getModel(m.providerID, m.modelID).then((m) => m.info)), + ) await sync(sessionID, [ { type: "session", @@ -143,6 +167,10 @@ export namespace ShareNext { type: "session_diff", data: diffs, }, + { + type: "model", + data: models, + }, ]) } }