From 7812e43bc4522d59f971e9bfc1a16ce400701398 Mon Sep 17 00:00:00 2001 From: loup Date: Sat, 4 Jan 2025 00:23:54 +0100 Subject: [PATCH 01/10] feat: initial hono implementation --- .../grafserv/__tests__/hono-adapter.test.ts | 88 ++++++++ grafast/grafserv/examples/example-hono.mjs | 21 ++ grafast/grafserv/package.json | 8 +- grafast/grafserv/src/servers/h3/v1/index.ts | 1 - grafast/grafserv/src/servers/hono/index.ts | 208 ++++++++++++++++++ yarn.lock | 29 ++- 6 files changed, 352 insertions(+), 3 deletions(-) create mode 100644 grafast/grafserv/__tests__/hono-adapter.test.ts create mode 100644 grafast/grafserv/examples/example-hono.mjs create mode 100644 grafast/grafserv/src/servers/hono/index.ts diff --git a/grafast/grafserv/__tests__/hono-adapter.test.ts b/grafast/grafserv/__tests__/hono-adapter.test.ts new file mode 100644 index 0000000000..87d2cb7ef3 --- /dev/null +++ b/grafast/grafserv/__tests__/hono-adapter.test.ts @@ -0,0 +1,88 @@ +import { Hono } from "hono"; +import { serverAudits } from "graphql-http"; +import { grafserv } from "../src/servers/hono/index.js"; +import type { GrafservConfig } from "../src/interfaces.js"; +import { constant, makeGrafastSchema } from "grafast"; +import { error } from "console"; +import { serve } from "@hono/node-server"; +import { AddressInfo } from "net"; + +const schema = makeGrafastSchema({ + typeDefs: /* GraphQL */ ` + type Query { + hello: String! + throwAnError: String + } + `, + plans: { + Query: { + hello() { + return constant("world"); + }, + throwAnError() { + return error(new Error("You asked for an error... Here it is.")); + }, + }, + }, +}); + +describe("Hono Adapter", () => { + // setup test server + const app = new Hono(); + const config: GrafservConfig = { + schema, // Mock schema for testing + preset: { + grafserv: { + graphqlOverGET: true, + graphqlPath: "/graphql", + dangerouslyAllowAllCORSRequests: true, + }, + }, + }; + const honoGrafserv = grafserv(config); + honoGrafserv.addTo(app); + + const server = serve({ + fetch: app.fetch, + port: 7777, + }); + const url = `http://0.0.0.0:7777/graphql`; + + it("SHOULD work for a simple request", async () => { + const res = await fetch(url, { + method: "POST", + headers: { + "content-type": "application/json", + }, + body: JSON.stringify({ query: "{ __typename }" }), + }); + + const responseBody = await res.json(); + expect(responseBody.data).toEqual({ + __typename: "Query", + }); + }); + + // run standard audits + const audits = serverAudits({ + url, + fetchFn: fetch, + }); + for (const audit of audits) { + it(audit.name, async () => { + const result = await audit.fn(); + if (audit.name.startsWith("MUST") || result.status === "ok") { + expect({ + ...result, + response: "", + }).toEqual( + expect.objectContaining({ + status: "ok", + }), + ); + } else { + console.warn(`Allowing failed test: ${audit.name}`); + } + }); + } +}); diff --git a/grafast/grafserv/examples/example-hono.mjs b/grafast/grafserv/examples/example-hono.mjs new file mode 100644 index 0000000000..11fcf3c312 --- /dev/null +++ b/grafast/grafserv/examples/example-hono.mjs @@ -0,0 +1,21 @@ +import { grafserv } from "grafserv/hono"; +import preset from "./graphile.config.mjs"; +import schema from "./schema.mjs"; + +import { Hono } from "hono"; +import { serve } from "@hono/node-server"; + +// Create a Node HTTP server +const app = new Hono(); + +// Create a Grafserv instance +const serv = grafserv({ schema, preset }); + +// Mount the request handler into a new HTTP server +serv.addTo(server).catch((e) => { + console.error(e); + process.exit(1); +}); + +// Start the Node server +serve(app); diff --git a/grafast/grafserv/package.json b/grafast/grafserv/package.json index 4c7cc5d14d..92b4035b07 100644 --- a/grafast/grafserv/package.json +++ b/grafast/grafserv/package.json @@ -38,6 +38,10 @@ "types": "./dist/servers/h3/v1/index.d.ts", "default": "./dist/servers/h3/v1/index.js" }, + "./hono": { + "types": "./dist/hono/index.d.ts", + "default": "./dist/hono/index.js" + }, "./ruru": { "types": "./fwd/ruru/index.d.ts", "node": "./fwd/ruru/index.js", @@ -81,7 +85,8 @@ "debug": "^4.3.4", "eventemitter3": "^5.0.1", "graphile-config": "workspace:^", - "graphql-ws": "^5.14.0", + "graphql-ws": "^5.16.0", + "hono": "^4.6.15", "ruru": "workspace:^", "tslib": "^2.6.2" }, @@ -110,6 +115,7 @@ "devDependencies": { "@envelop/core": "^5.0.0", "@fastify/websocket": "^8.2.0", + "@hono/node-server": "^1.13.7", "@types/aws-lambda": "^8.10.123", "@types/express": "^4.17.17", "@types/koa": "^2.13.8", diff --git a/grafast/grafserv/src/servers/h3/v1/index.ts b/grafast/grafserv/src/servers/h3/v1/index.ts index 7ccd5738a7..ace622827e 100644 --- a/grafast/grafserv/src/servers/h3/v1/index.ts +++ b/grafast/grafserv/src/servers/h3/v1/index.ts @@ -278,7 +278,6 @@ export class H3Grafserv extends GrafservBase { { socket: peer.websocket, request: peer.request }, ); client.closed = async (code, reason) => { - // @ts-expect-error fixed in unreleased https://github.com/enisdenjo/graphql-ws/pull/573 onClose(code, reason); }; }, diff --git a/grafast/grafserv/src/servers/hono/index.ts b/grafast/grafserv/src/servers/hono/index.ts new file mode 100644 index 0000000000..4975995f86 --- /dev/null +++ b/grafast/grafserv/src/servers/hono/index.ts @@ -0,0 +1,208 @@ +import { PassThrough } from "node:stream"; +import { Hono, Context as HonoContext } from "hono"; + +//@ts-expect-error type imports. +import type { Hooks, Peer } from "crossws"; +import { GRAPHQL_TRANSPORT_WS_PROTOCOL, makeServer } from "graphql-ws"; +import type { App, H3Event } from "h3"; + +import { + convertHandlerResultToResult, + GrafservBase, + makeGraphQLWSConfig, + normalizeRequest, + processHeaders, +} from "../../index.js"; +import type { + EventStreamHeandlerResult, + GrafservBodyBuffer, + GrafservBodyJSON, + GrafservConfig, + RequestDigest, + Result, +} from "../../interfaces.js"; +import { StatusCode } from "hono/utils/http-status"; +import { UpgradeWebSocket } from "hono/ws"; + +declare global { + namespace Grafast { + interface RequestContext { + Hono: { + honoContext: HonoContext; + }; + } + } +} + +function getDigest(honoContext: HonoContext): RequestDigest { + const req = honoContext.req; + const res = honoContext.res; + return { + httpVersionMajor: 1, // Hono uses Fetch API, which doesn't expose HTTP version + httpVersionMinor: 1, + isSecure: req.url.startsWith("https"), + method: req.method, + path: req.path, + headers: processHeaders(req.header()), + getQueryParams() { + return req.query(); + }, + async getBody() { + const json = await req.json(); + if (!json) { + throw new Error("Failed to retrieve body from hono"); + } + return { + type: "json", + json, + } as GrafservBodyJSON; + }, + requestContext: { + hono: { + context: honoContext, + }, + node: { + // @ts-expect-error type imports + req, + res, + }, + }, + }; +} + +export class HonoGrafserv extends GrafservBase { + constructor(config: GrafservConfig) { + super(config); + } + + /** + * @deprecated use handleGraphQLEvent instead + */ + public async handleEvent(honoContext: HonoContext) { + return this.handleGraphQLEvent(honoContext); + } + + public async handleGraphQLEvent(honoContext: HonoContext) { + const digest = getDigest(honoContext); + + const handlerResult = await this.graphqlHandler( + normalizeRequest(digest), + this.graphiqlHandler, + ); + const result = await convertHandlerResultToResult(handlerResult); + return this.send(honoContext, result); + } + + public async handleGraphiqlEvent(honoContext: HonoContext) { + const digest = getDigest(honoContext); + + const handlerResult = await this.graphiqlHandler(normalizeRequest(digest)); + const result = await convertHandlerResultToResult(handlerResult); + return this.send(honoContext, result); + } + + public async handleEventStreamEvent(honoContext: HonoContext) { + const digest = getDigest(honoContext); + + const handlerResult: EventStreamHeandlerResult = { + type: "event-stream", + request: normalizeRequest(digest), + dynamicOptions: this.dynamicOptions, + payload: this.makeStream(), + statusCode: 200, + }; + const result = await convertHandlerResultToResult(handlerResult); + return this.send(honoContext, result); + } + + public async send(honoContext: HonoContext, result: Result | null) { + if (result === null) { + // 404 + honoContext.status(404); + return honoContext.text("¯\\_(ツ)_/¯"); + } + + switch (result.type) { + case "error": { + const { statusCode, headers } = result; + this.setResponseHeaders(honoContext, headers); + honoContext.status(statusCode as StatusCode); + const errorWithStatus = Object.assign(result.error, { + status: statusCode, + }); + throw errorWithStatus; + } + case "buffer": { + const { statusCode, headers, buffer } = result; + this.setResponseHeaders(honoContext, headers); + honoContext.status(statusCode as StatusCode); + return honoContext.body(buffer); + } + case "json": { + const { statusCode, headers, json } = result; + this.setResponseHeaders(honoContext, headers); + honoContext.status(statusCode as StatusCode); + return honoContext.json(json); + } + case "noContent": { + const { statusCode, headers } = result; + this.setResponseHeaders(honoContext, headers); + honoContext.status(statusCode as StatusCode); + return honoContext.body(null); + } + // TODO : handle bufferStream ? + default: { + const never = result; + console.log("Unhandled:"); + console.dir(never); + this.setResponseHeaders(honoContext, { "Content-Type": "text/plain" }); + honoContext.status(501); + return "Server hasn't implemented this yet"; + } + } + } + + public async addTo(app: Hono) { + const dynamicOptions = this.dynamicOptions; + + app.on( + this.dynamicOptions.graphqlOverGET || + this.dynamicOptions.graphiqlOnGraphQLGET + ? ["GET", "POST"] + : ["POST"], + this.dynamicOptions.graphqlPath, + (c) => this.handleGraphQLEvent(c), + ); + + if (this.resolvedPreset.grafserv?.websockets) { + console.warn( + "You have enabled websockets, but the hono adapter doesn't support websockets yet.", + ); + } + + if (dynamicOptions.graphiql) { + app.get(this.dynamicOptions.graphiqlPath, (c) => + this.handleGraphiqlEvent(c), + ); + } + + if (dynamicOptions.watch) { + app.get(this.dynamicOptions.eventStreamPath, (c) => + this.handleEventStreamEvent(c), + ); + } + } + + private setResponseHeaders( + honoContext: HonoContext, + headers: Record, + ) { + for (const key in headers) { + honoContext.header(key, headers[key]); + } + } +} + +export function grafserv(config: GrafservConfig) { + return new HonoGrafserv(config); +} diff --git a/yarn.lock b/yarn.lock index 9bf846ab99..dc1d369d83 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3495,6 +3495,15 @@ __metadata: languageName: node linkType: hard +"@hono/node-server@npm:^1.13.7": + version: 1.13.7 + resolution: "@hono/node-server@npm:1.13.7" + peerDependencies: + hono: ^4 + checksum: 314671906b6a89f4611f7c7949c262287feff4a39c654c6831f78af7e8286537a97926849a7acdda654bea72b6a0d533b6df3f13a2b75325516a94393f8a6c8d + languageName: node + linkType: hard + "@humanwhocodes/config-array@npm:^0.11.10": version: 0.11.11 resolution: "@humanwhocodes/config-array@npm:0.11.11" @@ -11970,6 +11979,7 @@ __metadata: "@envelop/core": "npm:^5.0.0" "@fastify/websocket": "npm:^8.2.0" "@graphile/lru": "workspace:^" + "@hono/node-server": "npm:^1.13.7" "@types/aws-lambda": "npm:^8.10.123" "@types/express": "npm:^4.17.17" "@types/koa": "npm:^2.13.8" @@ -11982,8 +11992,9 @@ __metadata: grafast: "workspace:^" graphile-config: "workspace:^" graphql-http: "npm:^1.22.0" - graphql-ws: "npm:^5.14.0" + graphql-ws: "npm:^5.16.0" h3: "npm:^1.13.0" + hono: "npm:^4.6.15" jest: "npm:^29.6.4" jest-serializer-graphql-schema: "workspace:^" koa: "npm:^2.14.2" @@ -12309,6 +12320,15 @@ __metadata: languageName: node linkType: hard +"graphql-ws@npm:^5.16.0": + version: 5.16.0 + resolution: "graphql-ws@npm:5.16.0" + peerDependencies: + graphql: ">=0.11 <=16" + checksum: 6184e5b25e0b62f7cfd987176e09379fd7d8c900fc474a9cc1d79403a852ee49622e62c43ca9e4738113f9f7c29bcd82a80414f06c5cc9d2322d6eb846d20e59 + languageName: node + linkType: hard + "graphql@npm:16.1.0-experimental-stream-defer.6": version: 16.1.0-experimental-stream-defer.6 resolution: "graphql@npm:16.1.0-experimental-stream-defer.6" @@ -12587,6 +12607,13 @@ __metadata: languageName: node linkType: hard +"hono@npm:^4.6.15": + version: 4.6.15 + resolution: "hono@npm:4.6.15" + checksum: 3c9c4be676e90929eeec97cbea48786c4c76bc32c3a5fcc445db877acecd030c0091bfecd9b7463b9a38cf418c9c7745b37d92c9a2094250c8d7c3babe03f4e6 + languageName: node + linkType: hard + "hosted-git-info@npm:^2.1.4": version: 2.8.9 resolution: "hosted-git-info@npm:2.8.9" From 5161f52589b89fe9aac9c1fc0d77993df1715b5a Mon Sep 17 00:00:00 2001 From: loup Date: Sat, 4 Jan 2025 00:36:01 +0100 Subject: [PATCH 02/10] chore: add documentation and pass linter --- .../grafserv/__tests__/hono-adapter.test.ts | 13 +++++---- grafast/grafserv/src/servers/hono/index.ts | 13 ++------- grafast/website/grafserv/servers/hono.md | 29 +++++++++++++++++++ 3 files changed, 38 insertions(+), 17 deletions(-) create mode 100644 grafast/website/grafserv/servers/hono.md diff --git a/grafast/grafserv/__tests__/hono-adapter.test.ts b/grafast/grafserv/__tests__/hono-adapter.test.ts index 87d2cb7ef3..0e3dc7a67c 100644 --- a/grafast/grafserv/__tests__/hono-adapter.test.ts +++ b/grafast/grafserv/__tests__/hono-adapter.test.ts @@ -1,12 +1,13 @@ -import { Hono } from "hono"; -import { serverAudits } from "graphql-http"; -import { grafserv } from "../src/servers/hono/index.js"; -import type { GrafservConfig } from "../src/interfaces.js"; -import { constant, makeGrafastSchema } from "grafast"; -import { error } from "console"; import { serve } from "@hono/node-server"; +import { error } from "console"; +import { constant, makeGrafastSchema } from "grafast"; +import { serverAudits } from "graphql-http"; +import { Hono } from "hono"; import { AddressInfo } from "net"; +import type { GrafservConfig } from "../src/interfaces.js"; +import { grafserv } from "../src/servers/hono/index.js"; + const schema = makeGrafastSchema({ typeDefs: /* GraphQL */ ` type Query { diff --git a/grafast/grafserv/src/servers/hono/index.ts b/grafast/grafserv/src/servers/hono/index.ts index 4975995f86..4bb6617be8 100644 --- a/grafast/grafserv/src/servers/hono/index.ts +++ b/grafast/grafserv/src/servers/hono/index.ts @@ -1,28 +1,19 @@ -import { PassThrough } from "node:stream"; -import { Hono, Context as HonoContext } from "hono"; - -//@ts-expect-error type imports. -import type { Hooks, Peer } from "crossws"; -import { GRAPHQL_TRANSPORT_WS_PROTOCOL, makeServer } from "graphql-ws"; -import type { App, H3Event } from "h3"; +import type { Context as HonoContext, Hono } from "hono"; +import type { StatusCode } from "hono/utils/http-status"; import { convertHandlerResultToResult, GrafservBase, - makeGraphQLWSConfig, normalizeRequest, processHeaders, } from "../../index.js"; import type { EventStreamHeandlerResult, - GrafservBodyBuffer, GrafservBodyJSON, GrafservConfig, RequestDigest, Result, } from "../../interfaces.js"; -import { StatusCode } from "hono/utils/http-status"; -import { UpgradeWebSocket } from "hono/ws"; declare global { namespace Grafast { diff --git a/grafast/website/grafserv/servers/hono.md b/grafast/website/grafserv/servers/hono.md new file mode 100644 index 0000000000..d870784b37 --- /dev/null +++ b/grafast/website/grafserv/servers/hono.md @@ -0,0 +1,29 @@ +# Hono + +**THIS INTEGRATION IS EXPERIMENTAL**. PRs improving it are welcome. + +For now, websocket support is not available. + +```ts +import { grafserv } from "grafserv/hono"; +import preset from "./graphile.config.mjs"; +import schema from "./schema.mjs"; + +import { Hono } from "hono"; +import { serve } from "@hono/node-server"; + +// Create a Node HTTP server +const app = new Hono(); + +// Create a Grafserv instance +const serv = grafserv({ schema, preset }); + +// Mount the request handler into a new HTTP server +serv.addTo(server).catch((e) => { + console.error(e); + process.exit(1); +}); + +// Start the server with the chosen Hono adapter - here Node.js +serve(app); +``` From 51b553954444310f4db8294b3dfc3522c2c1b94b Mon Sep 17 00:00:00 2001 From: loup Date: Sun, 5 Jan 2025 01:25:24 +0100 Subject: [PATCH 03/10] feat: process websocket queries --- .../grafserv/__tests__/hono-adapter.test.ts | 60 ++++++++++++- grafast/grafserv/examples/example-hono.mjs | 4 +- grafast/grafserv/package.json | 3 +- grafast/grafserv/src/servers/hono/index.ts | 85 +++++++++++++++---- grafast/website/grafserv/servers/hono.md | 4 +- package.json | 4 + yarn.lock | 27 +++++- 7 files changed, 166 insertions(+), 21 deletions(-) diff --git a/grafast/grafserv/__tests__/hono-adapter.test.ts b/grafast/grafserv/__tests__/hono-adapter.test.ts index 0e3dc7a67c..1c4c7388c0 100644 --- a/grafast/grafserv/__tests__/hono-adapter.test.ts +++ b/grafast/grafserv/__tests__/hono-adapter.test.ts @@ -3,10 +3,12 @@ import { error } from "console"; import { constant, makeGrafastSchema } from "grafast"; import { serverAudits } from "graphql-http"; import { Hono } from "hono"; -import { AddressInfo } from "net"; +import { createNodeWebSocket } from "@hono/node-ws"; +import { createClient } from "graphql-ws"; import type { GrafservConfig } from "../src/interfaces.js"; import { grafserv } from "../src/servers/hono/index.js"; +import { WebSocket } from "ws"; const schema = makeGrafastSchema({ typeDefs: /* GraphQL */ ` @@ -14,6 +16,10 @@ const schema = makeGrafastSchema({ hello: String! throwAnError: String } + + type Subscription { + subscriptionTest: String! + } `, plans: { Query: { @@ -24,6 +30,15 @@ const schema = makeGrafastSchema({ return error(new Error("You asked for an error... Here it is.")); }, }, + Subscription: { + subscriptionTest: { + subscribe: async function* () { + console.log("subscriptionTest"); + yield { subscriptionTest: "test1" }; + yield { subscriptionTest: "test2" }; + }, + }, + }, }, }); @@ -87,3 +102,46 @@ describe("Hono Adapter", () => { }); } }); + +describe("Hono Adapter with websockets", () => { + // setup test server + const app = new Hono(); + const config: GrafservConfig = { + schema, // Mock schema for testing + preset: { + grafserv: { + graphqlOverGET: true, + websockets: true, + }, + }, + }; + const { injectWebSocket, upgradeWebSocket } = createNodeWebSocket({ app }); + + const honoGrafserv = grafserv(config, upgradeWebSocket); + honoGrafserv.addTo(app); + + const server = serve({ + fetch: app.fetch, + port: 7778, + }); + injectWebSocket(server); + + const url = `ws://0.0.0.0:7778/graphql`; + + it("SHOULD work for a simple subscription", async () => { + // make a graphql subscription + const client = createClient({ + url, + webSocketImpl: WebSocket, + }); + + const query = client.iterate({ + query: "subscription { subscriptionTest }", + }); + + const { value } = await query.next(); + expect(value).toEqual({ data: { subscriptionTest: "test1" } }); + const { value: value2 } = await query.next(); + expect(value2).toEqual({ data: { subscriptionTest: "test2" } }); + }); +}); diff --git a/grafast/grafserv/examples/example-hono.mjs b/grafast/grafserv/examples/example-hono.mjs index 11fcf3c312..933b55b303 100644 --- a/grafast/grafserv/examples/example-hono.mjs +++ b/grafast/grafserv/examples/example-hono.mjs @@ -9,6 +9,8 @@ import { serve } from "@hono/node-server"; const app = new Hono(); // Create a Grafserv instance +// the second argument is an optional websocket upgrade handler +// see https://hono.dev/docs/helpers/websocket const serv = grafserv({ schema, preset }); // Mount the request handler into a new HTTP server @@ -17,5 +19,5 @@ serv.addTo(server).catch((e) => { process.exit(1); }); -// Start the Node server +// Start the server with the chosen Hono adapter - here Node.js serve(app); diff --git a/grafast/grafserv/package.json b/grafast/grafserv/package.json index 92b4035b07..a9cd47b22f 100644 --- a/grafast/grafserv/package.json +++ b/grafast/grafserv/package.json @@ -132,7 +132,8 @@ "koa-bodyparser": "^4.4.1", "nodemon": "^3.0.1", "ts-node": "^10.9.1", - "typescript": "^5.2.2" + "typescript": "^5.2.2", + "ws": "^8.12.1" }, "files": [ "dist", diff --git a/grafast/grafserv/src/servers/hono/index.ts b/grafast/grafserv/src/servers/hono/index.ts index 4bb6617be8..28f3590b9b 100644 --- a/grafast/grafserv/src/servers/hono/index.ts +++ b/grafast/grafserv/src/servers/hono/index.ts @@ -1,9 +1,10 @@ -import type { Context as HonoContext, Hono } from "hono"; +import type { Context as HonoContext, Hono, MiddlewareHandler } from "hono"; import type { StatusCode } from "hono/utils/http-status"; import { convertHandlerResultToResult, GrafservBase, + makeGraphQLWSConfig, normalizeRequest, processHeaders, } from "../../index.js"; @@ -14,6 +15,9 @@ import type { RequestDigest, Result, } from "../../interfaces.js"; +import { GRAPHQL_TRANSPORT_WS_PROTOCOL, makeServer } from "graphql-ws"; +import { UpgradeWebSocket } from "hono/ws"; +import { web } from "webpack"; declare global { namespace Grafast { @@ -62,10 +66,50 @@ function getDigest(honoContext: HonoContext): RequestDigest { } export class HonoGrafserv extends GrafservBase { - constructor(config: GrafservConfig) { + constructor( + config: GrafservConfig, + private upgradeWebSocket?: UpgradeWebSocket, + ) { super(config); } + public makeWsHandler(upgradeWebSocket: UpgradeWebSocket): MiddlewareHandler { + const graphqlWsServer = makeServer(makeGraphQLWSConfig(this)); + return upgradeWebSocket((c) => { + let onMessage: ((data: string) => void) | undefined; + let onClose: ((code: number, reason: string) => void) | undefined; + return { + onOpen(evt, ws) { + onClose = graphqlWsServer.opened( + { + protocol: ws.protocol ?? GRAPHQL_TRANSPORT_WS_PROTOCOL, + send(data) { + ws.send(data); + }, + close(code, reason) { + console.log("close", code, reason); + ws.close(code, reason); + }, + onMessage(cb) { + onMessage = cb; + }, + }, + { socket: ws, request: c.req }, + ); + }, + onMessage(evt, ws) { + onMessage?.(evt.data); + }, + onClose(evt, ws) { + onClose?.(evt.code, evt.reason); + }, + onError(evt, ws) { + console.error("An error occured in the websocket:", evt); + }, + }; + }); + } + /** * @deprecated use handleGraphQLEvent instead */ @@ -156,19 +200,27 @@ export class HonoGrafserv extends GrafservBase { public async addTo(app: Hono) { const dynamicOptions = this.dynamicOptions; - app.on( - this.dynamicOptions.graphqlOverGET || - this.dynamicOptions.graphiqlOnGraphQLGET - ? ["GET", "POST"] - : ["POST"], - this.dynamicOptions.graphqlPath, - (c) => this.handleGraphQLEvent(c), + app.post(this.dynamicOptions.graphqlPath, (c) => + this.handleGraphQLEvent(c), ); - if (this.resolvedPreset.grafserv?.websockets) { - console.warn( - "You have enabled websockets, but the hono adapter doesn't support websockets yet.", - ); + const websocketHandler = + this.resolvedPreset.grafserv?.websockets && this.upgradeWebSocket + ? this.makeWsHandler(this.upgradeWebSocket) + : undefined; + + const shouldServeGetHandler = + this.dynamicOptions.graphqlOverGET || + this.dynamicOptions.graphiqlOnGraphQLGET || + websocketHandler; + + if (shouldServeGetHandler) { + app.get(this.dynamicOptions.graphqlPath, (c, next) => { + if (c.req.header("Upgrade") === "websocket" && websocketHandler) { + return websocketHandler(c, next); + } + return this.handleGraphQLEvent(c); + }); } if (dynamicOptions.graphiql) { @@ -194,6 +246,9 @@ export class HonoGrafserv extends GrafservBase { } } -export function grafserv(config: GrafservConfig) { - return new HonoGrafserv(config); +export function grafserv( + config: GrafservConfig, + upgradeWebSocket?: UpgradeWebSocket, +) { + return new HonoGrafserv(config, upgradeWebSocket); } diff --git a/grafast/website/grafserv/servers/hono.md b/grafast/website/grafserv/servers/hono.md index d870784b37..21265ba3a2 100644 --- a/grafast/website/grafserv/servers/hono.md +++ b/grafast/website/grafserv/servers/hono.md @@ -2,8 +2,6 @@ **THIS INTEGRATION IS EXPERIMENTAL**. PRs improving it are welcome. -For now, websocket support is not available. - ```ts import { grafserv } from "grafserv/hono"; import preset from "./graphile.config.mjs"; @@ -16,6 +14,8 @@ import { serve } from "@hono/node-server"; const app = new Hono(); // Create a Grafserv instance +// the second argument is an optional websocket upgrade handler +// see https://hono.dev/docs/helpers/websocket const serv = grafserv({ schema, preset }); // Mount the request handler into a new HTTP server diff --git a/package.json b/package.json index fd7a7b3aa5..f23cb55405 100644 --- a/package.json +++ b/package.json @@ -36,6 +36,7 @@ "@babel/preset-env": "^7.25.4", "@babel/preset-react": "^7.24.7", "@babel/preset-typescript": "^7.24.7", + "@hono/node-ws": "^1.0.5", "@knodes/typedoc-plugin-monorepo-readmes": "^0.23.1", "@knodes/typedoc-plugin-pages": "^0.23.4", "@localrepo/prettier2-for-jest": "npm:prettier@^2", @@ -44,6 +45,7 @@ "@types/mock-fs": "4.13.1", "@types/node": "^20.5.7", "@types/rimraf": "^4.0.5", + "@types/ws": "^8", "@typescript-eslint/eslint-plugin": "^6.5.0", "@typescript-eslint/parser": "^6.5.0", "@typescript-eslint/typescript-estree": "^6.5.0", @@ -63,6 +65,7 @@ "eslint_d": "^12.2.1", "glob": "^10.3.4", "graphql": "16.1.0-experimental-stream-defer.6", + "graphql-ws": "^5.16.0", "jest": "^29.6.4", "mock-fs": "^5.2.0", "pg": "^8.11.3", @@ -72,6 +75,7 @@ "typedoc-monorepo-link-types": "^0.0.4", "typescript": "^5.2.2", "webpack": "^5.94.0", + "ws": "^8.18.0", "zx": "^7.2.3" }, "workspaces": [ diff --git a/yarn.lock b/yarn.lock index dc1d369d83..3de87f09fd 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3504,6 +3504,17 @@ __metadata: languageName: node linkType: hard +"@hono/node-ws@npm:^1.0.5": + version: 1.0.5 + resolution: "@hono/node-ws@npm:1.0.5" + dependencies: + ws: "npm:^8.17.0" + peerDependencies: + "@hono/node-server": ^1.11.1 + checksum: 0126850d73984ab71211c3841731fc493abaf92a8d15a17e36d5ddda8d85d04efcd29c0cff894cad19dc69c6bcca8ebdae9ed0bc43e3782322474231db5dc604 + languageName: node + linkType: hard + "@humanwhocodes/config-array@npm:^0.11.10": version: 0.11.11 resolution: "@humanwhocodes/config-array@npm:0.11.11" @@ -6045,6 +6056,15 @@ __metadata: languageName: node linkType: hard +"@types/ws@npm:^8": + version: 8.5.13 + resolution: "@types/ws@npm:8.5.13" + dependencies: + "@types/node": "npm:*" + checksum: 8604665cd238ae0aae420fbdbb8065d8cefb5eb461a5108022102ace01eac662bf29a0a440bd6caf8f135d5ffac470e668705d44140e6b6a9d43cf561dff5a36 + languageName: node + linkType: hard + "@types/ws@npm:^8.5.5": version: 8.5.5 resolution: "@types/ws@npm:8.5.5" @@ -12004,6 +12024,7 @@ __metadata: ts-node: "npm:^10.9.1" tslib: "npm:^2.6.2" typescript: "npm:^5.2.2" + ws: "npm:^8.12.1" peerDependencies: "@envelop/core": ^5.0.0 grafast: "workspace:^" @@ -19100,6 +19121,7 @@ __metadata: "@babel/preset-typescript": "npm:^7.24.7" "@changesets/changelog-github": "npm:^0.4.8" "@changesets/cli": "npm:^2.26.2" + "@hono/node-ws": "npm:^1.0.5" "@knodes/typedoc-plugin-monorepo-readmes": "npm:^0.23.1" "@knodes/typedoc-plugin-pages": "npm:^0.23.4" "@localrepo/prettier2-for-jest": "npm:prettier@^2" @@ -19108,6 +19130,7 @@ __metadata: "@types/mock-fs": "npm:4.13.1" "@types/node": "npm:^20.5.7" "@types/rimraf": "npm:^4.0.5" + "@types/ws": "npm:^8" "@typescript-eslint/eslint-plugin": "npm:^6.5.0" "@typescript-eslint/parser": "npm:^6.5.0" "@typescript-eslint/typescript-estree": "npm:^6.5.0" @@ -19127,6 +19150,7 @@ __metadata: eslint_d: "npm:^12.2.1" glob: "npm:^10.3.4" graphql: "npm:16.1.0-experimental-stream-defer.6" + graphql-ws: "npm:^5.16.0" jest: "npm:^29.6.4" mock-fs: "npm:^5.2.0" pg: "npm:^8.11.3" @@ -19136,6 +19160,7 @@ __metadata: typedoc-monorepo-link-types: "npm:^0.0.4" typescript: "npm:^5.2.2" webpack: "npm:^5.94.0" + ws: "npm:^8.18.0" zx: "npm:^7.2.3" languageName: unknown linkType: soft @@ -22426,7 +22451,7 @@ __metadata: languageName: node linkType: hard -"ws@npm:^8.0.0, ws@npm:^8.13.0, ws@npm:^8.17.1": +"ws@npm:^8.0.0, ws@npm:^8.12.1, ws@npm:^8.13.0, ws@npm:^8.17.0, ws@npm:^8.17.1, ws@npm:^8.18.0": version: 8.18.0 resolution: "ws@npm:8.18.0" peerDependencies: From cef0a4b3750a40d7ae4e8cd92997fc36ce5af259 Mon Sep 17 00:00:00 2001 From: loup Date: Sun, 5 Jan 2025 13:54:42 +0100 Subject: [PATCH 04/10] fix: directory structure with version --- grafast/grafserv/__tests__/hono-adapter.test.ts | 2 +- grafast/grafserv/package.json | 8 ++++++-- grafast/grafserv/src/servers/hono/{ => v4}/index.ts | 4 ++-- yarn.lock | 5 ++++- 4 files changed, 13 insertions(+), 6 deletions(-) rename grafast/grafserv/src/servers/hono/{ => v4}/index.ts (99%) diff --git a/grafast/grafserv/__tests__/hono-adapter.test.ts b/grafast/grafserv/__tests__/hono-adapter.test.ts index 1c4c7388c0..68eb276796 100644 --- a/grafast/grafserv/__tests__/hono-adapter.test.ts +++ b/grafast/grafserv/__tests__/hono-adapter.test.ts @@ -7,7 +7,7 @@ import { createNodeWebSocket } from "@hono/node-ws"; import { createClient } from "graphql-ws"; import type { GrafservConfig } from "../src/interfaces.js"; -import { grafserv } from "../src/servers/hono/index.js"; +import { grafserv } from "../src/servers/hono/v4/index.js"; import { WebSocket } from "ws"; const schema = makeGrafastSchema({ diff --git a/grafast/grafserv/package.json b/grafast/grafserv/package.json index a9cd47b22f..6518be6f51 100644 --- a/grafast/grafserv/package.json +++ b/grafast/grafserv/package.json @@ -85,8 +85,7 @@ "debug": "^4.3.4", "eventemitter3": "^5.0.1", "graphile-config": "workspace:^", - "graphql-ws": "^5.16.0", - "hono": "^4.6.15", + "graphql-ws": "^5.14.0", "ruru": "workspace:^", "tslib": "^2.6.2" }, @@ -99,6 +98,7 @@ "graphile-config": "workspace:^", "graphql": "^16.1.0-experimental-stream-defer.6", "h3": "^1.13.0", + "hono": "^4.6.15", "ws": "^8.12.1" }, "peerDependenciesMeta": { @@ -108,6 +108,9 @@ "h3": { "optional": true }, + "hono": { + "optional": true + }, "ws": { "optional": true } @@ -126,6 +129,7 @@ "grafast": "workspace:^", "graphql-http": "^1.22.0", "h3": "^1.13.0", + "hono": "^4.6.15", "jest": "^29.6.4", "jest-serializer-graphql-schema": "workspace:^", "koa": "^2.14.2", diff --git a/grafast/grafserv/src/servers/hono/index.ts b/grafast/grafserv/src/servers/hono/v4/index.ts similarity index 99% rename from grafast/grafserv/src/servers/hono/index.ts rename to grafast/grafserv/src/servers/hono/v4/index.ts index 28f3590b9b..ae3e242178 100644 --- a/grafast/grafserv/src/servers/hono/index.ts +++ b/grafast/grafserv/src/servers/hono/v4/index.ts @@ -7,14 +7,14 @@ import { makeGraphQLWSConfig, normalizeRequest, processHeaders, -} from "../../index.js"; +} from "../../../index.js"; import type { EventStreamHeandlerResult, GrafservBodyJSON, GrafservConfig, RequestDigest, Result, -} from "../../interfaces.js"; +} from "../../../interfaces.js"; import { GRAPHQL_TRANSPORT_WS_PROTOCOL, makeServer } from "graphql-ws"; import { UpgradeWebSocket } from "hono/ws"; import { web } from "webpack"; diff --git a/yarn.lock b/yarn.lock index 3de87f09fd..4f3b5b6f13 100644 --- a/yarn.lock +++ b/yarn.lock @@ -12012,7 +12012,7 @@ __metadata: grafast: "workspace:^" graphile-config: "workspace:^" graphql-http: "npm:^1.22.0" - graphql-ws: "npm:^5.16.0" + graphql-ws: "npm:^5.14.0" h3: "npm:^1.13.0" hono: "npm:^4.6.15" jest: "npm:^29.6.4" @@ -12031,12 +12031,15 @@ __metadata: graphile-config: "workspace:^" graphql: ^16.1.0-experimental-stream-defer.6 h3: ^1.13.0 + hono: ^4.6.15 ws: ^8.12.1 peerDependenciesMeta: "@envelop/core": optional: true h3: optional: true + hono: + optional: true ws: optional: true languageName: unknown From 4fc61033711d0b5f3607cc3513b2a627d4567faf Mon Sep 17 00:00:00 2001 From: loup Date: Sun, 5 Jan 2025 21:14:26 +0100 Subject: [PATCH 05/10] chore: manage cloudflare websocket runtime, where onOpen event is not available --- grafast/grafserv/src/servers/h3/v1/index.ts | 2 +- grafast/grafserv/src/servers/hono/v4/index.ts | 137 ++++++++++-------- 2 files changed, 76 insertions(+), 63 deletions(-) diff --git a/grafast/grafserv/src/servers/h3/v1/index.ts b/grafast/grafserv/src/servers/h3/v1/index.ts index ace622827e..4a8c5d0f1c 100644 --- a/grafast/grafserv/src/servers/h3/v1/index.ts +++ b/grafast/grafserv/src/servers/h3/v1/index.ts @@ -278,7 +278,7 @@ export class H3Grafserv extends GrafservBase { { socket: peer.websocket, request: peer.request }, ); client.closed = async (code, reason) => { - onClose(code, reason); + onClose(code as number, reason as string); }; }, message(peer, message) { diff --git a/grafast/grafserv/src/servers/hono/v4/index.ts b/grafast/grafserv/src/servers/hono/v4/index.ts index ae3e242178..e04c9acb96 100644 --- a/grafast/grafserv/src/servers/hono/v4/index.ts +++ b/grafast/grafserv/src/servers/hono/v4/index.ts @@ -1,5 +1,7 @@ -import type { Context as HonoContext, Hono, MiddlewareHandler } from "hono"; +import { GRAPHQL_TRANSPORT_WS_PROTOCOL, makeServer } from "graphql-ws"; +import type { Context as Ctx, Hono, MiddlewareHandler } from "hono"; import type { StatusCode } from "hono/utils/http-status"; +import type { UpgradeWebSocket, WSContext } from "hono/ws"; import { convertHandlerResultToResult, @@ -15,27 +17,24 @@ import type { RequestDigest, Result, } from "../../../interfaces.js"; -import { GRAPHQL_TRANSPORT_WS_PROTOCOL, makeServer } from "graphql-ws"; -import { UpgradeWebSocket } from "hono/ws"; -import { web } from "webpack"; declare global { namespace Grafast { interface RequestContext { - Hono: { - honoContext: HonoContext; + honov4: { + ctx: Ctx; }; } } } -function getDigest(honoContext: HonoContext): RequestDigest { - const req = honoContext.req; - const res = honoContext.res; +function getDigest(ctx: Ctx): RequestDigest { + const req = ctx.req; + const res = ctx.res; return { httpVersionMajor: 1, // Hono uses Fetch API, which doesn't expose HTTP version httpVersionMinor: 1, - isSecure: req.url.startsWith("https"), + isSecure: req.url.startsWith("https:"), method: req.method, path: req.path, headers: processHeaders(req.header()), @@ -53,8 +52,8 @@ function getDigest(honoContext: HonoContext): RequestDigest { } as GrafservBodyJSON; }, requestContext: { - hono: { - context: honoContext, + honov4: { + ctx: ctx, }, node: { // @ts-expect-error type imports @@ -78,32 +77,45 @@ export class HonoGrafserv extends GrafservBase { return upgradeWebSocket((c) => { let onMessage: ((data: string) => void) | undefined; let onClose: ((code: number, reason: string) => void) | undefined; + let isOpened = false; + + const initGraphqlServer = (ws: WSContext) => { + onClose = graphqlWsServer.opened( + { + protocol: ws.protocol ?? GRAPHQL_TRANSPORT_WS_PROTOCOL, + send(data) { + ws.send(data); + }, + close(code, reason) { + console.log("close", code, reason); + ws.close(code, reason); + isOpened = false; + }, + onMessage(cb) { + onMessage = cb; + }, + }, + { socket: ws, request: c.req }, + ); + isOpened = true; + }; + return { onOpen(evt, ws) { - onClose = graphqlWsServer.opened( - { - protocol: ws.protocol ?? GRAPHQL_TRANSPORT_WS_PROTOCOL, - send(data) { - ws.send(data); - }, - close(code, reason) { - console.log("close", code, reason); - ws.close(code, reason); - }, - onMessage(cb) { - onMessage = cb; - }, - }, - { socket: ws, request: c.req }, - ); + initGraphqlServer(ws); }, onMessage(evt, ws) { + // cloudflare workers don't support the open event + // so we initialize the server on the first message + if (!isOpened) { + initGraphqlServer(ws); + } onMessage?.(evt.data); }, - onClose(evt, ws) { + onClose(evt) { onClose?.(evt.code, evt.reason); }, - onError(evt, ws) { + onError(evt) { console.error("An error occured in the websocket:", evt); }, }; @@ -113,31 +125,31 @@ export class HonoGrafserv extends GrafservBase { /** * @deprecated use handleGraphQLEvent instead */ - public async handleEvent(honoContext: HonoContext) { - return this.handleGraphQLEvent(honoContext); + public async handleEvent(ctx: Ctx) { + return this.handleGraphQLEvent(ctx); } - public async handleGraphQLEvent(honoContext: HonoContext) { - const digest = getDigest(honoContext); + public async handleGraphQLEvent(ctx: Ctx) { + const digest = getDigest(ctx); const handlerResult = await this.graphqlHandler( normalizeRequest(digest), this.graphiqlHandler, ); const result = await convertHandlerResultToResult(handlerResult); - return this.send(honoContext, result); + return this.send(ctx, result); } - public async handleGraphiqlEvent(honoContext: HonoContext) { - const digest = getDigest(honoContext); + public async handleGraphiqlEvent(ctx: Ctx) { + const digest = getDigest(ctx); const handlerResult = await this.graphiqlHandler(normalizeRequest(digest)); const result = await convertHandlerResultToResult(handlerResult); - return this.send(honoContext, result); + return this.send(ctx, result); } - public async handleEventStreamEvent(honoContext: HonoContext) { - const digest = getDigest(honoContext); + public async handleEventStreamEvent(ctx: Ctx) { + const digest = getDigest(ctx); const handlerResult: EventStreamHeandlerResult = { type: "event-stream", @@ -147,21 +159,21 @@ export class HonoGrafserv extends GrafservBase { statusCode: 200, }; const result = await convertHandlerResultToResult(handlerResult); - return this.send(honoContext, result); + return this.send(ctx, result); } - public async send(honoContext: HonoContext, result: Result | null) { + public async send(ctx: Ctx, result: Result | null) { if (result === null) { // 404 - honoContext.status(404); - return honoContext.text("¯\\_(ツ)_/¯"); + ctx.status(404); + return ctx.text("¯\\_(ツ)_/¯"); } switch (result.type) { case "error": { const { statusCode, headers } = result; - this.setResponseHeaders(honoContext, headers); - honoContext.status(statusCode as StatusCode); + this.setResponseHeaders(ctx, headers); + ctx.status(statusCode as StatusCode); const errorWithStatus = Object.assign(result.error, { status: statusCode, }); @@ -169,29 +181,29 @@ export class HonoGrafserv extends GrafservBase { } case "buffer": { const { statusCode, headers, buffer } = result; - this.setResponseHeaders(honoContext, headers); - honoContext.status(statusCode as StatusCode); - return honoContext.body(buffer); + this.setResponseHeaders(ctx, headers); + ctx.status(statusCode as StatusCode); + return ctx.body(buffer); } case "json": { const { statusCode, headers, json } = result; - this.setResponseHeaders(honoContext, headers); - honoContext.status(statusCode as StatusCode); - return honoContext.json(json); + this.setResponseHeaders(ctx, headers); + ctx.status(statusCode as StatusCode); + return ctx.json(json); } case "noContent": { const { statusCode, headers } = result; - this.setResponseHeaders(honoContext, headers); - honoContext.status(statusCode as StatusCode); - return honoContext.body(null); + this.setResponseHeaders(ctx, headers); + ctx.status(statusCode as StatusCode); + return ctx.body(null); } // TODO : handle bufferStream ? default: { const never = result; console.log("Unhandled:"); console.dir(never); - this.setResponseHeaders(honoContext, { "Content-Type": "text/plain" }); - honoContext.status(501); + this.setResponseHeaders(ctx, { "Content-Type": "text/plain" }); + ctx.status(501); return "Server hasn't implemented this yet"; } } @@ -236,18 +248,19 @@ export class HonoGrafserv extends GrafservBase { } } - private setResponseHeaders( - honoContext: HonoContext, - headers: Record, - ) { + private setResponseHeaders(ctx: Ctx, headers: Record) { for (const key in headers) { - honoContext.header(key, headers[key]); + ctx.header(key, headers[key]); } } } export function grafserv( config: GrafservConfig, + /** + * Required when using websockets. Hono uses upgradeWebsocket helper depending + * on the environment. Check https://hono.dev/docs/helpers/websocket + */ upgradeWebSocket?: UpgradeWebSocket, ) { return new HonoGrafserv(config, upgradeWebSocket); From 7a6b1dbeadf34b75e205a7220bbe9002b9742c80 Mon Sep 17 00:00:00 2001 From: loup Date: Sun, 5 Jan 2025 21:33:37 +0100 Subject: [PATCH 06/10] chore: add error message on misconfiguration and test for it --- .../grafserv/__tests__/hono-adapter.test.ts | 24 +++++++++++++++---- grafast/grafserv/src/servers/hono/v4/index.ts | 11 +++++++++ 2 files changed, 31 insertions(+), 4 deletions(-) diff --git a/grafast/grafserv/__tests__/hono-adapter.test.ts b/grafast/grafserv/__tests__/hono-adapter.test.ts index 68eb276796..3e10d1fbc5 100644 --- a/grafast/grafserv/__tests__/hono-adapter.test.ts +++ b/grafast/grafserv/__tests__/hono-adapter.test.ts @@ -1,14 +1,14 @@ import { serve } from "@hono/node-server"; +import { createNodeWebSocket } from "@hono/node-ws"; import { error } from "console"; import { constant, makeGrafastSchema } from "grafast"; import { serverAudits } from "graphql-http"; -import { Hono } from "hono"; -import { createNodeWebSocket } from "@hono/node-ws"; import { createClient } from "graphql-ws"; +import { Hono } from "hono"; +import { WebSocket } from "ws"; import type { GrafservConfig } from "../src/interfaces.js"; import { grafserv } from "../src/servers/hono/v4/index.js"; -import { WebSocket } from "ws"; const schema = makeGrafastSchema({ typeDefs: /* GraphQL */ ` @@ -32,8 +32,8 @@ const schema = makeGrafastSchema({ }, Subscription: { subscriptionTest: { + // eslint-disable-next-line graphile-export/export-methods subscribe: async function* () { - console.log("subscriptionTest"); yield { subscriptionTest: "test1" }; yield { subscriptionTest: "test2" }; }, @@ -144,4 +144,20 @@ describe("Hono Adapter with websockets", () => { const { value: value2 } = await query.next(); expect(value2).toEqual({ data: { subscriptionTest: "test2" } }); }); + + it("SHOULD throw an error is websocket is enabled but no upgradeWebSocket was provided", async () => { + const config: GrafservConfig = { + schema, // Mock schema for testing + preset: { + grafserv: { + graphqlOverGET: true, + websockets: true, + }, + }, + }; + const honoGrafserv = grafserv(config); + expect(async () => { + await honoGrafserv.addTo(app); + }).rejects.toThrow(); + }); }); diff --git a/grafast/grafserv/src/servers/hono/v4/index.ts b/grafast/grafserv/src/servers/hono/v4/index.ts index e04c9acb96..401b665d9e 100644 --- a/grafast/grafserv/src/servers/hono/v4/index.ts +++ b/grafast/grafserv/src/servers/hono/v4/index.ts @@ -212,6 +212,17 @@ export class HonoGrafserv extends GrafservBase { public async addTo(app: Hono) { const dynamicOptions = this.dynamicOptions; + if (this.resolvedPreset.grafserv?.websockets && !this.upgradeWebSocket) { + throw new Error( + "grafserv.websockets is enabled but no upgradeWebSocket was provided", + ); + } + if (!this.resolvedPreset.grafserv?.websockets && this.upgradeWebSocket) { + console.warn( + "UpgradeWebSocket was provided but grafserv.websockets is disabled - websockets will not be activated", + ); + } + app.post(this.dynamicOptions.graphqlPath, (c) => this.handleGraphQLEvent(c), ); From ffee72ff823599a0eb3ab28ca8c454c774853eeb Mon Sep 17 00:00:00 2001 From: loup Date: Sun, 5 Jan 2025 21:52:09 +0100 Subject: [PATCH 07/10] chore: jsdoc for clarification of the last parameter --- .../grafserv/__tests__/hono-adapter.test.ts | 1 - grafast/grafserv/examples/example-hono.mjs | 6 +++--- grafast/grafserv/src/servers/hono/v4/index.ts | 19 ++++++++++++------- 3 files changed, 15 insertions(+), 11 deletions(-) diff --git a/grafast/grafserv/__tests__/hono-adapter.test.ts b/grafast/grafserv/__tests__/hono-adapter.test.ts index 3e10d1fbc5..aa99bcb46d 100644 --- a/grafast/grafserv/__tests__/hono-adapter.test.ts +++ b/grafast/grafserv/__tests__/hono-adapter.test.ts @@ -150,7 +150,6 @@ describe("Hono Adapter with websockets", () => { schema, // Mock schema for testing preset: { grafserv: { - graphqlOverGET: true, websockets: true, }, }, diff --git a/grafast/grafserv/examples/example-hono.mjs b/grafast/grafserv/examples/example-hono.mjs index 933b55b303..e11828c7d4 100644 --- a/grafast/grafserv/examples/example-hono.mjs +++ b/grafast/grafserv/examples/example-hono.mjs @@ -1,10 +1,10 @@ +import { serve } from "@hono/node-server"; import { grafserv } from "grafserv/hono"; +import { Hono } from "hono"; + import preset from "./graphile.config.mjs"; import schema from "./schema.mjs"; -import { Hono } from "hono"; -import { serve } from "@hono/node-server"; - // Create a Node HTTP server const app = new Hono(); diff --git a/grafast/grafserv/src/servers/hono/v4/index.ts b/grafast/grafserv/src/servers/hono/v4/index.ts index 401b665d9e..e82b4f7d19 100644 --- a/grafast/grafserv/src/servers/hono/v4/index.ts +++ b/grafast/grafserv/src/servers/hono/v4/index.ts @@ -266,13 +266,18 @@ export class HonoGrafserv extends GrafservBase { } } -export function grafserv( +/** + * Creates a new instance of HonoGrafserv. + * + * @param config - The configuration object for Grafserv. + * @param upgradeWebSocket - Optional parameter required when using websockets. + * Hono uses the upgradeWebsocket helper depending on the environment. + * Refer to https://hono.dev/docs/helpers/websocket for more details. + * @returns An instance of HonoGrafserv. + */ +export const grafserv = ( config: GrafservConfig, - /** - * Required when using websockets. Hono uses upgradeWebsocket helper depending - * on the environment. Check https://hono.dev/docs/helpers/websocket - */ upgradeWebSocket?: UpgradeWebSocket, -) { +) => { return new HonoGrafserv(config, upgradeWebSocket); -} +}; From 82973e442635578ce9b6e1bc298ac2204c1160c1 Mon Sep 17 00:00:00 2001 From: loup Date: Mon, 6 Jan 2025 09:23:22 +0100 Subject: [PATCH 08/10] fix: imports --- grafast/grafserv/package.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/grafast/grafserv/package.json b/grafast/grafserv/package.json index 6518be6f51..bb04f6370a 100644 --- a/grafast/grafserv/package.json +++ b/grafast/grafserv/package.json @@ -38,9 +38,9 @@ "types": "./dist/servers/h3/v1/index.d.ts", "default": "./dist/servers/h3/v1/index.js" }, - "./hono": { - "types": "./dist/hono/index.d.ts", - "default": "./dist/hono/index.js" + "./hono/v4": { + "types": "./dist/hono/v4/index.d.ts", + "default": "./dist/hono/v4/index.js" }, "./ruru": { "types": "./fwd/ruru/index.d.ts", From 8ca66b83d5d72bdfb4ded855c3eb1f091cfcdaee Mon Sep 17 00:00:00 2001 From: loup Date: Mon, 6 Jan 2025 09:46:13 +0100 Subject: [PATCH 09/10] fix: exports --- grafast/grafserv/package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/grafast/grafserv/package.json b/grafast/grafserv/package.json index bb04f6370a..3bee156d67 100644 --- a/grafast/grafserv/package.json +++ b/grafast/grafserv/package.json @@ -39,8 +39,8 @@ "default": "./dist/servers/h3/v1/index.js" }, "./hono/v4": { - "types": "./dist/hono/v4/index.d.ts", - "default": "./dist/hono/v4/index.js" + "types": "./dist/servers/hono/v4/index.d.ts", + "default": "./dist/servers/hono/v4/index.js" }, "./ruru": { "types": "./fwd/ruru/index.d.ts", From bf2a34744694bf2bbb49361a3fa22cf278748d20 Mon Sep 17 00:00:00 2001 From: loup Date: Mon, 6 Jan 2025 10:07:19 +0100 Subject: [PATCH 10/10] fix: unhandle response error --- grafast/grafserv/src/servers/hono/v4/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/grafast/grafserv/src/servers/hono/v4/index.ts b/grafast/grafserv/src/servers/hono/v4/index.ts index e82b4f7d19..7e97d44be9 100644 --- a/grafast/grafserv/src/servers/hono/v4/index.ts +++ b/grafast/grafserv/src/servers/hono/v4/index.ts @@ -204,7 +204,7 @@ export class HonoGrafserv extends GrafservBase { console.dir(never); this.setResponseHeaders(ctx, { "Content-Type": "text/plain" }); ctx.status(501); - return "Server hasn't implemented this yet"; + return ctx.text("Server hasn't implemented this yet"); } } }