@memberjunction/server 5.12.0 → 5.13.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,41 @@
1
+ /**
2
+ * MJ's built-in multi-tenancy middleware.
3
+ *
4
+ * Registers with key 'mj:tenantFilter' so that downstream layers (e.g., BCSaaS)
5
+ * can replace it by registering with the same key at higher ClassFactory priority.
6
+ *
7
+ * Conditionally enabled via configInfo.multiTenancy?.enabled.
8
+ */
9
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
10
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
11
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
12
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
13
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
14
+ };
15
+ import { RegisterClass } from '@memberjunction/global';
16
+ import { BaseServerMiddleware } from './BaseServerMiddleware.js';
17
+ import { configInfo } from '../config.js';
18
+ import { createTenantMiddleware, createTenantPreRunViewHook, createTenantPreSaveHook } from '../multiTenancy/index.js';
19
+ let MJTenantFilterMiddleware = class MJTenantFilterMiddleware extends BaseServerMiddleware {
20
+ get Label() { return 'mj:tenantFilter'; }
21
+ /**
22
+ * Only active when multiTenancy is enabled in config.
23
+ */
24
+ get Enabled() {
25
+ return configInfo.multiTenancy?.enabled ?? false;
26
+ }
27
+ GetPostAuthMiddleware() {
28
+ return [createTenantMiddleware(configInfo.multiTenancy)];
29
+ }
30
+ PreRunView(params, contextUser) {
31
+ return createTenantPreRunViewHook(configInfo.multiTenancy)(params, contextUser);
32
+ }
33
+ PreSave(entity, contextUser) {
34
+ return createTenantPreSaveHook(configInfo.multiTenancy)(entity, contextUser);
35
+ }
36
+ };
37
+ MJTenantFilterMiddleware = __decorate([
38
+ RegisterClass(BaseServerMiddleware, 'mj:tenantFilter')
39
+ ], MJTenantFilterMiddleware);
40
+ export { MJTenantFilterMiddleware };
41
+ //# sourceMappingURL=MJTenantFilterMiddleware.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MJTenantFilterMiddleware.js","sourceRoot":"","sources":["../../src/middleware/MJTenantFilterMiddleware.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;;;;;;;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC1C,OAAO,EAAE,sBAAsB,EAAE,0BAA0B,EAAE,uBAAuB,EAAE,MAAM,0BAA0B,CAAC;AAKhH,IAAM,wBAAwB,GAA9B,MAAM,wBAAyB,SAAQ,oBAAoB;IAC9D,IAAI,KAAK,KAAa,OAAO,iBAAiB,CAAC,CAAC,CAAC;IAEjD;;OAEG;IACH,IAAI,OAAO;QACP,OAAO,UAAU,CAAC,YAAY,EAAE,OAAO,IAAI,KAAK,CAAC;IACrD,CAAC;IAED,qBAAqB;QACjB,OAAO,CAAC,sBAAsB,CAAC,UAAU,CAAC,YAAa,CAAC,CAAC,CAAC;IAC9D,CAAC;IAED,UAAU,CAAC,MAAqB,EAAE,WAAiC;QAC/D,OAAO,0BAA0B,CAAC,UAAU,CAAC,YAAa,CAAC,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;IACrF,CAAC;IAED,OAAO,CAAC,MAAkB,EAAE,WAAiC;QACzD,OAAO,uBAAuB,CAAC,UAAU,CAAC,YAAa,CAAC,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;IAClF,CAAC;CACJ,CAAA;AArBY,wBAAwB;IADpC,aAAa,CAAC,oBAAoB,EAAE,iBAAiB,CAAC;GAC1C,wBAAwB,CAqBpC"}
@@ -0,0 +1,3 @@
1
+ export * from './BaseServerMiddleware.js';
2
+ export * from './MJTenantFilterMiddleware.js';
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/middleware/index.ts"],"names":[],"mappings":"AAAA,cAAc,2BAA2B,CAAC;AAC1C,cAAc,+BAA+B,CAAC"}
@@ -0,0 +1,3 @@
1
+ export * from './BaseServerMiddleware.js';
2
+ export * from './MJTenantFilterMiddleware.js';
3
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/middleware/index.ts"],"names":[],"mappings":"AAAA,cAAc,2BAA2B,CAAC;AAC1C,cAAc,+BAA+B,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@memberjunction/server",
3
- "version": "5.12.0",
3
+ "version": "5.13.0",
4
4
  "description": "MemberJunction: This project provides API access via GraphQL to the common data store.",
5
5
  "main": "./dist/index.js",
6
6
  "types": "./src/index.ts",
@@ -27,62 +27,62 @@
27
27
  "@as-integrations/express5": "^1.0.0",
28
28
  "@graphql-tools/schema": "latest",
29
29
  "@graphql-tools/utils": "^11.0.0",
30
- "@memberjunction/actions": "5.12.0",
31
- "@memberjunction/actions-base": "5.12.0",
32
- "@memberjunction/actions-apollo": "5.12.0",
33
- "@memberjunction/actions-bizapps-accounting": "5.12.0",
34
- "@memberjunction/actions-bizapps-crm": "5.12.0",
35
- "@memberjunction/actions-bizapps-formbuilders": "5.12.0",
36
- "@memberjunction/actions-bizapps-lms": "5.12.0",
37
- "@memberjunction/actions-bizapps-social": "5.12.0",
38
- "@memberjunction/ai": "5.12.0",
39
- "@memberjunction/ai-mcp-client": "5.12.0",
40
- "@memberjunction/ai-agent-manager": "5.12.0",
41
- "@memberjunction/ai-agent-manager-actions": "5.12.0",
42
- "@memberjunction/ai-agents": "5.12.0",
43
- "@memberjunction/ai-core-plus": "5.12.0",
44
- "@memberjunction/ai-prompts": "5.12.0",
45
- "@memberjunction/ai-provider-bundle": "5.12.0",
46
- "@memberjunction/ai-vectors-pinecone": "5.12.0",
47
- "@memberjunction/aiengine": "5.12.0",
48
- "@memberjunction/communication-ms-graph": "5.12.0",
49
- "@memberjunction/communication-sendgrid": "5.12.0",
50
- "@memberjunction/communication-types": "5.12.0",
51
- "@memberjunction/component-registry-client-sdk": "5.12.0",
52
- "@memberjunction/config": "5.12.0",
53
- "@memberjunction/core": "5.12.0",
54
- "@memberjunction/core-actions": "5.12.0",
55
- "@memberjunction/core-entities": "5.12.0",
56
- "@memberjunction/core-entities-server": "5.12.0",
57
- "@memberjunction/data-context": "5.12.0",
58
- "@memberjunction/data-context-server": "5.12.0",
59
- "@memberjunction/doc-utils": "5.12.0",
60
- "@memberjunction/api-keys": "5.12.0",
61
- "@memberjunction/encryption": "5.12.0",
62
- "@memberjunction/entity-communications-base": "5.12.0",
63
- "@memberjunction/entity-communications-server": "5.12.0",
64
- "@memberjunction/external-change-detection": "5.12.0",
65
- "@memberjunction/generic-database-provider": "5.12.0",
66
- "@memberjunction/global": "5.12.0",
67
- "@memberjunction/graphql-dataprovider": "5.12.0",
68
- "@memberjunction/integration-engine": "5.12.0",
69
- "@memberjunction/integration-schema-builder": "5.12.0",
70
- "@memberjunction/interactive-component-types": "5.12.0",
71
- "@memberjunction/computer-use-engine": "5.12.0",
72
- "@memberjunction/notifications": "5.12.0",
73
- "@memberjunction/queue": "5.12.0",
74
- "@memberjunction/redis-provider": "5.12.0",
75
- "@memberjunction/scheduling-actions": "5.12.0",
76
- "@memberjunction/scheduling-base-types": "5.12.0",
77
- "@memberjunction/scheduling-engine": "5.12.0",
78
- "@memberjunction/scheduling-engine-base": "5.12.0",
79
- "@memberjunction/skip-types": "5.12.0",
80
- "@memberjunction/sqlserver-dataprovider": "5.12.0",
81
- "@memberjunction/storage": "5.12.0",
82
- "@memberjunction/templates": "5.12.0",
83
- "@memberjunction/testing-engine": "5.12.0",
84
- "@memberjunction/testing-engine-base": "5.12.0",
85
- "@memberjunction/version-history": "5.12.0",
30
+ "@memberjunction/actions": "5.13.0",
31
+ "@memberjunction/actions-base": "5.13.0",
32
+ "@memberjunction/actions-apollo": "5.13.0",
33
+ "@memberjunction/actions-bizapps-accounting": "5.13.0",
34
+ "@memberjunction/actions-bizapps-crm": "5.13.0",
35
+ "@memberjunction/actions-bizapps-formbuilders": "5.13.0",
36
+ "@memberjunction/actions-bizapps-lms": "5.13.0",
37
+ "@memberjunction/actions-bizapps-social": "5.13.0",
38
+ "@memberjunction/ai": "5.13.0",
39
+ "@memberjunction/ai-mcp-client": "5.13.0",
40
+ "@memberjunction/ai-agent-manager": "5.13.0",
41
+ "@memberjunction/ai-agent-manager-actions": "5.13.0",
42
+ "@memberjunction/ai-agents": "5.13.0",
43
+ "@memberjunction/ai-core-plus": "5.13.0",
44
+ "@memberjunction/ai-prompts": "5.13.0",
45
+ "@memberjunction/ai-provider-bundle": "5.13.0",
46
+ "@memberjunction/ai-vectors-pinecone": "5.13.0",
47
+ "@memberjunction/aiengine": "5.13.0",
48
+ "@memberjunction/communication-ms-graph": "5.13.0",
49
+ "@memberjunction/communication-sendgrid": "5.13.0",
50
+ "@memberjunction/communication-types": "5.13.0",
51
+ "@memberjunction/component-registry-client-sdk": "5.13.0",
52
+ "@memberjunction/config": "5.13.0",
53
+ "@memberjunction/core": "5.13.0",
54
+ "@memberjunction/core-actions": "5.13.0",
55
+ "@memberjunction/core-entities": "5.13.0",
56
+ "@memberjunction/core-entities-server": "5.13.0",
57
+ "@memberjunction/data-context": "5.13.0",
58
+ "@memberjunction/data-context-server": "5.13.0",
59
+ "@memberjunction/doc-utils": "5.13.0",
60
+ "@memberjunction/api-keys": "5.13.0",
61
+ "@memberjunction/encryption": "5.13.0",
62
+ "@memberjunction/entity-communications-base": "5.13.0",
63
+ "@memberjunction/entity-communications-server": "5.13.0",
64
+ "@memberjunction/external-change-detection": "5.13.0",
65
+ "@memberjunction/generic-database-provider": "5.13.0",
66
+ "@memberjunction/global": "5.13.0",
67
+ "@memberjunction/graphql-dataprovider": "5.13.0",
68
+ "@memberjunction/integration-engine": "5.13.0",
69
+ "@memberjunction/integration-schema-builder": "5.13.0",
70
+ "@memberjunction/interactive-component-types": "5.13.0",
71
+ "@memberjunction/computer-use-engine": "5.13.0",
72
+ "@memberjunction/notifications": "5.13.0",
73
+ "@memberjunction/queue": "5.13.0",
74
+ "@memberjunction/redis-provider": "5.13.0",
75
+ "@memberjunction/scheduling-actions": "5.13.0",
76
+ "@memberjunction/scheduling-base-types": "5.13.0",
77
+ "@memberjunction/scheduling-engine": "5.13.0",
78
+ "@memberjunction/scheduling-engine-base": "5.13.0",
79
+ "@memberjunction/skip-types": "5.13.0",
80
+ "@memberjunction/sqlserver-dataprovider": "5.13.0",
81
+ "@memberjunction/storage": "5.13.0",
82
+ "@memberjunction/templates": "5.13.0",
83
+ "@memberjunction/testing-engine": "5.13.0",
84
+ "@memberjunction/testing-engine-base": "5.13.0",
85
+ "@memberjunction/version-history": "5.13.0",
86
86
  "@types/compression": "^1.8.1",
87
87
  "@types/cors": "^2.8.19",
88
88
  "@types/jsonwebtoken": "9.0.10",
@@ -107,8 +107,8 @@
107
107
  "jwks-rsa": "^3.2.2",
108
108
  "lru-cache": "^11.2.5",
109
109
  "mssql": "^12.2.0",
110
- "@memberjunction/postgresql-dataprovider": "5.12.0",
111
- "@memberjunction/sql-dialect": "5.12.0",
110
+ "@memberjunction/postgresql-dataprovider": "5.13.0",
111
+ "@memberjunction/sql-dialect": "5.13.0",
112
112
  "pg": "^8.13.3",
113
113
  "@types/pg": "^8.11.11",
114
114
  "reflect-metadata": "0.2.2",
@@ -2,7 +2,7 @@
2
2
  * BCSaaS Integration Tests — Phase 4, 5, 6
3
3
  *
4
4
  * Tests the BCSaaS middle-layer plugin architecture against a RUNNING MJAPI instance
5
- * that has BCSaaS loaded via DynamicPackageLoader and a database with BCSaaS entities.
5
+ * that has BCSaaS loaded via @RegisterClass(BaseServerMiddleware) and a database with BCSaaS entities.
6
6
  *
7
7
  * Test data setup (mj_test database):
8
8
  * - Organizations: Acme Corp (11111111-...), Beta Inc (22222222-...), Acme West Division (33333333-...)
package/src/index.ts CHANGED
@@ -126,32 +126,16 @@ export * from './resolvers/CurrentUserContextResolver.js';
126
126
  export { GetReadOnlyDataSource, GetReadWriteDataSource, GetReadWriteProvider, GetReadOnlyProvider } from './util.js';
127
127
 
128
128
  export * from './generated/generated.js';
129
- export * from './hooks.js';
130
129
  export * from './multiTenancy/index.js';
130
+ export * from './middleware/index.js';
131
131
 
132
- import type { ServerExtensibilityOptions, HookWithOptions } from './hooks.js';
133
- import { HookRegistry } from '@memberjunction/core';
134
- import type { HookRegistrationOptions } from '@memberjunction/core';
132
+ import { RegisterDataHook } from '@memberjunction/core';
133
+ import type { RequestHandler, ErrorRequestHandler } from 'express';
135
134
  import type { ApolloServerPlugin } from '@apollo/server';
136
- import { createTenantMiddleware, createTenantPreRunViewHook, createTenantPreSaveHook } from './multiTenancy/index.js';
135
+ import type { GraphQLSchema } from 'graphql';
136
+ import { BaseServerMiddleware } from './middleware/BaseServerMiddleware.js';
137
137
 
138
- /**
139
- * Register a hook that may be a plain function or a `{ hook, Priority, Namespace }` object.
140
- * Dynamic packages (e.g., BCSaaS) return hooks in object form to declare registration metadata.
141
- */
142
- function registerHookEntry<T>(hookName: string, entry: T | HookWithOptions<T>): void {
143
- if (typeof entry === 'function') {
144
- HookRegistry.Register(hookName, entry);
145
- } else if (entry && typeof entry === 'object' && 'hook' in entry) {
146
- const { hook, Priority, Namespace } = entry as HookWithOptions<T>;
147
- const options: HookRegistrationOptions = {};
148
- if (Priority != null) options.Priority = Priority;
149
- if (Namespace != null) options.Namespace = Namespace;
150
- HookRegistry.Register(hookName, hook, options);
151
- }
152
- }
153
-
154
- export type MJServerOptions = ServerExtensibilityOptions & {
138
+ export type MJServerOptions = {
155
139
  onBeforeServe?: () => void | Promise<void>;
156
140
  restApiOptions?: Partial<RESTApiOptions>; // Options for REST API configuration
157
141
  };
@@ -417,6 +401,91 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
417
401
  Object.values(module).filter((value) => typeof value === 'function')
418
402
  ) as BuildSchemaOptions['resolvers'];
419
403
 
404
+ // ─── Discover all server middleware via ClassFactory ─────────────────────
405
+ const allRegistrations = MJGlobal.Instance.ClassFactory.GetAllRegistrations(BaseServerMiddleware);
406
+
407
+ // Deduplicate by key (same key -> highest priority wins).
408
+ // This is the replacement mechanism: if BCSaaS registers with the same
409
+ // key as MJ's built-in tenant filter, ClassFactory's priority system
410
+ // ensures only the higher-priority one is used.
411
+ const uniqueKeys = new Set(
412
+ allRegistrations.map(r => r.Key?.trim().toLowerCase()).filter((k): k is string => k != null)
413
+ );
414
+
415
+ const winnerRegistrations: typeof allRegistrations = [];
416
+ for (const key of uniqueKeys) {
417
+ const winner = MJGlobal.Instance.ClassFactory.GetRegistration(BaseServerMiddleware, key);
418
+ if (winner) winnerRegistrations.push(winner);
419
+ }
420
+
421
+ // Instantiate and filter by Enabled
422
+ const middlewares: BaseServerMiddleware[] = [];
423
+ for (const reg of winnerRegistrations) {
424
+ const MwClass = reg.SubClass as new () => BaseServerMiddleware;
425
+ const mw = new MwClass();
426
+ if (mw.Enabled) {
427
+ middlewares.push(mw);
428
+ }
429
+ }
430
+
431
+ // Initialize all middleware
432
+ for (const mw of middlewares) {
433
+ await mw.Initialize();
434
+ console.log(` [Middleware] ${mw.Label}`);
435
+ }
436
+
437
+ // Collect middleware contributions for each pipeline stage
438
+ const mwPreAuth: RequestHandler[] = [];
439
+ const mwPostAuth: RequestHandler[] = [];
440
+ const mwPostRoute: (RequestHandler | ErrorRequestHandler)[] = [];
441
+ const mwApolloPlugins: ApolloServerPlugin[] = [];
442
+ const mwSchemaTransformers: ((schema: GraphQLSchema) => GraphQLSchema)[] = [];
443
+ const mwResolverPaths: string[] = [];
444
+
445
+ for (const mw of middlewares) {
446
+ mwPreAuth.push(...mw.GetPreAuthMiddleware());
447
+ mwPostAuth.push(...mw.GetPostAuthMiddleware());
448
+ mwPostRoute.push(...mw.GetPostRouteMiddleware());
449
+ mwApolloPlugins.push(...mw.GetApolloPlugins());
450
+ mwSchemaTransformers.push(...mw.GetSchemaTransformers());
451
+ mwResolverPaths.push(...mw.GetResolverPaths());
452
+
453
+ // Express app configuration escape hatch
454
+ if (mw.ConfigureExpressApp) {
455
+ await mw.ConfigureExpressApp(app);
456
+ }
457
+
458
+ // Extract hook methods and register in the global hook store
459
+ // (ProviderBase/BaseEntity will read these via GetDataHooks())
460
+ RegisterDataHook('PreRunView', mw.PreRunView.bind(mw));
461
+ RegisterDataHook('PostRunView', mw.PostRunView.bind(mw));
462
+ RegisterDataHook('PreSave', mw.PreSave.bind(mw));
463
+ }
464
+
465
+ // ─── Resolve middleware-contributed resolver paths and merge into resolvers ───
466
+ let allResolvers = resolvers;
467
+ if (mwResolverPaths.length > 0) {
468
+ const mwGlobs = mwResolverPaths.flatMap((p) => (isWindows ? p.replace(/\\/g, '/') : p));
469
+ const mwResolverFiles = fg.globSync(mwGlobs);
470
+ if (mwResolverFiles.length > 0) {
471
+ const mwModules = await Promise.all(
472
+ mwResolverFiles.map((modulePath) => {
473
+ try {
474
+ return import(isWindows ? `file://${modulePath}` : modulePath);
475
+ } catch (e) {
476
+ console.error(`Error loading middleware resolver at '${modulePath}'`, e);
477
+ throw e;
478
+ }
479
+ })
480
+ );
481
+ const mwResolvers = mwModules.flatMap((module) =>
482
+ Object.values(module).filter((value) => typeof value === 'function')
483
+ );
484
+ allResolvers = [...resolvers, ...mwResolvers] as BuildSchemaOptions['resolvers'];
485
+ console.log(` [Middleware Resolvers] Loaded ${mwResolverFiles.length} resolver file(s) from middleware`);
486
+ }
487
+ }
488
+
420
489
  // Create an explicit PubSub instance so we can reference it outside of resolvers
421
490
  // graphql-subscriptions v3 renamed asyncIterator→asyncIterableIterator, but
422
491
  // type-graphql still calls asyncIterator. Shim for compatibility.
@@ -451,7 +520,7 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
451
520
  let schema = mergeSchemas({
452
521
  schemas: [
453
522
  buildSchemaSync({
454
- resolvers,
523
+ resolvers: allResolvers,
455
524
  validate: false,
456
525
  scalarsMap: [{ type: Date, scalar: GraphQLTimestamp }],
457
526
  emitSchemaFile: websiteRunFromPackage !== 1,
@@ -463,11 +532,9 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
463
532
  schema = requireSystemUserDirective.transformer(schema);
464
533
  schema = publicDirective.transformer(schema);
465
534
 
466
- // Apply user-provided schema transformers (after built-in directive transformers)
467
- if (options?.SchemaTransformers) {
468
- for (const transformer of options.SchemaTransformers) {
469
- schema = transformer(schema);
470
- }
535
+ // Apply middleware-contributed schema transformers (after built-in directive transformers)
536
+ for (const transformer of mwSchemaTransformers) {
537
+ schema = transformer(schema);
471
538
  }
472
539
 
473
540
  const httpServer = createServer(app);
@@ -502,7 +569,7 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
502
569
  const apolloServer = buildApolloServer(
503
570
  { schema },
504
571
  { httpServer, serverCleanup },
505
- options?.ApolloPlugins
572
+ mwApolloPlugins.length > 0 ? mwApolloPlugins : undefined
506
573
  );
507
574
  await apolloServer.start();
508
575
 
@@ -534,16 +601,9 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
534
601
  res.status(200).json({ status: 'ok' });
535
602
  });
536
603
 
537
- // Apply user-provided Express middleware (after compression, before routes)
538
- if (options?.ExpressMiddlewareBefore) {
539
- for (const mw of options.ExpressMiddlewareBefore) {
540
- app.use(mw);
541
- }
542
- }
543
-
544
- // Escape hatch for advanced Express app configuration
545
- if (options?.ConfigureExpressApp) {
546
- await Promise.resolve(options.ConfigureExpressApp(app));
604
+ // Apply middleware-contributed pre-auth handlers (after compression, before routes)
605
+ for (const mw of mwPreAuth) {
606
+ app.use(mw);
547
607
  }
548
608
 
549
609
  // ─── OAuth callback routes (unauthenticated, registered BEFORE auth) ─────
@@ -575,20 +635,12 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
575
635
  // ─── Unified auth middleware (replaces both REST authMiddleware and contextFunction auth) ─────
576
636
  app.use(createUnifiedAuthMiddleware(dataSources));
577
637
 
578
- // ─── Built-in post-auth middleware (multi-tenancy) ─────
579
- // Config-driven multi-tenancy middleware runs after auth so it can read req.userPayload.
580
- if (configInfo.multiTenancy?.enabled) {
581
- const tenantMiddleware = createTenantMiddleware(configInfo.multiTenancy);
582
- app.use(tenantMiddleware);
583
- }
584
-
585
- // ─── Post-auth middleware from plugins ─────
638
+ // ─── Post-auth middleware from BaseServerMiddleware plugins ─────
586
639
  // Middleware here has access to the authenticated user via req.userPayload.
587
- // Use this for tenant context resolution, org membership loading, etc.
588
- if (options?.ExpressMiddlewarePostAuth) {
589
- for (const mw of options.ExpressMiddlewarePostAuth) {
590
- app.use(mw);
591
- }
640
+ // Contributions come from @RegisterClass(BaseServerMiddleware, key) classes
641
+ // (e.g., MJTenantFilterMiddleware for multi-tenancy, BCSaaSMiddleware for org context).
642
+ for (const mw of mwPostAuth) {
643
+ app.use(mw);
592
644
  }
593
645
 
594
646
  // ─── OAuth authenticated routes (auth already handled by unified middleware) ─────
@@ -645,10 +697,8 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
645
697
  );
646
698
 
647
699
  // ─── Post-route middleware (error handlers, catch-alls) ─────
648
- if (options?.ExpressMiddlewareAfter) {
649
- for (const mw of options.ExpressMiddlewareAfter) {
650
- app.use(mw);
651
- }
700
+ for (const mw of mwPostRoute) {
701
+ app.use(mw);
652
702
  }
653
703
 
654
704
  // Initialize and start scheduled jobs service if enabled
@@ -664,45 +714,9 @@ export const serve = async (resolverPaths: Array<string>, app: Application = cre
664
714
  }
665
715
  }
666
716
 
667
- // Register provider-level hooks with the global HookRegistry.
668
- // Each entry can be a plain function or a { hook, Priority, Namespace } object.
669
- if (options?.PreRunViewHooks) {
670
- for (const entry of options.PreRunViewHooks) {
671
- registerHookEntry('PreRunView', entry);
672
- }
673
- }
674
- if (options?.PostRunViewHooks) {
675
- for (const entry of options.PostRunViewHooks) {
676
- registerHookEntry('PostRunView', entry);
677
- }
678
- }
679
- if (options?.PreSaveHooks) {
680
- for (const entry of options.PreSaveHooks) {
681
- registerHookEntry('PreSave', entry);
682
- }
683
- }
684
-
685
- // Auto-register multi-tenancy hooks when enabled in config
686
- // (The tenant Express middleware was already registered above in the post-auth slot)
687
- if (configInfo.multiTenancy?.enabled) {
688
- console.log('[MultiTenancy] Enabled — registering tenant isolation hooks');
689
- const tenantConfig = configInfo.multiTenancy;
690
-
691
- // Register tenant PreRunView hook (injects WHERE clauses)
692
- // Priority 50 + namespace allows middle layers to replace with their own implementation
693
- HookRegistry.Register('PreRunView', createTenantPreRunViewHook(tenantConfig), {
694
- Priority: 50,
695
- Namespace: 'mj:tenantFilter',
696
- });
697
-
698
- // Register tenant PreSave hook (validates tenant column on writes)
699
- HookRegistry.Register('PreSave', createTenantPreSaveHook(tenantConfig), {
700
- Priority: 50,
701
- Namespace: 'mj:tenantSave',
702
- });
703
-
704
- console.log(`[MultiTenancy] Context source: ${tenantConfig.contextSource}, scoping: ${tenantConfig.scopingStrategy}, write protection: ${tenantConfig.writeProtection}`);
705
- }
717
+ // Data hooks are now registered via BaseServerMiddleware classes above
718
+ // (e.g., MJTenantFilterMiddleware registers PreRunView and PreSave hooks).
719
+ // No config-bag hook registration needed.
706
720
 
707
721
  if (options?.onBeforeServe) {
708
722
  await Promise.resolve(options.onBeforeServe());
@@ -0,0 +1,141 @@
1
+ /**
2
+ * Base class for server middleware that intercepts the MJServer request pipeline.
3
+ *
4
+ * Subclasses register via @RegisterClass(BaseServerMiddleware, key) and are
5
+ * discovered by serve() via ClassFactory.GetAllRegistrations().
6
+ *
7
+ * Key-based deduplication: If two middleware classes register with the same key,
8
+ * ClassFactory returns the highest-priority one -- the other is replaced.
9
+ * This is how BCSaaS replaces MJ's built-in tenant filtering.
10
+ *
11
+ * For adding new routes/endpoints (webhooks, bot endpoints), see
12
+ * ServerExtensionsCore and BaseServerExtension (PR #2037).
13
+ * BaseServerMiddleware is for intercepting the existing pipeline, not adding to it.
14
+ */
15
+
16
+ import type { RequestHandler, ErrorRequestHandler, Application } from 'express';
17
+ import type { ApolloServerPlugin } from '@apollo/server';
18
+ import type { GraphQLSchema } from 'graphql';
19
+ import type { RunViewParams, UserInfo, BaseEntity, RunViewResult } from '@memberjunction/core';
20
+
21
+ export abstract class BaseServerMiddleware {
22
+ // --- Identity ---
23
+
24
+ /**
25
+ * Human-readable label for logging (e.g., 'mj:tenantFilter', 'bcsaas').
26
+ */
27
+ abstract get Label(): string;
28
+
29
+ /**
30
+ * Whether this middleware is active. Override to check config, env vars, etc.
31
+ * Disabled middleware classes are never instantiated or activated by serve().
32
+ * Default: true.
33
+ */
34
+ get Enabled(): boolean { return true; }
35
+
36
+ // --- Lifecycle ---
37
+
38
+ /**
39
+ * Optional async initialization (read config, warm caches, etc.).
40
+ * Called by serve() after instantiation, before middleware/hooks are extracted.
41
+ */
42
+ async Initialize(): Promise<void> { /* no-op by default */ }
43
+
44
+ // --- Express Middleware ---
45
+ // Each method corresponds to a named slot in the request pipeline.
46
+ // Override to contribute middleware at that stage.
47
+ // Default implementations return empty arrays (no-op).
48
+
49
+ /**
50
+ * Express middleware applied BEFORE authentication.
51
+ * Runs after compression but before OAuth/REST/GraphQL routes.
52
+ * Use for: rate limiting, request logging, custom headers.
53
+ */
54
+ GetPreAuthMiddleware(): RequestHandler[] { return []; }
55
+
56
+ /**
57
+ * Express middleware that runs AFTER authentication has resolved UserInfo.
58
+ * The authenticated user payload is available at req.userPayload.
59
+ * Use for: tenant context resolution, org membership loading.
60
+ */
61
+ GetPostAuthMiddleware(): RequestHandler[] { return []; }
62
+
63
+ /**
64
+ * Express middleware/error handlers applied AFTER all routes.
65
+ * Use for: catch-all error handlers, response logging.
66
+ */
67
+ GetPostRouteMiddleware(): (RequestHandler | ErrorRequestHandler)[] { return []; }
68
+
69
+ // --- Express App Configuration ---
70
+
71
+ /**
72
+ * Optional escape hatch for advanced Express app configuration.
73
+ * Called once during server setup with the Express app instance.
74
+ * Use for: trust proxy settings, custom CORS, body parser config.
75
+ * Return undefined to skip (default).
76
+ */
77
+ ConfigureExpressApp?(app: Application): void | Promise<void>;
78
+
79
+ // --- Apollo / GraphQL Extensions ---
80
+
81
+ /**
82
+ * Additional Apollo Server plugins merged with built-in plugins.
83
+ * Use for: query tracing, caching, custom error formatting.
84
+ */
85
+ GetApolloPlugins(): ApolloServerPlugin[] { return []; }
86
+
87
+ /**
88
+ * Schema transformers applied after built-in directive transformers.
89
+ * Use for: custom directives, schema stitching, field-level auth.
90
+ */
91
+ GetSchemaTransformers(): ((schema: GraphQLSchema) => GraphQLSchema)[] { return []; }
92
+
93
+ // --- Resolver Discovery ---
94
+
95
+ /**
96
+ * Additional TypeGraphQL resolver glob paths to include in the Apollo schema.
97
+ * serve() collects these from all active middleware and adds them to the
98
+ * resolvers array passed to buildSchema().
99
+ *
100
+ * Return absolute glob paths (use `path.join(__dirname, ...)` or equivalent).
101
+ *
102
+ * Use for: Open App resolvers, domain-specific GraphQL queries/mutations
103
+ * that live outside MJServer's built-in resolver set.
104
+ *
105
+ * @example
106
+ * ```typescript
107
+ * GetResolverPaths(): string[] {
108
+ * return [path.join(__dirname, 'resolvers', '*Resolver.{js,ts}')];
109
+ * }
110
+ * ```
111
+ */
112
+ GetResolverPaths(): string[] { return []; }
113
+
114
+ // --- Data Hooks ---
115
+ // Override any of these to intercept data operations.
116
+ // Default implementations are pass-through (no-op).
117
+
118
+ /**
119
+ * Hook that runs before a RunView operation. Can modify the RunViewParams
120
+ * (e.g., injecting tenant filters) before execution.
121
+ */
122
+ PreRunView(params: RunViewParams, contextUser: UserInfo | undefined): RunViewParams | Promise<RunViewParams> {
123
+ return params;
124
+ }
125
+
126
+ /**
127
+ * Hook that runs after a RunView operation completes. Can modify the result
128
+ * (e.g., filtering or augmenting data) before it is returned to the caller.
129
+ */
130
+ PostRunView(params: RunViewParams, results: RunViewResult, contextUser: UserInfo | undefined): RunViewResult | Promise<RunViewResult> {
131
+ return results;
132
+ }
133
+
134
+ /**
135
+ * Hook that runs before a Save operation on a BaseEntity.
136
+ * Return true to allow, false to reject silently, or a string to reject with that error message.
137
+ */
138
+ PreSave(entity: BaseEntity, contextUser: UserInfo | undefined): boolean | string | Promise<boolean | string> {
139
+ return true;
140
+ }
141
+ }