401 lines
17 KiB
TypeScript
401 lines
17 KiB
TypeScript
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 } from "@opencode-ai/ui/context"
|
|
import { DiffComponentProvider } from "@opencode-ai/ui/context/diff"
|
|
import { createAsync, query, useParams } from "@solidjs/router"
|
|
import { createEffect, createMemo, ErrorBoundary, For, Match, Show, Switch } from "solid-js"
|
|
import { Share } from "~/core/share"
|
|
import { Logo, Mark } from "@opencode-ai/ui/logo"
|
|
import { IconButton } from "@opencode-ai/ui/icon-button"
|
|
import { createDefaultOptions } from "@opencode-ai/ui/pierre"
|
|
import { iife } from "@opencode-ai/util/iife"
|
|
import { Binary } from "@opencode-ai/util/binary"
|
|
import { NamedError } from "@opencode-ai/util/error"
|
|
import { DateTime } from "luxon"
|
|
import { SessionMessageRail } from "@opencode-ai/ui/session-message-rail"
|
|
import { createStore } from "solid-js/store"
|
|
import z from "zod"
|
|
import NotFound from "../[...404]"
|
|
import { Tabs } from "@opencode-ai/ui/tabs"
|
|
import { preloadMultiFileDiff, PreloadMultiFileDiffResult } from "@pierre/precision-diffs/ssr"
|
|
import { Diff as SSRDiff } from "@opencode-ai/ui/diff-ssr"
|
|
import { clientOnly } from "@solidjs/start"
|
|
|
|
const ClientOnlyDiff = clientOnly(() => import("@opencode-ai/ui/diff").then((m) => ({ default: m.Diff })))
|
|
|
|
const SessionDataMissingError = NamedError.create(
|
|
"SessionDataMissingError",
|
|
z.object({
|
|
sessionID: z.string(),
|
|
message: z.string().optional(),
|
|
}),
|
|
)
|
|
|
|
const getData = query(async (shareID) => {
|
|
"use server"
|
|
const share = await Share.get(shareID)
|
|
if (!share) throw new SessionDataMissingError({ sessionID: shareID })
|
|
const data = await Share.data(shareID)
|
|
const result: {
|
|
sessionID: string
|
|
session: Session[]
|
|
session_diff: {
|
|
[sessionID: string]: FileDiff[]
|
|
}
|
|
session_diff_preload: {
|
|
[sessionID: string]: PreloadMultiFileDiffResult<any>[]
|
|
}
|
|
session_diff_preload_split: {
|
|
[sessionID: string]: PreloadMultiFileDiffResult<any>[]
|
|
}
|
|
session_status: {
|
|
[sessionID: string]: SessionStatus
|
|
}
|
|
message: {
|
|
[sessionID: string]: Message[]
|
|
}
|
|
part: {
|
|
[messageID: string]: Part[]
|
|
}
|
|
model: {
|
|
[sessionID: string]: Model[]
|
|
}
|
|
} = {
|
|
sessionID: share.sessionID,
|
|
session: [],
|
|
session_diff: {
|
|
[share.sessionID]: [],
|
|
},
|
|
session_diff_preload: {
|
|
[share.sessionID]: [],
|
|
},
|
|
session_diff_preload_split: {
|
|
[share.sessionID]: [],
|
|
},
|
|
session_status: {
|
|
[share.sessionID]: {
|
|
type: "idle",
|
|
},
|
|
},
|
|
message: {},
|
|
part: {},
|
|
model: {},
|
|
}
|
|
for (const item of data) {
|
|
switch (item.type) {
|
|
case "session":
|
|
result.session.push(item.data)
|
|
break
|
|
case "session_diff":
|
|
result.session_diff[share.sessionID] = item.data
|
|
await Promise.all([
|
|
Promise.all(
|
|
item.data.map(async (diff) =>
|
|
preloadMultiFileDiff<any>({
|
|
oldFile: { name: diff.file, contents: diff.before },
|
|
newFile: { name: diff.file, contents: diff.after },
|
|
options: createDefaultOptions("unified"),
|
|
// annotations,
|
|
}),
|
|
),
|
|
).then((r) => (result.session_diff_preload[share.sessionID] = r)),
|
|
Promise.all(
|
|
item.data.map(async (diff) =>
|
|
preloadMultiFileDiff<any>({
|
|
oldFile: { name: diff.file, contents: diff.before },
|
|
newFile: { name: diff.file, contents: diff.after },
|
|
options: createDefaultOptions("split"),
|
|
// annotations,
|
|
}),
|
|
),
|
|
).then((r) => (result.session_diff_preload_split[share.sessionID] = r)),
|
|
])
|
|
break
|
|
case "message":
|
|
result.message[item.data.sessionID] = result.message[item.data.sessionID] ?? []
|
|
result.message[item.data.sessionID].push(item.data)
|
|
break
|
|
case "part":
|
|
result.part[item.data.messageID] = result.part[item.data.messageID] ?? []
|
|
result.part[item.data.messageID].push(item.data)
|
|
break
|
|
case "model":
|
|
result.model[share.sessionID] = item.data
|
|
break
|
|
}
|
|
}
|
|
const match = Binary.search(result.session, share.sessionID, (s) => s.id)
|
|
if (!match.found) throw new SessionDataMissingError({ sessionID: share.sessionID })
|
|
return result
|
|
}, "getShareData")
|
|
|
|
export default function () {
|
|
const params = useParams()
|
|
const data = createAsync(async () => {
|
|
if (!params.shareID) throw new Error("Missing shareID")
|
|
return getData(params.shareID)
|
|
})
|
|
|
|
createEffect(() => {
|
|
console.log(data())
|
|
})
|
|
|
|
return (
|
|
<ErrorBoundary
|
|
fallback={(e) => {
|
|
return (
|
|
<Show when={e.message === "SessionDataMissingError"}>
|
|
<NotFound />
|
|
</Show>
|
|
)
|
|
}}
|
|
>
|
|
<Show when={data()}>
|
|
{(data) => {
|
|
const match = createMemo(() => Binary.search(data().session, data().sessionID, (s) => s.id))
|
|
if (!match().found) throw new Error(`Session ${data().sessionID} not found`)
|
|
const info = createMemo(() => data().session[match().index])
|
|
|
|
return (
|
|
<DiffComponentProvider component={ClientOnlyDiff}>
|
|
<DataProvider data={data()} directory={info().directory}>
|
|
{iife(() => {
|
|
const [store, setStore] = createStore({
|
|
messageId: undefined as string | undefined,
|
|
})
|
|
const messages = createMemo(() =>
|
|
data().sessionID
|
|
? (data().message[data().sessionID]?.filter((m) => m.role === "user") ?? []).sort(
|
|
(a, b) => b.time.created - a.time.created,
|
|
)
|
|
: [],
|
|
)
|
|
const firstUserMessage = createMemo(() => messages().at(0))
|
|
const activeMessage = createMemo(
|
|
() => messages().find((m) => m.id === store.messageId) ?? firstUserMessage(),
|
|
)
|
|
function setActiveMessage(message: UserMessage | undefined) {
|
|
if (message) {
|
|
setStore("messageId", message.id)
|
|
} else {
|
|
setStore("messageId", undefined)
|
|
}
|
|
}
|
|
const provider = createMemo(() => activeMessage()?.model?.providerID)
|
|
const modelID = createMemo(() => activeMessage()?.model?.modelID)
|
|
const model = createMemo(() => data().model[data().sessionID]?.find((m) => m.id === modelID()))
|
|
const diffs = createMemo(() => {
|
|
const diffs = data().session_diff[data().sessionID] ?? []
|
|
const preloaded = data().session_diff_preload[data().sessionID] ?? []
|
|
return diffs.map((diff) => ({
|
|
...diff,
|
|
preloaded: preloaded.find((d) => d.newFile.name === diff.file),
|
|
}))
|
|
})
|
|
const splitDiffs = createMemo(() => {
|
|
const diffs = data().session_diff[data().sessionID] ?? []
|
|
const preloaded = data().session_diff_preload_split[data().sessionID] ?? []
|
|
return diffs.map((diff) => ({
|
|
...diff,
|
|
preloaded: preloaded.find((d) => d.newFile.name === diff.file),
|
|
}))
|
|
})
|
|
|
|
const title = () => (
|
|
<div class="flex flex-col gap-4">
|
|
<div class="h-8 flex gap-4 items-center justify-start self-stretch">
|
|
<div class="pl-[2.5px] pr-2 flex items-center gap-1.75 bg-surface-strong shadow-xs-border-base">
|
|
<Mark class="shrink-0 w-3 my-0.5" />
|
|
<div class="text-12-mono text-text-base">v{info().version}</div>
|
|
</div>
|
|
<div class="flex gap-2 items-center">
|
|
<img
|
|
src={`https://models.dev/logos/${provider()}.svg`}
|
|
class="size-3.5 shrink-0 dark:invert"
|
|
/>
|
|
<div class="text-12-regular text-text-base">{model()?.name ?? modelID()}</div>
|
|
</div>
|
|
<div class="text-12-regular text-text-weaker">
|
|
{DateTime.fromMillis(info().time.created).toFormat("dd MMM yyyy, HH:mm")}
|
|
</div>
|
|
</div>
|
|
<div class="text-left text-16-medium text-text-strong">{info().title}</div>
|
|
</div>
|
|
)
|
|
|
|
const turns = () => (
|
|
<div class="relative mt-2 pt-6 pb-8 min-w-0 w-full h-full overflow-y-auto no-scrollbar">
|
|
<div class="px-4">{title()}</div>
|
|
<div class="flex flex-col gap-15 items-start justify-start mt-4">
|
|
<For each={messages()}>
|
|
{(message) => (
|
|
<SessionTurn
|
|
sessionID={data().sessionID}
|
|
messageID={message.id}
|
|
classes={{
|
|
root: "min-w-0 w-full relative",
|
|
content:
|
|
"flex flex-col justify-between !overflow-visible [&_[data-slot=session-turn-message-header]]:top-[-32px]",
|
|
container: "px-4",
|
|
}}
|
|
/>
|
|
)}
|
|
</For>
|
|
</div>
|
|
<div class="px-4 flex items-center justify-center pt-20 pb-8 shrink-0">
|
|
<Logo class="w-58.5 opacity-12" />
|
|
</div>
|
|
</div>
|
|
)
|
|
|
|
const wide = createMemo(() => diffs().length === 0)
|
|
|
|
return (
|
|
<div class="relative bg-background-stronger w-screen h-screen overflow-hidden flex flex-col">
|
|
<header class="h-12 px-6 py-2 flex items-center justify-between self-stretch bg-background-base border-b border-border-weak-base">
|
|
<div class="">
|
|
<a href="https://opencode.ai">
|
|
<Mark />
|
|
</a>
|
|
</div>
|
|
<div class="flex gap-3 items-center">
|
|
<IconButton
|
|
as={"a"}
|
|
href="https://github.com/sst/opencode"
|
|
target="_blank"
|
|
icon="github"
|
|
variant="ghost"
|
|
/>
|
|
<IconButton
|
|
as={"a"}
|
|
href="https://opencode.ai/discord"
|
|
target="_blank"
|
|
icon="discord"
|
|
variant="ghost"
|
|
/>
|
|
</div>
|
|
</header>
|
|
<div class="select-text flex flex-col flex-1 min-h-0">
|
|
<div
|
|
classList={{ "hidden w-full flex-1 min-h-0": true, "md:flex": wide(), "lg:flex": !wide() }}
|
|
>
|
|
<div
|
|
classList={{
|
|
"@container relative shrink-0 pt-14 flex flex-col gap-10 min-h-0 w-full": true,
|
|
"mx-auto max-w-146": !wide(),
|
|
}}
|
|
>
|
|
<div
|
|
classList={{
|
|
"w-full flex justify-start items-start min-w-0": true,
|
|
"max-w-146 mx-auto px-6": wide(),
|
|
"pr-6 pl-18": !wide() && messages().length > 1,
|
|
"px-6": !wide() && messages().length === 1,
|
|
}}
|
|
>
|
|
{title()}
|
|
</div>
|
|
<div class="flex items-start justify-start h-full min-h-0">
|
|
<SessionMessageRail
|
|
messages={messages()}
|
|
current={activeMessage()}
|
|
onMessageSelect={setActiveMessage}
|
|
wide={wide()}
|
|
/>
|
|
<SessionTurn
|
|
sessionID={data().sessionID}
|
|
messageID={store.messageId ?? firstUserMessage()!.id!}
|
|
classes={{
|
|
root: "grow",
|
|
content: "flex flex-col justify-between items-start",
|
|
container:
|
|
"w-full pb-20 " +
|
|
(wide() ? "max-w-146 mx-auto px-6" : messages().length > 1 ? "pr-6 pl-18" : "px-6"),
|
|
}}
|
|
>
|
|
<div classList={{ "w-full flex items-center justify-center pb-8 shrink-0": true }}>
|
|
<Logo class="w-58.5 opacity-12" />
|
|
</div>
|
|
</SessionTurn>
|
|
</div>
|
|
</div>
|
|
<Show when={diffs().length > 0}>
|
|
<DiffComponentProvider component={SSRDiff}>
|
|
<div class="@container relative grow pt-14 flex-1 min-h-0 border-l border-border-weak-base">
|
|
<SessionReview
|
|
class="@4xl:hidden"
|
|
diffs={diffs()}
|
|
classes={{
|
|
root: "pb-20",
|
|
header: "px-6",
|
|
container: "px-6",
|
|
}}
|
|
/>
|
|
<SessionReview
|
|
split
|
|
class="hidden @4xl:flex"
|
|
diffs={splitDiffs()}
|
|
classes={{
|
|
root: "pb-20",
|
|
header: "px-6",
|
|
container: "px-6",
|
|
}}
|
|
/>
|
|
</div>
|
|
</DiffComponentProvider>
|
|
</Show>
|
|
</div>
|
|
<Switch>
|
|
<Match when={diffs().length > 0}>
|
|
<Tabs classList={{ "md:hidden": wide(), "lg:hidden": !wide() }}>
|
|
<Tabs.List>
|
|
<Tabs.Trigger value="session" class="w-1/2" classes={{ button: "w-full" }}>
|
|
Session
|
|
</Tabs.Trigger>
|
|
<Tabs.Trigger value="review" class="w-1/2 !border-r-0" classes={{ button: "w-full" }}>
|
|
{diffs().length} Files Changed
|
|
</Tabs.Trigger>
|
|
</Tabs.List>
|
|
<Tabs.Content value="session" class="!overflow-hidden">
|
|
{turns()}
|
|
</Tabs.Content>
|
|
<Tabs.Content
|
|
forceMount
|
|
value="review"
|
|
class="!overflow-hidden hidden data-[selected]:block"
|
|
>
|
|
<div class="relative h-full pt-8 overflow-y-auto no-scrollbar">
|
|
<DiffComponentProvider component={SSRDiff}>
|
|
<SessionReview
|
|
diffs={diffs()}
|
|
classes={{
|
|
root: "pb-20",
|
|
header: "px-4",
|
|
container: "px-4",
|
|
}}
|
|
/>
|
|
</DiffComponentProvider>
|
|
</div>
|
|
</Tabs.Content>
|
|
</Tabs>
|
|
</Match>
|
|
<Match when={true}>
|
|
<div classList={{ "!overflow-hidden": true, "md:hidden": wide(), "lg:hidden": !wide() }}>
|
|
{turns()}
|
|
</div>
|
|
</Match>
|
|
</Switch>
|
|
</div>
|
|
</div>
|
|
)
|
|
})}
|
|
</DataProvider>
|
|
</DiffComponentProvider>
|
|
)
|
|
}}
|
|
</Show>
|
|
</ErrorBoundary>
|
|
)
|
|
}
|