mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-30 03:54:59 +00:00
feat(server): mount question web handler
This commit is contained in:
parent
bee094ff7b
commit
816ba270a5
5 changed files with 57 additions and 28 deletions
|
|
@ -2,15 +2,10 @@ import { AppLayer } from "@/effect/app-runtime"
|
|||
import { memoMap } from "@/effect/run-service"
|
||||
import { Question } from "@/question"
|
||||
import { QuestionID } from "@/question/schema"
|
||||
import { lazy } from "@/util/lazy"
|
||||
import { makeQuestionHandler, questionApi } from "@opencode-ai/server"
|
||||
import { makeQuestionHandler, makeQuestionWebHandler } from "@opencode-ai/server"
|
||||
import { Effect, Layer } from "effect"
|
||||
import { HttpRouter, HttpServer } from "effect/unstable/http"
|
||||
import { HttpApiBuilder } from "effect/unstable/httpapi"
|
||||
import type { Handler } from "hono"
|
||||
|
||||
const root = "/experimental/httpapi/question"
|
||||
|
||||
const QuestionLive = makeQuestionHandler({
|
||||
list: Effect.fn("QuestionHttpApi.host.list")(function* () {
|
||||
const svc = yield* Question.Service
|
||||
|
|
@ -25,20 +20,10 @@ const QuestionLive = makeQuestionHandler({
|
|||
}),
|
||||
}).pipe(Layer.provide(Question.defaultLayer))
|
||||
|
||||
const web = lazy(() =>
|
||||
HttpRouter.toWebHandler(
|
||||
Layer.mergeAll(
|
||||
AppLayer,
|
||||
HttpApiBuilder.layer(questionApi, { openapiPath: `${root}/doc` }).pipe(
|
||||
Layer.provide(QuestionLive),
|
||||
Layer.provide(HttpServer.layerServices),
|
||||
),
|
||||
),
|
||||
{
|
||||
disableLogger: true,
|
||||
memoMap,
|
||||
},
|
||||
),
|
||||
)
|
||||
const web = makeQuestionWebHandler({
|
||||
app: AppLayer,
|
||||
live: QuestionLive,
|
||||
memoMap,
|
||||
})
|
||||
|
||||
export const QuestionHttpApiHandler: Handler = (c, _next) => web().handler(c.req.raw)
|
||||
export const QuestionHttpApiHandler: Handler = (c, _next) => web(c.req.raw)
|
||||
|
|
|
|||
|
|
@ -1,2 +1,3 @@
|
|||
export { makeQuestionHandler } from "./question.js"
|
||||
export { makeQuestionWebHandler } from "./question.js"
|
||||
export type { QuestionOps } from "./question.js"
|
||||
|
|
|
|||
|
|
@ -1,6 +1,7 @@
|
|||
import { Effect, Schema } from "effect"
|
||||
import { Effect, Layer, Schema } from "effect"
|
||||
import { HttpRouter, HttpServer } from "effect/unstable/http"
|
||||
import { HttpApiBuilder } from "effect/unstable/httpapi"
|
||||
import { QuestionReply, QuestionRequest, questionApi } from "../definition/question.js"
|
||||
import { QuestionReply, QuestionRequest, questionApi, questionRoot } from "../definition/question.js"
|
||||
|
||||
export interface QuestionOps<R = never> {
|
||||
readonly list: () => Effect.Effect<ReadonlyArray<unknown>, never, R>
|
||||
|
|
@ -35,3 +36,44 @@ export const makeQuestionHandler = <R>(ops: QuestionOps<R>) =>
|
|||
return handlers.handle("list", list).handle("reply", reply)
|
||||
}),
|
||||
)
|
||||
|
||||
export const makeQuestionWebHandler = <A, E, R, B, F, S>(opts: {
|
||||
readonly app: Layer.Layer<A, E, R>
|
||||
readonly live: Layer.Layer<B, F, S>
|
||||
readonly memoMap?: Layer.MemoMap
|
||||
}) => {
|
||||
const app = Layer.mergeAll(
|
||||
opts.app,
|
||||
HttpApiBuilder.layer(questionApi, { openapiPath: `${questionRoot}/doc` }).pipe(
|
||||
Layer.provide(opts.live),
|
||||
Layer.provide(HttpServer.layerServices),
|
||||
),
|
||||
)
|
||||
|
||||
const init = () =>
|
||||
HttpRouter.toWebHandler(
|
||||
app as Layer.Layer<
|
||||
A,
|
||||
E | F,
|
||||
| HttpRouter.HttpRouter
|
||||
| HttpRouter.Request<"Requires", unknown>
|
||||
| HttpRouter.Request<"GlobalRequires", unknown>
|
||||
| HttpRouter.Request<"Error", unknown>
|
||||
| HttpRouter.Request<"GlobalError", unknown>
|
||||
>,
|
||||
{
|
||||
disableLogger: true,
|
||||
memoMap: opts.memoMap,
|
||||
},
|
||||
) as {
|
||||
readonly handler: (request: Request) => Promise<Response>
|
||||
readonly dispose: () => Promise<void>
|
||||
}
|
||||
|
||||
let web: ReturnType<typeof init> | undefined
|
||||
|
||||
return (request: Request) => {
|
||||
web ??= init()
|
||||
return web.handler(request)
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,7 +1,7 @@
|
|||
import { Schema } from "effect"
|
||||
import { HttpApi, HttpApiEndpoint, HttpApiGroup, OpenApi } from "effect/unstable/httpapi"
|
||||
|
||||
const root = "/experimental/httpapi/question"
|
||||
export const questionRoot = "/experimental/httpapi/question"
|
||||
|
||||
// Temporary transport-local schemas until canonical question schemas move into packages/core.
|
||||
export const QuestionID = Schema.String.annotate({ identifier: "QuestionID" })
|
||||
|
|
@ -64,7 +64,7 @@ export class QuestionReply extends Schema.Class<QuestionReply>("QuestionReply")(
|
|||
export const questionApi = HttpApi.make("question").add(
|
||||
HttpApiGroup.make("question")
|
||||
.add(
|
||||
HttpApiEndpoint.get("list", root, {
|
||||
HttpApiEndpoint.get("list", questionRoot, {
|
||||
success: Schema.Array(QuestionRequest),
|
||||
}).annotateMerge(
|
||||
OpenApi.annotations({
|
||||
|
|
@ -73,7 +73,7 @@ export const questionApi = HttpApi.make("question").add(
|
|||
description: "Get all pending question requests across all sessions.",
|
||||
}),
|
||||
),
|
||||
HttpApiEndpoint.post("reply", `${root}/:requestID/reply`, {
|
||||
HttpApiEndpoint.post("reply", `${questionRoot}/:requestID/reply`, {
|
||||
params: { requestID: QuestionID },
|
||||
payload: QuestionReply,
|
||||
success: Schema.Boolean,
|
||||
|
|
|
|||
|
|
@ -1,6 +1,7 @@
|
|||
export { openapi } from "./openapi.js"
|
||||
export { makeQuestionHandler } from "./api/question.js"
|
||||
export { makeQuestionWebHandler } from "./api/question.js"
|
||||
export { api } from "./definition/api.js"
|
||||
export { questionApi, QuestionReply, QuestionRequest } from "./definition/question.js"
|
||||
export { questionApi, questionRoot, QuestionReply, QuestionRequest } from "./definition/question.js"
|
||||
export type { OpenApiSpec, ServerApi } from "./types.js"
|
||||
export type { QuestionOps } from "./api/question.js"
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue