Initial version #1
6
bun.lock
6
bun.lock
@@ -22,6 +22,8 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@effect/platform": "^0.90.8",
|
"@effect/platform": "^0.90.8",
|
||||||
"@effect/platform-bun": "^0.79.0",
|
"@effect/platform-bun": "^0.79.0",
|
||||||
|
"@effect/rpc": "^0.69.2",
|
||||||
|
"@website/common": "workspace:*",
|
||||||
"effect": "^3.17.13",
|
"effect": "^3.17.13",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -95,7 +97,7 @@
|
|||||||
|
|
||||||
"@effect/language-service": ["@effect/language-service@0.40.0", "", { "bin": { "effect-language-service": "cli.js" } }, "sha512-VHikOhYXm+ECy+mLSszHgCfP9YyKQyB9GLFGBKY2nbnru9xwgCnar8pBLA0AkSUjAgn3hc/mdcFdb/XL9uywLQ=="],
|
"@effect/language-service": ["@effect/language-service@0.40.0", "", { "bin": { "effect-language-service": "cli.js" } }, "sha512-VHikOhYXm+ECy+mLSszHgCfP9YyKQyB9GLFGBKY2nbnru9xwgCnar8pBLA0AkSUjAgn3hc/mdcFdb/XL9uywLQ=="],
|
||||||
|
|
||||||
"@effect/platform": ["@effect/platform@0.90.8", "", { "dependencies": { "find-my-way-ts": "^0.1.6", "msgpackr": "^1.11.4", "multipasta": "^0.2.7" }, "peerDependencies": { "effect": "^3.17.13" } }, "sha512-lQejPI2VbUAwPIU6oG12sAhqMar8+KkBjxgBVqQTERzHz4vnG9zaiDDytJM+IGWE40MKbCG3pc2wK6nw2Ugk8A=="],
|
"@effect/platform": ["@effect/platform@0.90.9", "", { "dependencies": { "find-my-way-ts": "^0.1.6", "msgpackr": "^1.11.4", "multipasta": "^0.2.7" }, "peerDependencies": { "effect": "^3.17.13" } }, "sha512-ZLyGMenOXGERsaJ7urioWwugkluWjhrcQgr0/vfgxqe/TCumL3CK93YZKSlJtohDrFSfWjaXCbXyhqz3iypFUg=="],
|
||||||
|
|
||||||
"@effect/platform-browser": ["@effect/platform-browser@0.70.0", "", { "dependencies": { "multipasta": "^0.2.7" }, "peerDependencies": { "@effect/platform": "^0.90.0", "effect": "^3.17.0" } }, "sha512-dq2ukUralavQIipSsQVuIOLLxBlFWL5Mkg1Fnr8esYPuPv0BXAI39nwNNi75X2tPAgak8OCSD0qh9qhmNj2gPA=="],
|
"@effect/platform-browser": ["@effect/platform-browser@0.70.0", "", { "dependencies": { "multipasta": "^0.2.7" }, "peerDependencies": { "@effect/platform": "^0.90.0", "effect": "^3.17.0" } }, "sha512-dq2ukUralavQIipSsQVuIOLLxBlFWL5Mkg1Fnr8esYPuPv0BXAI39nwNNi75X2tPAgak8OCSD0qh9qhmNj2gPA=="],
|
||||||
|
|
||||||
@@ -611,8 +613,6 @@
|
|||||||
|
|
||||||
"@typescript-eslint/typescript-estree/semver": ["semver@7.7.2", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA=="],
|
"@typescript-eslint/typescript-estree/semver": ["semver@7.7.2", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA=="],
|
||||||
|
|
||||||
"@website/webapp/@effect/platform": ["@effect/platform@0.90.9", "", { "dependencies": { "find-my-way-ts": "^0.1.6", "msgpackr": "^1.11.4", "multipasta": "^0.2.7" }, "peerDependencies": { "effect": "^3.17.13" } }, "sha512-ZLyGMenOXGERsaJ7urioWwugkluWjhrcQgr0/vfgxqe/TCumL3CK93YZKSlJtohDrFSfWjaXCbXyhqz3iypFUg=="],
|
|
||||||
|
|
||||||
"fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="],
|
"fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="],
|
||||||
|
|
||||||
"micromatch/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
|
"micromatch/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
|
||||||
|
|||||||
5
packages/common/src/webrpc/WebRpcSerialization.ts
Normal file
5
packages/common/src/webrpc/WebRpcSerialization.ts
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import { RpcSerialization } from "@effect/rpc"
|
||||||
|
|
||||||
|
|
||||||
|
export const WebRpcSerializationDevelopment = RpcSerialization.layerJson
|
||||||
|
export const WebRpcSerializationProduction = RpcSerialization.layerNdjson
|
||||||
@@ -1,2 +1,3 @@
|
|||||||
export * as WebRpc from "./WebRpc"
|
export * as WebRpc from "./WebRpc"
|
||||||
|
export * as WebRpcSerialization from "./WebRpcSerialization"
|
||||||
export * as WebRpcTest from "./WebRpcTest"
|
export * as WebRpcTest from "./WebRpcTest"
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@effect/platform": "^0.90.8",
|
"@effect/platform": "^0.90.8",
|
||||||
"@effect/platform-bun": "^0.79.0",
|
"@effect/platform-bun": "^0.79.0",
|
||||||
|
"@effect/rpc": "^0.69.2",
|
||||||
"@website/common": "workspace:*",
|
"@website/common": "workspace:*",
|
||||||
"effect": "^3.17.13"
|
"effect": "^3.17.13"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,10 +5,13 @@ import { Server } from "./server"
|
|||||||
|
|
||||||
|
|
||||||
Layer.launch(Server).pipe(
|
Layer.launch(Server).pipe(
|
||||||
Effect.provide(ServerConfig.httpPort.pipe(
|
Effect.provide(Layer.empty.pipe(
|
||||||
Effect.map(port => BunHttpServer.layer({ port })),
|
Layer.provideMerge(ServerConfig.httpPort.pipe(
|
||||||
Layer.unwrapEffect,
|
Effect.map(port => BunHttpServer.layer({ port })),
|
||||||
|
Layer.unwrapEffect,
|
||||||
|
)),
|
||||||
|
Layer.provideMerge(BunContext.layer),
|
||||||
)),
|
)),
|
||||||
Effect.provide(BunContext.layer),
|
|
||||||
BunRuntime.runMain,
|
BunRuntime.runMain,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,39 +1,66 @@
|
|||||||
import { FileSystem, HttpMiddleware, HttpRouter, HttpServer, HttpServerRequest, HttpServerResponse, Path } from "@effect/platform"
|
import { FileSystem, HttpMiddleware, HttpRouter, HttpServer, HttpServerRequest, HttpServerResponse, Path } from "@effect/platform"
|
||||||
import { Duration, Effect, flow } from "effect"
|
import { RpcServer } from "@effect/rpc"
|
||||||
|
import { CommonConfig } from "@website/common/config"
|
||||||
|
import { WebRpc, WebRpcSerialization } from "@website/common/webrpc"
|
||||||
|
import { Duration, Effect, flow, Layer } from "effect"
|
||||||
|
import { WebRpcLive } from "./webrpc"
|
||||||
|
|
||||||
|
|
||||||
const router = HttpRouter.empty
|
const router = HttpRouter.empty
|
||||||
|
|
||||||
export const HttpAppDevelopment = router.pipe(
|
const makeWebRpcRoute = Effect.all([
|
||||||
HttpServer.serve(flow(
|
CommonConfig.webRpcHttpPath,
|
||||||
HttpMiddleware.logger,
|
RpcServer.toHttpApp(WebRpc.WebRpc),
|
||||||
HttpMiddleware.xForwardedHeaders,
|
]).pipe(
|
||||||
)),
|
Effect.map(([path, app]) => HttpRouter.mountApp(path, app)),
|
||||||
HttpServer.withLogAddress,
|
Effect.provide(WebRpcLive.WebRpcLive),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const makeProductionWebappMiddleware = Effect.gen(function*() {
|
||||||
const serveProductionWebapp = HttpMiddleware.make(() => Effect.gen(function*() {
|
|
||||||
const path = yield* Path.Path
|
const path = yield* Path.Path
|
||||||
const fs = yield* FileSystem.FileSystem
|
const fs = yield* FileSystem.FileSystem
|
||||||
const req = yield* HttpServerRequest.HttpServerRequest
|
|
||||||
|
|
||||||
const dist = path.join(yield* path.fromFileUrl(new URL(".", import.meta.resolve("@website/webapp"))), "dist")
|
const dist = path.join(yield* path.fromFileUrl(new URL(".", import.meta.resolve("@website/webapp"))), "dist")
|
||||||
const source = path.join(dist, req.url)
|
|
||||||
const isValid = yield* fs.stat(source).pipe(
|
|
||||||
Effect.andThen(stat => stat.type === "File"),
|
|
||||||
Effect.catchAll(() => Effect.succeed(false)),
|
|
||||||
)
|
|
||||||
|
|
||||||
return yield* HttpServerResponse.setHeader(
|
return () => Effect.gen(function*() {
|
||||||
yield* HttpServerResponse.file(isValid ? source : path.join(dist, "index.html")),
|
const req = yield* HttpServerRequest.HttpServerRequest
|
||||||
"Cache-Control",
|
const source = path.join(dist, req.url)
|
||||||
`public, max-age=${Duration.toSeconds("365 days")}, immutable`
|
const isValid = yield* fs.stat(source).pipe(
|
||||||
)
|
Effect.andThen(stat => stat.type === "File"),
|
||||||
}))
|
Effect.catchAll(() => Effect.succeed(false)),
|
||||||
|
)
|
||||||
|
|
||||||
export const HttpAppProduction = router.pipe(
|
return yield* HttpServerResponse.setHeader(
|
||||||
serveProductionWebapp,
|
yield* HttpServerResponse.file(isValid ? source : path.join(dist, "index.html")),
|
||||||
HttpServer.serve(HttpMiddleware.xForwardedHeaders),
|
"Cache-Control",
|
||||||
HttpServer.withLogAddress,
|
`public, max-age=${Duration.toSeconds("365 days")}, immutable`
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
export const HttpAppDevelopment = Effect.provide(makeWebRpcRoute, WebRpcSerialization.WebRpcSerializationDevelopment).pipe(
|
||||||
|
Effect.map(serveWebRpc => router.pipe(
|
||||||
|
serveWebRpc,
|
||||||
|
HttpServer.serve(flow(
|
||||||
|
HttpMiddleware.logger,
|
||||||
|
HttpMiddleware.xForwardedHeaders,
|
||||||
|
)),
|
||||||
|
HttpServer.withLogAddress,
|
||||||
|
)),
|
||||||
|
|
||||||
|
Layer.unwrapScoped,
|
||||||
|
)
|
||||||
|
|
||||||
|
export const HttpAppProduction = Effect.all([
|
||||||
|
Effect.provide(makeWebRpcRoute, WebRpcSerialization.WebRpcSerializationProduction),
|
||||||
|
makeProductionWebappMiddleware,
|
||||||
|
]).pipe(
|
||||||
|
Effect.map(([serveWebRpc, serveProductionWebapp]) => router.pipe(
|
||||||
|
serveWebRpc,
|
||||||
|
serveProductionWebapp,
|
||||||
|
HttpServer.serve(HttpMiddleware.xForwardedHeaders),
|
||||||
|
HttpServer.withLogAddress,
|
||||||
|
)),
|
||||||
|
|
||||||
|
Layer.unwrapScoped,
|
||||||
)
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user