fix(realtime): CI gate fixes surfaced by Phase 11 smoke tests

- no-direct-socket-io: extend allowlist to cover apps/*/src/**/*.test.ts so
  the realtime-ping e2e integration test can import socket.io/socket.io-client
  directly; add two valid test cases to keep the rule's own test suite green
- tsconfig.json (root): add root-level tsconfig with experimentalDecorators +
  emitDecoratorMetadata and no "include" so tsx 4.21.0's createFilesMatcher
  resolves decorator config for all workspace packages, not just web-next's
  own source tree
- web-next dev script: pass TSX_TSCONFIG_PATH=../../tsconfig.json so the
  custom Node server uses the root tsconfig for all modules it loads
- next.config.mjs: add @repo/core-events and @repo/core-realtime to
  transpilePackages so Next.js webpack can resolve their workspace source files
- server.ts: replace static authContainer import with a dynamic import inside
  IRealtimeAuthenticator.authenticate so Inversify decorators are applied only
  after bindAll() has already populated the container

All CI gates pass: lint (0 errors), typecheck, 20 tests (incl. realtime-ping
e2e), boundaries (0 issues).

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-09 01:32:43 +02:00
parent cc53600339
commit 6a0ac63bb9
6 changed files with 19 additions and 2 deletions

View File

@@ -7,6 +7,8 @@ const nextConfig = {
"@repo/blog",
"@repo/core-api",
"@repo/core-cms",
"@repo/core-events",
"@repo/core-realtime",
"@repo/core-shared",
"@repo/core-trpc",
"@repo/core-ui",

View File

@@ -5,7 +5,7 @@
"type": "module",
"scripts": {
"build": "echo 'Next.js build requires full environment — use pnpm dev or docker'",
"dev": "tsx server.ts",
"dev": "TSX_TSCONFIG_PATH=../../tsconfig.json tsx server.ts",
"start": "node --import tsx server.ts",
"lint": "eslint .",
"test": "vitest run --passWithNoTests",

View File

@@ -11,7 +11,6 @@ import {
type IRealtimeAuthenticator,
} from "@repo/core-realtime";
import { SESSION_COOKIE } from "@repo/auth";
import { authContainer } from "@repo/auth/di/container";
import { AUTH_SYMBOLS } from "@repo/auth/di/symbols";
import { bindAll } from "./src/server/bind-production.js";
@@ -35,6 +34,11 @@ const authenticator: IRealtimeAuthenticator = {
authenticate: async ({ cookies }) => {
const sessionId = cookies[SESSION_COOKIE];
if (!sessionId) return null;
// Lazy-import the auth container after bindAll() has already populated it.
// Dynamic import defers tsx's module transformation to runtime, which lets
// reflect-metadata and the DI decorators resolve correctly in this server
// entry point.
const { authContainer } = await import("@repo/auth/di/container");
const authService = authContainer.get<{
validateSession: (id: string) => Promise<{ user: { id: string }; session: unknown } | null>;
}>(AUTH_SYMBOLS.IAuthenticationService);

View File

@@ -2,6 +2,7 @@
const ALLOWED = [
/\/packages\/core-realtime\/src\//,
/\/apps\/[^/]+\/server\.ts$/,
/\/apps\/[^/]+\/src\/.*\.test\.ts$/,
];
export default {

View File

@@ -12,6 +12,9 @@ tester.run("no-direct-socket-io", rule, {
{ code: 'import { Server } from "socket.io";', filename: "/repo/packages/core-realtime/src/socket-io-realtime-server.ts" },
// Allowed in app servers
{ code: 'import { Server } from "socket.io";', filename: "/repo/apps/web-next/server.ts" },
// Allowed in app integration tests (e.g. realtime-ping e2e)
{ code: 'import { Server } from "socket.io";', filename: "/repo/apps/web-next/src/__tests__/realtime-ping.test.ts" },
{ code: 'import { io } from "socket.io-client";', filename: "/repo/apps/web-next/src/__tests__/realtime-ping.test.ts" },
// Allowed elsewhere when not importing socket.io
{ code: 'import { foo } from "bar";', filename: "/repo/packages/blog/src/foo.ts" },
],

7
tsconfig.json Normal file
View File

@@ -0,0 +1,7 @@
{
"$schema": "https://json.schemastore.org/tsconfig",
"compilerOptions": {
"experimentalDecorators": true,
"emitDecoratorMetadata": true
}
}