@lssm/example.personalization 0.0.0-canary-20251213172311
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.
- package/.turbo/turbo-build.log +18 -0
- package/CHANGELOG.md +12 -0
- package/dist/behavior-tracking.js +1 -0
- package/dist/docs/index.js +1 -0
- package/dist/docs/personalization.docblock.js +9 -0
- package/dist/example.js +1 -0
- package/dist/index.js +1 -0
- package/dist/overlay-customization.js +1 -0
- package/dist/workflow-extension.js +1 -0
- package/package.json +56 -0
- package/src/behavior-tracking.ts +44 -0
- package/src/docs/index.ts +3 -0
- package/src/docs/personalization.docblock.ts +28 -0
- package/src/example.ts +27 -0
- package/src/index.ts +7 -0
- package/src/overlay-customization.ts +50 -0
- package/src/workflow-extension.ts +62 -0
- package/tsconfig.json +11 -0
- package/tsconfig.tsbuildinfo +1 -0
- package/tsdown.config.js +6 -0
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
$ bun build:bundle && bun build:types
|
|
2
|
+
$ tsdown
|
|
3
|
+
[34mℹ[39m tsdown [2mv0.17.0[22m powered by rolldown [2mv1.0.0-beta.53[22m
|
|
4
|
+
[34mℹ[39m config file: [4m/home/runner/work/contractspec/contractspec/packages/examples/personalization/tsdown.config.js[24m
|
|
5
|
+
[34mℹ[39m entry: [34msrc/behavior-tracking.ts, src/example.ts, src/index.ts, src/overlay-customization.ts, src/workflow-extension.ts, src/docs/index.ts, src/docs/personalization.docblock.ts[39m
|
|
6
|
+
[34mℹ[39m target: [34mesnext[39m
|
|
7
|
+
[34mℹ[39m tsconfig: [34mtsconfig.json[39m
|
|
8
|
+
[34mℹ[39m Build start
|
|
9
|
+
[34mℹ[39m [2mdist/[22m[1mdocs/personalization.docblock.js[22m [2m1.26 kB[22m [2m│ gzip: 0.63 kB[22m
|
|
10
|
+
[34mℹ[39m [2mdist/[22m[1moverlay-customization.js[22m [2m1.09 kB[22m [2m│ gzip: 0.56 kB[22m
|
|
11
|
+
[34mℹ[39m [2mdist/[22m[1mworkflow-extension.js[22m [2m1.09 kB[22m [2m│ gzip: 0.56 kB[22m
|
|
12
|
+
[34mℹ[39m [2mdist/[22m[1mbehavior-tracking.js[22m [2m1.01 kB[22m [2m│ gzip: 0.52 kB[22m
|
|
13
|
+
[34mℹ[39m [2mdist/[22m[1mexample.js[22m [2m0.58 kB[22m [2m│ gzip: 0.34 kB[22m
|
|
14
|
+
[34mℹ[39m [2mdist/[22m[1mindex.js[22m [2m0.45 kB[22m [2m│ gzip: 0.21 kB[22m
|
|
15
|
+
[34mℹ[39m [2mdist/[22m[1mdocs/index.js[22m [2m0.04 kB[22m [2m│ gzip: 0.06 kB[22m
|
|
16
|
+
[34mℹ[39m 7 files, total: 5.51 kB
|
|
17
|
+
[32m✔[39m Build complete in [32m37ms[39m
|
|
18
|
+
$ tsc --noEmit
|
package/CHANGELOG.md
ADDED
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
# @lssm/example.personalization
|
|
2
|
+
|
|
3
|
+
## 0.0.0-canary-20251213172311
|
|
4
|
+
|
|
5
|
+
### Patch Changes
|
|
6
|
+
|
|
7
|
+
- Updated dependencies [3086383]
|
|
8
|
+
- @lssm/lib.workflow-composer@0.0.0-canary-20251213172311
|
|
9
|
+
- @lssm/lib.personalization@0.0.0-canary-20251213172311
|
|
10
|
+
- @lssm/lib.overlay-engine@0.0.0-canary-20251213172311
|
|
11
|
+
- @lssm/lib.contracts@0.0.0-canary-20251213172311
|
|
12
|
+
- @lssm/lib.logger@0.0.0-canary-20251213172311
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{createBehaviorTracker as e}from"@lssm/lib.personalization/tracker";import{InMemoryBehaviorStore as t}from"@lssm/lib.personalization/store";import{BehaviorAnalyzer as n}from"@lssm/lib.personalization/analyzer";import{LogLevel as r,Logger as i}from"@lssm/lib.logger";const a=new i({level:process.env.NODE_ENV===`production`?r.INFO:r.DEBUG,environment:process.env.NODE_ENV||`development`,enableColors:process.env.NODE_ENV!==`production`});async function o(){let r=new t,i=e({store:r,context:{tenantId:`acme`,userId:`user-123`}});i.trackFieldAccess({operation:`billing.createOrder`,field:`internalNotes`}),i.trackFieldAccess({operation:`billing.createOrder`,field:`customerReference`}),i.trackFeatureUsage({feature:`workflow-editor`,action:`opened`}),i.trackWorkflowStep({workflow:`invoice-approval`,step:`review`,status:`entered`}),await i.flush();let o=await new n(r).analyze({tenantId:`acme`,userId:`user-123`});a.info(`Behavior insights computed`,{insights:o})}export{o as runBehaviorTrackingExample};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import"./personalization.docblock.js";
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import{registerDocBlocks as e}from"@lssm/lib.contracts/docs";e([{id:`docs.examples.personalization`,title:`Personalization Patterns (example)`,summary:`Behavior tracking, overlay-driven UI tweaks, and tenant workflow extension patterns.`,kind:`reference`,visibility:`public`,route:`/docs/examples/personalization`,tags:[`personalization`,`overlays`,`workflows`,`example`],body:`## Includes
|
|
2
|
+
- Behavior tracking + insight analysis.
|
|
3
|
+
- Overlay customization (hide fields, rename labels).
|
|
4
|
+
- Workflow extension (inject tenant-specific steps).
|
|
5
|
+
|
|
6
|
+
## Guardrails
|
|
7
|
+
- Keep tracking events structured and non-PII.
|
|
8
|
+
- Keep overlays signed and scoped.
|
|
9
|
+
- Keep workflow composition deterministic.`},{id:`docs.examples.personalization.usage`,title:`Personalization — Usage`,summary:`How to run the small examples and swap adapters.`,kind:`usage`,visibility:`public`,route:`/docs/examples/personalization/usage`,tags:[`personalization`,`usage`],body:"## Usage\n- Call `runBehaviorTrackingExample()` for insights.\n- Call `runOverlayCustomizationExample()` to apply overlays.\n- Call `composeTenantWorkflowExample()` and `logTenantWorkflowSteps()` to inspect steps.\n\n## Notes\n- Replace in-memory stores with app-layer storage.\n- Keep PII out of logs and markdown outputs."}]);
|
package/dist/example.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
var e={id:`personalization`,title:`Personalization Patterns`,summary:`Small examples for behavior tracking, overlay-based UI customization, and tenant workflow extension.`,tags:[`personalization`,`overlays`,`behavior`,`workflows`],kind:`library`,visibility:`public`,docs:{rootDocId:`docs.examples.personalization`,usageDocId:`docs.examples.personalization.usage`},entrypoints:{packageName:`@lssm/example.personalization`,docs:`./docs`},surfaces:{templates:!0,sandbox:{enabled:!0,modes:[`markdown`,`specs`]},studio:{enabled:!0,installable:!0},mcp:{enabled:!0}}};export{e as default};
|
package/dist/index.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{runBehaviorTrackingExample as e}from"./behavior-tracking.js";import t from"./example.js";import{runOverlayCustomizationExample as n}from"./overlay-customization.js";import{composeTenantWorkflowExample as r,logTenantWorkflowSteps as i}from"./workflow-extension.js";import"./docs/index.js";export{r as composeTenantWorkflowExample,t as example,i as logTenantWorkflowSteps,e as runBehaviorTrackingExample,n as runOverlayCustomizationExample};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{LogLevel as e,Logger as t}from"@lssm/lib.logger";import{defineOverlay as n}from"@lssm/lib.overlay-engine/spec";import{signOverlay as r}from"@lssm/lib.overlay-engine/signer";import{OverlayEngine as i,OverlayRegistry as a}from"@lssm/lib.overlay-engine";const o=new t({level:process.env.NODE_ENV===`production`?e.INFO:e.DEBUG,environment:process.env.NODE_ENV||`development`,enableColors:process.env.NODE_ENV!==`production`});async function s(){let e=new a({allowUnsigned:!0}),t=new i({registry:e}),s=await r(n({overlayId:`demo-overlay`,version:`1.0.0`,appliesTo:{capability:`billing.createOrder`,tenantId:`demo`},modifications:[{type:`hideField`,field:`internalNotes`},{type:`renameLabel`,field:`customerReference`,newLabel:`PO Number`}]}),process.env.PRIVATE_KEY_PEM??``);e.register(s);let c=t.apply({target:{fields:[{key:`customerReference`,label:`Customer Reference`,visible:!0},{key:`internalNotes`,label:`Internal Notes`,visible:!0}]},capability:`billing.createOrder`,tenantId:`demo`});o.info(`Overlay applied`,{fields:c.target.fields})}export{s as runOverlayCustomizationExample};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{LogLevel as e,Logger as t}from"@lssm/lib.logger";import{WorkflowComposer as n}from"@lssm/lib.workflow-composer";const r=new t({level:process.env.NODE_ENV===`production`?e.INFO:e.DEBUG,environment:process.env.NODE_ENV||`development`,enableColors:process.env.NODE_ENV!==`production`}),i={meta:{name:`billing.invoiceApproval`,version:1,title:`Invoice Approval`,owners:[],tags:[],description:``,domain:`billing`,stability:`stable`},definition:{steps:[{id:`validate-invoice`,type:`automation`,label:`Validate Invoice`},{id:`final-approval`,type:`human`,label:`Final Approval`}],transitions:[{from:`validate-invoice`,to:`final-approval`}]}};function a(){let e=new n;return e.register({workflow:`billing.invoiceApproval`,tenantId:`acme`,customSteps:[{after:`validate-invoice`,inject:{id:`acme-legal`,type:`human`,label:`ACME Legal Review`},transitionTo:`final-approval`}]}),e.compose({base:i,tenantId:`acme`})}function o(e){r.info(`Tenant workflow composed`,{workflow:e.meta.name,steps:e.definition.steps.map(e=>e.id)})}export{a as composeTenantWorkflowExample,o as logTenantWorkflowSteps};
|
package/package.json
ADDED
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@lssm/example.personalization",
|
|
3
|
+
"version": "0.0.0-canary-20251213172311",
|
|
4
|
+
"description": "Personalization examples: behavior tracking, overlay customization, workflow extension.",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"main": "./dist/index.js",
|
|
7
|
+
"types": "./dist/index.d.ts",
|
|
8
|
+
"exports": {
|
|
9
|
+
".": "./src/index.ts",
|
|
10
|
+
"./behavior-tracking": "./src/behavior-tracking.ts",
|
|
11
|
+
"./docs": "./src/docs/index.ts",
|
|
12
|
+
"./docs/personalization.docblock": "./src/docs/personalization.docblock.ts",
|
|
13
|
+
"./example": "./src/example.ts",
|
|
14
|
+
"./overlay-customization": "./src/overlay-customization.ts",
|
|
15
|
+
"./workflow-extension": "./src/workflow-extension.ts",
|
|
16
|
+
"./*": "./*"
|
|
17
|
+
},
|
|
18
|
+
"scripts": {
|
|
19
|
+
"build": "bun build:bundle && bun build:types",
|
|
20
|
+
"build:bundle": "tsdown",
|
|
21
|
+
"build:types": "tsc --noEmit",
|
|
22
|
+
"dev": "bun build:bundle --watch",
|
|
23
|
+
"clean": "rimraf dist .turbo",
|
|
24
|
+
"lint": "bun lint:fix",
|
|
25
|
+
"lint:fix": "eslint src --fix",
|
|
26
|
+
"lint:check": "eslint src",
|
|
27
|
+
"test": "bun test"
|
|
28
|
+
},
|
|
29
|
+
"dependencies": {
|
|
30
|
+
"@lssm/lib.personalization": "workspace:*",
|
|
31
|
+
"@lssm/lib.overlay-engine": "workspace:*",
|
|
32
|
+
"@lssm/lib.workflow-composer": "workspace:*",
|
|
33
|
+
"@lssm/lib.contracts": "workspace:*",
|
|
34
|
+
"@lssm/lib.logger": "workspace:*"
|
|
35
|
+
},
|
|
36
|
+
"devDependencies": {
|
|
37
|
+
"@lssm/tool.tsdown": "workspace:*",
|
|
38
|
+
"@lssm/tool.typescript": "workspace:*",
|
|
39
|
+
"tsdown": "^0.17.0",
|
|
40
|
+
"typescript": "^5.9.3"
|
|
41
|
+
},
|
|
42
|
+
"publishConfig": {
|
|
43
|
+
"access": "public",
|
|
44
|
+
"exports": {
|
|
45
|
+
".": "./dist/index.js",
|
|
46
|
+
"./behavior-tracking": "./dist/behavior-tracking.js",
|
|
47
|
+
"./docs": "./dist/docs/index.js",
|
|
48
|
+
"./docs/personalization.docblock": "./dist/docs/personalization.docblock.js",
|
|
49
|
+
"./example": "./dist/example.js",
|
|
50
|
+
"./overlay-customization": "./dist/overlay-customization.js",
|
|
51
|
+
"./workflow-extension": "./dist/workflow-extension.js",
|
|
52
|
+
"./*": "./*"
|
|
53
|
+
}
|
|
54
|
+
},
|
|
55
|
+
"module": "./dist/index.js"
|
|
56
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { createBehaviorTracker } from '@lssm/lib.personalization/tracker';
|
|
2
|
+
import { InMemoryBehaviorStore } from '@lssm/lib.personalization/store';
|
|
3
|
+
import { BehaviorAnalyzer } from '@lssm/lib.personalization/analyzer';
|
|
4
|
+
import { Logger, LogLevel } from '@lssm/lib.logger';
|
|
5
|
+
|
|
6
|
+
const logger = new Logger({
|
|
7
|
+
level: process.env.NODE_ENV === 'production' ? LogLevel.INFO : LogLevel.DEBUG,
|
|
8
|
+
environment: process.env.NODE_ENV || 'development',
|
|
9
|
+
enableColors: process.env.NODE_ENV !== 'production',
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
export async function runBehaviorTrackingExample(): Promise<void> {
|
|
13
|
+
const store = new InMemoryBehaviorStore();
|
|
14
|
+
const tracker = createBehaviorTracker({
|
|
15
|
+
store,
|
|
16
|
+
context: {
|
|
17
|
+
tenantId: 'acme',
|
|
18
|
+
userId: 'user-123',
|
|
19
|
+
},
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
tracker.trackFieldAccess({
|
|
23
|
+
operation: 'billing.createOrder',
|
|
24
|
+
field: 'internalNotes',
|
|
25
|
+
});
|
|
26
|
+
tracker.trackFieldAccess({
|
|
27
|
+
operation: 'billing.createOrder',
|
|
28
|
+
field: 'customerReference',
|
|
29
|
+
});
|
|
30
|
+
tracker.trackFeatureUsage({ feature: 'workflow-editor', action: 'opened' });
|
|
31
|
+
tracker.trackWorkflowStep({
|
|
32
|
+
workflow: 'invoice-approval',
|
|
33
|
+
step: 'review',
|
|
34
|
+
status: 'entered',
|
|
35
|
+
});
|
|
36
|
+
await tracker.flush();
|
|
37
|
+
|
|
38
|
+
const analyzer = new BehaviorAnalyzer(store);
|
|
39
|
+
const insights = await analyzer.analyze({ tenantId: 'acme', userId: 'user-123' });
|
|
40
|
+
|
|
41
|
+
logger.info('Behavior insights computed', { insights });
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import type { DocBlock } from '@lssm/lib.contracts/docs';
|
|
2
|
+
import { registerDocBlocks } from '@lssm/lib.contracts/docs';
|
|
3
|
+
|
|
4
|
+
const blocks: DocBlock[] = [
|
|
5
|
+
{
|
|
6
|
+
id: 'docs.examples.personalization',
|
|
7
|
+
title: 'Personalization Patterns (example)',
|
|
8
|
+
summary:
|
|
9
|
+
'Behavior tracking, overlay-driven UI tweaks, and tenant workflow extension patterns.',
|
|
10
|
+
kind: 'reference',
|
|
11
|
+
visibility: 'public',
|
|
12
|
+
route: '/docs/examples/personalization',
|
|
13
|
+
tags: ['personalization', 'overlays', 'workflows', 'example'],
|
|
14
|
+
body: `## Includes\n- Behavior tracking + insight analysis.\n- Overlay customization (hide fields, rename labels).\n- Workflow extension (inject tenant-specific steps).\n\n## Guardrails\n- Keep tracking events structured and non-PII.\n- Keep overlays signed and scoped.\n- Keep workflow composition deterministic.`,
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
id: 'docs.examples.personalization.usage',
|
|
18
|
+
title: 'Personalization — Usage',
|
|
19
|
+
summary: 'How to run the small examples and swap adapters.',
|
|
20
|
+
kind: 'usage',
|
|
21
|
+
visibility: 'public',
|
|
22
|
+
route: '/docs/examples/personalization/usage',
|
|
23
|
+
tags: ['personalization', 'usage'],
|
|
24
|
+
body: `## Usage\n- Call \`runBehaviorTrackingExample()\` for insights.\n- Call \`runOverlayCustomizationExample()\` to apply overlays.\n- Call \`composeTenantWorkflowExample()\` and \`logTenantWorkflowSteps()\` to inspect steps.\n\n## Notes\n- Replace in-memory stores with app-layer storage.\n- Keep PII out of logs and markdown outputs.`,
|
|
25
|
+
},
|
|
26
|
+
];
|
|
27
|
+
|
|
28
|
+
registerDocBlocks(blocks);
|
package/src/example.ts
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
const example = {
|
|
2
|
+
id: 'personalization',
|
|
3
|
+
title: 'Personalization Patterns',
|
|
4
|
+
summary:
|
|
5
|
+
'Small examples for behavior tracking, overlay-based UI customization, and tenant workflow extension.',
|
|
6
|
+
tags: ['personalization', 'overlays', 'behavior', 'workflows'],
|
|
7
|
+
kind: 'library',
|
|
8
|
+
visibility: 'public',
|
|
9
|
+
docs: {
|
|
10
|
+
rootDocId: 'docs.examples.personalization',
|
|
11
|
+
usageDocId: 'docs.examples.personalization.usage',
|
|
12
|
+
},
|
|
13
|
+
entrypoints: {
|
|
14
|
+
packageName: '@lssm/example.personalization',
|
|
15
|
+
docs: './docs',
|
|
16
|
+
},
|
|
17
|
+
surfaces: {
|
|
18
|
+
templates: true,
|
|
19
|
+
sandbox: { enabled: true, modes: ['markdown', 'specs'] },
|
|
20
|
+
studio: { enabled: true, installable: true },
|
|
21
|
+
mcp: { enabled: true },
|
|
22
|
+
},
|
|
23
|
+
} as const;
|
|
24
|
+
|
|
25
|
+
export default example;
|
|
26
|
+
|
|
27
|
+
|
package/src/index.ts
ADDED
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { defineOverlay } from '@lssm/lib.overlay-engine/spec';
|
|
2
|
+
import { signOverlay } from '@lssm/lib.overlay-engine/signer';
|
|
3
|
+
import { OverlayEngine, OverlayRegistry } from '@lssm/lib.overlay-engine';
|
|
4
|
+
import { Logger, LogLevel } from '@lssm/lib.logger';
|
|
5
|
+
|
|
6
|
+
const logger = new Logger({
|
|
7
|
+
level: process.env.NODE_ENV === 'production' ? LogLevel.INFO : LogLevel.DEBUG,
|
|
8
|
+
environment: process.env.NODE_ENV || 'development',
|
|
9
|
+
enableColors: process.env.NODE_ENV !== 'production',
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
export async function runOverlayCustomizationExample(): Promise<void> {
|
|
13
|
+
const registry = new OverlayRegistry({ allowUnsigned: true });
|
|
14
|
+
const engine = new OverlayEngine({ registry });
|
|
15
|
+
|
|
16
|
+
const overlay = defineOverlay({
|
|
17
|
+
overlayId: 'demo-overlay',
|
|
18
|
+
version: '1.0.0',
|
|
19
|
+
appliesTo: {
|
|
20
|
+
capability: 'billing.createOrder',
|
|
21
|
+
tenantId: 'demo',
|
|
22
|
+
},
|
|
23
|
+
modifications: [
|
|
24
|
+
{ type: 'hideField', field: 'internalNotes' },
|
|
25
|
+
{
|
|
26
|
+
type: 'renameLabel',
|
|
27
|
+
field: 'customerReference',
|
|
28
|
+
newLabel: 'PO Number',
|
|
29
|
+
},
|
|
30
|
+
],
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
const signed = await signOverlay(overlay, process.env.PRIVATE_KEY_PEM ?? '');
|
|
34
|
+
registry.register(signed);
|
|
35
|
+
|
|
36
|
+
const result = engine.apply({
|
|
37
|
+
target: {
|
|
38
|
+
fields: [
|
|
39
|
+
{ key: 'customerReference', label: 'Customer Reference', visible: true },
|
|
40
|
+
{ key: 'internalNotes', label: 'Internal Notes', visible: true },
|
|
41
|
+
],
|
|
42
|
+
},
|
|
43
|
+
capability: 'billing.createOrder',
|
|
44
|
+
tenantId: 'demo',
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
logger.info('Overlay applied', { fields: result.target.fields });
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import type { WorkflowSpec } from '@lssm/lib.contracts/workflow/spec';
|
|
2
|
+
import { WorkflowComposer } from '@lssm/lib.workflow-composer';
|
|
3
|
+
import { Logger, LogLevel } from '@lssm/lib.logger';
|
|
4
|
+
|
|
5
|
+
const logger = new Logger({
|
|
6
|
+
level: process.env.NODE_ENV === 'production' ? LogLevel.INFO : LogLevel.DEBUG,
|
|
7
|
+
environment: process.env.NODE_ENV || 'development',
|
|
8
|
+
enableColors: process.env.NODE_ENV !== 'production',
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
const BaseWorkflow: WorkflowSpec = {
|
|
12
|
+
meta: {
|
|
13
|
+
name: 'billing.invoiceApproval',
|
|
14
|
+
version: 1,
|
|
15
|
+
title: 'Invoice Approval',
|
|
16
|
+
owners: [],
|
|
17
|
+
tags: [],
|
|
18
|
+
description: '',
|
|
19
|
+
domain: 'billing',
|
|
20
|
+
stability: 'stable',
|
|
21
|
+
},
|
|
22
|
+
definition: {
|
|
23
|
+
steps: [
|
|
24
|
+
{ id: 'validate-invoice', type: 'automation', label: 'Validate Invoice' },
|
|
25
|
+
{ id: 'final-approval', type: 'human', label: 'Final Approval' },
|
|
26
|
+
],
|
|
27
|
+
transitions: [{ from: 'validate-invoice', to: 'final-approval' }],
|
|
28
|
+
},
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
export function composeTenantWorkflowExample(): WorkflowSpec {
|
|
32
|
+
const composer = new WorkflowComposer();
|
|
33
|
+
composer.register({
|
|
34
|
+
workflow: 'billing.invoiceApproval',
|
|
35
|
+
tenantId: 'acme',
|
|
36
|
+
customSteps: [
|
|
37
|
+
{
|
|
38
|
+
after: 'validate-invoice',
|
|
39
|
+
inject: {
|
|
40
|
+
id: 'acme-legal',
|
|
41
|
+
type: 'human',
|
|
42
|
+
label: 'ACME Legal Review',
|
|
43
|
+
},
|
|
44
|
+
transitionTo: 'final-approval',
|
|
45
|
+
},
|
|
46
|
+
],
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
return composer.compose({
|
|
50
|
+
base: BaseWorkflow,
|
|
51
|
+
tenantId: 'acme',
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export function logTenantWorkflowSteps(workflow: WorkflowSpec): void {
|
|
56
|
+
logger.info('Tenant workflow composed', {
|
|
57
|
+
workflow: workflow.meta.name,
|
|
58
|
+
steps: workflow.definition.steps.map((step) => step.id),
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
|