@lssm/example.openbanking-powens 0.0.0-canary-20251217060834 → 0.0.0-canary-20251217072406

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.
Files changed (43) hide show
  1. package/.turbo/turbo-build$colon$bundle.log +57 -15
  2. package/.turbo/turbo-build.log +56 -14
  3. package/CHANGELOG.md +4 -3
  4. package/dist/docs/index.js +1 -1
  5. package/dist/docs/openbanking-powens.docblock.js +28 -12
  6. package/dist/example.js +39 -1
  7. package/dist/handlers/oauth-callback.js +63 -1
  8. package/dist/handlers/webhook-handler.js +87 -1
  9. package/dist/index.js +6 -1
  10. package/dist/integrations/providers-impls/dist/impls/powens-client.js +171 -0
  11. package/dist/integrations/providers-impls/dist/impls/powens-openbanking.js +218 -0
  12. package/dist/libs/contracts/dist/docs/PUBLISHING.docblock.js +16 -0
  13. package/dist/libs/contracts/dist/docs/accessibility_wcag_compliance_specs.docblock.js +16 -0
  14. package/dist/libs/contracts/dist/docs/index.js +29 -0
  15. package/dist/libs/contracts/dist/docs/presentations.js +71 -0
  16. package/dist/libs/contracts/dist/docs/registry.js +44 -0
  17. package/dist/libs/contracts/dist/docs/tech/PHASE_1_QUICKSTART.docblock.js +16 -0
  18. package/dist/libs/contracts/dist/docs/tech/PHASE_2_AI_NATIVE_OPERATIONS.docblock.js +16 -0
  19. package/dist/libs/contracts/dist/docs/tech/PHASE_3_AUTO_EVOLUTION.docblock.js +16 -0
  20. package/dist/libs/contracts/dist/docs/tech/PHASE_4_PERSONALIZATION_ENGINE.docblock.js +16 -0
  21. package/dist/libs/contracts/dist/docs/tech/PHASE_5_ZERO_TOUCH_OPERATIONS.docblock.js +16 -0
  22. package/dist/libs/contracts/dist/docs/tech/auth/better-auth-nextjs.docblock.js +80 -0
  23. package/dist/libs/contracts/dist/docs/tech/contracts/openapi-export.docblock.js +57 -0
  24. package/dist/libs/contracts/dist/docs/tech/lifecycle-stage-system.docblock.js +16 -0
  25. package/dist/libs/contracts/dist/docs/tech/llm/llm-integration.docblock.js +357 -0
  26. package/dist/libs/contracts/dist/docs/tech/mcp-endpoints.docblock.js +37 -0
  27. package/dist/libs/contracts/dist/docs/tech/presentation-runtime.docblock.js +16 -0
  28. package/dist/libs/contracts/dist/docs/tech/schema/README.docblock.js +20 -0
  29. package/dist/libs/contracts/dist/docs/tech/studio/learning-events.docblock.js +48 -0
  30. package/dist/libs/contracts/dist/docs/tech/studio/learning-journeys.docblock.js +79 -0
  31. package/dist/libs/contracts/dist/docs/tech/studio/platform-admin-panel.docblock.js +84 -0
  32. package/dist/libs/contracts/dist/docs/tech/studio/project-access-teams.docblock.js +45 -0
  33. package/dist/libs/contracts/dist/docs/tech/studio/project-routing.docblock.js +67 -0
  34. package/dist/libs/contracts/dist/docs/tech/studio/sandbox-unlogged.docblock.js +40 -0
  35. package/dist/libs/contracts/dist/docs/tech/studio/team-invitations.docblock.js +69 -0
  36. package/dist/libs/contracts/dist/docs/tech/studio/workspace-ops.docblock.js +47 -0
  37. package/dist/libs/contracts/dist/docs/tech/studio/workspaces.docblock.js +62 -0
  38. package/dist/libs/contracts/dist/docs/tech/telemetry-ingest.docblock.js +155 -0
  39. package/dist/libs/contracts/dist/docs/tech/templates/runtime.docblock.js +20 -0
  40. package/dist/libs/contracts/dist/docs/tech/vscode-extension.docblock.js +101 -0
  41. package/dist/libs/contracts/dist/docs/tech/workflows/overview.docblock.js +20 -0
  42. package/package.json +6 -5
  43. package/tsconfig.tsbuildinfo +1 -1
@@ -5,20 +5,52 @@ $ tsdown
5
5
  ℹ target: esnext
6
6
  ℹ tsconfig: tsconfig.json
7
7
  ℹ Build start
8
- ℹ Cleaning 14 files
9
- ℹ dist/handlers/webhook-handler.js 1.86 kB │ gzip: 0.85 kB
10
- ℹ dist/docs/openbanking-powens.docblock.js 1.35 kB │ gzip: 0.68 kB
11
- ℹ dist/handlers/oauth-callback.js 1.29 kB │ gzip: 0.67 kB
12
- ℹ dist/example.js 0.62 kB │ gzip: 0.37 kB
13
- ℹ dist/index.js 0.27 kB │ gzip: 0.16 kB
14
- ℹ dist/docs/index.js 0.04 kB │ gzip: 0.06 kB
15
- ℹ dist/example.d.ts 1.07 kB │ gzip: 0.44 kB
16
- ℹ dist/index.d.ts 0.25 kB │ gzip: 0.13 kB
17
- ℹ dist/handlers/oauth-callback.d.ts 0.17 kB │ gzip: 0.14 kB
18
- ℹ dist/handlers/webhook-handler.d.ts 0.16 kB │ gzip: 0.14 kB
19
- ℹ dist/docs/index.d.ts 0.01 kB │ gzip: 0.03 kB
20
- ℹ dist/docs/openbanking-powens.docblock.d.ts 0.01 kB │ gzip: 0.03 kB
21
- ℹ 12 files, total: 7.11 kB
8
+ ℹ Cleaning 62 files
9
+ ℹ dist/handlers/webhook-handler.js  3.28 kB │ gzip: 1.21 kB
10
+ ℹ dist/handlers/oauth-callback.js  2.37 kB │ gzip: 1.00 kB
11
+ ℹ dist/docs/openbanking-powens.docblock.js  1.57 kB │ gzip: 0.75 kB
12
+ ℹ dist/example.js  0.84 kB │ gzip: 0.44 kB
13
+ ℹ dist/index.js  0.30 kB │ gzip: 0.16 kB
14
+ ℹ dist/docs/index.js  0.04 kB │ gzip: 0.06 kB
15
+ ℹ dist/libs/contracts/dist/docs/accessibility_wcag_compliance_specs.docblock.js 16.91 kB │ gzip: 7.23 kB
16
+ ℹ dist/libs/contracts/dist/docs/tech/schema/README.docblock.js 11.21 kB │ gzip: 4.33 kB
17
+ ℹ dist/libs/contracts/dist/docs/tech/lifecycle-stage-system.docblock.js  9.96 kB │ gzip: 3.83 kB
18
+ ℹ dist/libs/contracts/dist/docs/tech/llm/llm-integration.docblock.js  9.53 kB │ gzip: 3.50 kB
19
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_1_QUICKSTART.docblock.js  9.22 kB │ gzip: 2.77 kB
20
+ ℹ dist/integrations/providers-impls/dist/impls/powens-openbanking.js  7.06 kB │ gzip: 1.87 kB
21
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_3_AUTO_EVOLUTION.docblock.js  6.16 kB │ gzip: 2.65 kB
22
+ ℹ dist/libs/contracts/dist/docs/tech/templates/runtime.docblock.js  5.85 kB │ gzip: 2.66 kB
23
+ ℹ dist/integrations/providers-impls/dist/impls/powens-client.js  5.81 kB │ gzip: 1.89 kB
24
+ ℹ dist/libs/contracts/dist/docs/tech/workflows/overview.docblock.js  5.66 kB │ gzip: 2.34 kB
25
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_5_ZERO_TOUCH_OPERATIONS.docblock.js  4.62 kB │ gzip: 2.13 kB
26
+ ℹ dist/libs/contracts/dist/docs/tech/telemetry-ingest.docblock.js  4.16 kB │ gzip: 1.78 kB
27
+ ℹ dist/libs/contracts/dist/docs/tech/vscode-extension.docblock.js  3.85 kB │ gzip: 1.47 kB
28
+ ℹ dist/libs/contracts/dist/docs/PUBLISHING.docblock.js  3.62 kB │ gzip: 1.65 kB
29
+ ℹ dist/libs/contracts/dist/docs/tech/auth/better-auth-nextjs.docblock.js  3.60 kB │ gzip: 1.53 kB
30
+ ℹ dist/libs/contracts/dist/docs/tech/studio/learning-journeys.docblock.js  3.33 kB │ gzip: 1.47 kB
31
+ ℹ dist/libs/contracts/dist/docs/tech/studio/platform-admin-panel.docblock.js  3.30 kB │ gzip: 1.42 kB
32
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_2_AI_NATIVE_OPERATIONS.docblock.js  3.25 kB │ gzip: 1.59 kB
33
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_4_PERSONALIZATION_ENGINE.docblock.js  3.13 kB │ gzip: 1.47 kB
34
+ ℹ dist/libs/contracts/dist/docs/tech/presentation-runtime.docblock.js  2.97 kB │ gzip: 1.38 kB
35
+ ℹ dist/libs/contracts/dist/docs/tech/studio/project-routing.docblock.js  2.75 kB │ gzip: 1.22 kB
36
+ ℹ dist/libs/contracts/dist/docs/tech/studio/team-invitations.docblock.js  2.40 kB │ gzip: 1.13 kB
37
+ ℹ dist/libs/contracts/dist/docs/presentations.js  2.18 kB │ gzip: 0.77 kB
38
+ ℹ dist/libs/contracts/dist/docs/tech/studio/workspaces.docblock.js  2.13 kB │ gzip: 1.02 kB
39
+ ℹ dist/libs/contracts/dist/docs/tech/studio/learning-events.docblock.js  2.04 kB │ gzip: 0.97 kB
40
+ ℹ dist/libs/contracts/dist/docs/tech/contracts/openapi-export.docblock.js  1.96 kB │ gzip: 0.97 kB
41
+ ℹ dist/libs/contracts/dist/docs/tech/studio/project-access-teams.docblock.js  1.85 kB │ gzip: 0.87 kB
42
+ ℹ dist/libs/contracts/dist/docs/tech/studio/workspace-ops.docblock.js  1.73 kB │ gzip: 0.82 kB
43
+ ℹ dist/libs/contracts/dist/docs/index.js  1.58 kB │ gzip: 0.51 kB
44
+ ℹ dist/libs/contracts/dist/docs/tech/mcp-endpoints.docblock.js  1.46 kB │ gzip: 0.77 kB
45
+ ℹ dist/libs/contracts/dist/docs/registry.js  1.25 kB │ gzip: 0.56 kB
46
+ ℹ dist/libs/contracts/dist/docs/tech/studio/sandbox-unlogged.docblock.js  1.14 kB │ gzip: 0.65 kB
47
+ ℹ dist/example.d.ts  1.07 kB │ gzip: 0.44 kB
48
+ ℹ dist/index.d.ts  0.25 kB │ gzip: 0.13 kB
49
+ ℹ dist/handlers/oauth-callback.d.ts  0.17 kB │ gzip: 0.14 kB
50
+ ℹ dist/handlers/webhook-handler.d.ts  0.16 kB │ gzip: 0.14 kB
51
+ ℹ dist/docs/index.d.ts  0.01 kB │ gzip: 0.03 kB
52
+ ℹ dist/docs/openbanking-powens.docblock.d.ts  0.01 kB │ gzip: 0.03 kB
53
+ ℹ 44 files, total: 155.76 kB
22
54
  [UNRESOLVED_IMPORT] Warning: Could not resolve 'node:crypto' in src/handlers/webhook-handler.ts
23
55
  ╭─[ src/handlers/webhook-handler.ts:7:45 ]
24
56
  │
@@ -29,4 +61,14 @@ $ tsdown
29
61
   │ Help: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
30
62
  ───╯
31
63
 
32
- ✔ Build complete in 14547ms
64
+ [UNRESOLVED_IMPORT] Warning: Could not resolve 'node:url' in ../../integrations/providers-impls/dist/impls/powens-client.js
65
+ ╭─[ ../../integrations/providers-impls/dist/impls/powens-client.js:1:21 ]
66
+ │
67
+ 1 │ import { URL } from "node:url";
68
+  │ ─────┬────
69
+  │ ╰────── Module not found, treating it as an external dependency
70
+  │
71
+  │ Help: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
72
+ ───╯
73
+
74
+ ✔ Build complete in 15225ms
@@ -6,19 +6,51 @@ $ tsdown
6
6
  ℹ target: esnext
7
7
  ℹ tsconfig: tsconfig.json
8
8
  ℹ Build start
9
- ℹ dist/handlers/webhook-handler.js 1.86 kB │ gzip: 0.85 kB
10
- ℹ dist/docs/openbanking-powens.docblock.js 1.35 kB │ gzip: 0.68 kB
11
- ℹ dist/handlers/oauth-callback.js 1.29 kB │ gzip: 0.67 kB
12
- ℹ dist/example.js 0.62 kB │ gzip: 0.37 kB
13
- ℹ dist/index.js 0.27 kB │ gzip: 0.16 kB
14
- ℹ dist/docs/index.js 0.04 kB │ gzip: 0.06 kB
15
- ℹ dist/example.d.ts 1.07 kB │ gzip: 0.44 kB
16
- ℹ dist/index.d.ts 0.25 kB │ gzip: 0.13 kB
17
- ℹ dist/handlers/oauth-callback.d.ts 0.17 kB │ gzip: 0.14 kB
18
- ℹ dist/handlers/webhook-handler.d.ts 0.16 kB │ gzip: 0.14 kB
19
- ℹ dist/docs/index.d.ts 0.01 kB │ gzip: 0.03 kB
20
- ℹ dist/docs/openbanking-powens.docblock.d.ts 0.01 kB │ gzip: 0.03 kB
21
- ℹ 12 files, total: 7.11 kB
9
+ ℹ dist/handlers/webhook-handler.js  3.28 kB │ gzip: 1.21 kB
10
+ ℹ dist/handlers/oauth-callback.js  2.37 kB │ gzip: 1.00 kB
11
+ ℹ dist/docs/openbanking-powens.docblock.js  1.57 kB │ gzip: 0.75 kB
12
+ ℹ dist/example.js  0.84 kB │ gzip: 0.44 kB
13
+ ℹ dist/index.js  0.30 kB │ gzip: 0.16 kB
14
+ ℹ dist/docs/index.js  0.04 kB │ gzip: 0.06 kB
15
+ ℹ dist/libs/contracts/dist/docs/accessibility_wcag_compliance_specs.docblock.js 16.91 kB │ gzip: 7.23 kB
16
+ ℹ dist/libs/contracts/dist/docs/tech/schema/README.docblock.js 11.21 kB │ gzip: 4.33 kB
17
+ ℹ dist/libs/contracts/dist/docs/tech/lifecycle-stage-system.docblock.js  9.96 kB │ gzip: 3.83 kB
18
+ ℹ dist/libs/contracts/dist/docs/tech/llm/llm-integration.docblock.js  9.53 kB │ gzip: 3.50 kB
19
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_1_QUICKSTART.docblock.js  9.22 kB │ gzip: 2.77 kB
20
+ ℹ dist/integrations/providers-impls/dist/impls/powens-openbanking.js  7.06 kB │ gzip: 1.87 kB
21
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_3_AUTO_EVOLUTION.docblock.js  6.16 kB │ gzip: 2.65 kB
22
+ ℹ dist/libs/contracts/dist/docs/tech/templates/runtime.docblock.js  5.85 kB │ gzip: 2.66 kB
23
+ ℹ dist/integrations/providers-impls/dist/impls/powens-client.js  5.81 kB │ gzip: 1.89 kB
24
+ ℹ dist/libs/contracts/dist/docs/tech/workflows/overview.docblock.js  5.66 kB │ gzip: 2.34 kB
25
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_5_ZERO_TOUCH_OPERATIONS.docblock.js  4.62 kB │ gzip: 2.13 kB
26
+ ℹ dist/libs/contracts/dist/docs/tech/telemetry-ingest.docblock.js  4.16 kB │ gzip: 1.78 kB
27
+ ℹ dist/libs/contracts/dist/docs/tech/vscode-extension.docblock.js  3.85 kB │ gzip: 1.47 kB
28
+ ℹ dist/libs/contracts/dist/docs/PUBLISHING.docblock.js  3.62 kB │ gzip: 1.65 kB
29
+ ℹ dist/libs/contracts/dist/docs/tech/auth/better-auth-nextjs.docblock.js  3.60 kB │ gzip: 1.53 kB
30
+ ℹ dist/libs/contracts/dist/docs/tech/studio/learning-journeys.docblock.js  3.33 kB │ gzip: 1.47 kB
31
+ ℹ dist/libs/contracts/dist/docs/tech/studio/platform-admin-panel.docblock.js  3.30 kB │ gzip: 1.42 kB
32
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_2_AI_NATIVE_OPERATIONS.docblock.js  3.25 kB │ gzip: 1.59 kB
33
+ ℹ dist/libs/contracts/dist/docs/tech/PHASE_4_PERSONALIZATION_ENGINE.docblock.js  3.13 kB │ gzip: 1.47 kB
34
+ ℹ dist/libs/contracts/dist/docs/tech/presentation-runtime.docblock.js  2.97 kB │ gzip: 1.38 kB
35
+ ℹ dist/libs/contracts/dist/docs/tech/studio/project-routing.docblock.js  2.75 kB │ gzip: 1.22 kB
36
+ ℹ dist/libs/contracts/dist/docs/tech/studio/team-invitations.docblock.js  2.40 kB │ gzip: 1.13 kB
37
+ ℹ dist/libs/contracts/dist/docs/presentations.js  2.18 kB │ gzip: 0.77 kB
38
+ ℹ dist/libs/contracts/dist/docs/tech/studio/workspaces.docblock.js  2.13 kB │ gzip: 1.02 kB
39
+ ℹ dist/libs/contracts/dist/docs/tech/studio/learning-events.docblock.js  2.04 kB │ gzip: 0.97 kB
40
+ ℹ dist/libs/contracts/dist/docs/tech/contracts/openapi-export.docblock.js  1.96 kB │ gzip: 0.97 kB
41
+ ℹ dist/libs/contracts/dist/docs/tech/studio/project-access-teams.docblock.js  1.85 kB │ gzip: 0.87 kB
42
+ ℹ dist/libs/contracts/dist/docs/tech/studio/workspace-ops.docblock.js  1.73 kB │ gzip: 0.82 kB
43
+ ℹ dist/libs/contracts/dist/docs/index.js  1.58 kB │ gzip: 0.51 kB
44
+ ℹ dist/libs/contracts/dist/docs/tech/mcp-endpoints.docblock.js  1.46 kB │ gzip: 0.77 kB
45
+ ℹ dist/libs/contracts/dist/docs/registry.js  1.25 kB │ gzip: 0.56 kB
46
+ ℹ dist/libs/contracts/dist/docs/tech/studio/sandbox-unlogged.docblock.js  1.14 kB │ gzip: 0.65 kB
47
+ ℹ dist/example.d.ts  1.07 kB │ gzip: 0.44 kB
48
+ ℹ dist/index.d.ts  0.25 kB │ gzip: 0.13 kB
49
+ ℹ dist/handlers/oauth-callback.d.ts  0.17 kB │ gzip: 0.14 kB
50
+ ℹ dist/handlers/webhook-handler.d.ts  0.16 kB │ gzip: 0.14 kB
51
+ ℹ dist/docs/index.d.ts  0.01 kB │ gzip: 0.03 kB
52
+ ℹ dist/docs/openbanking-powens.docblock.d.ts  0.01 kB │ gzip: 0.03 kB
53
+ ℹ 44 files, total: 155.76 kB
22
54
  [UNRESOLVED_IMPORT] Warning: Could not resolve 'node:crypto' in src/handlers/webhook-handler.ts
23
55
  ╭─[ src/handlers/webhook-handler.ts:7:45 ]
24
56
  │
@@ -29,5 +61,15 @@ $ tsdown
29
61
   │ Help: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
30
62
  ───╯
31
63
 
32
- ✔ Build complete in 5848ms
64
+ [UNRESOLVED_IMPORT] Warning: Could not resolve 'node:url' in ../../integrations/providers-impls/dist/impls/powens-client.js
65
+ ╭─[ ../../integrations/providers-impls/dist/impls/powens-client.js:1:21 ]
66
+ │
67
+ 1 │ import { URL } from "node:url";
68
+  │ ─────┬────
69
+  │ ╰────── Module not found, treating it as an external dependency
70
+  │
71
+  │ Help: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
72
+ ───╯
73
+
74
+ ✔ Build complete in 12876ms
33
75
  $ tsc --noEmit
package/CHANGELOG.md CHANGED
@@ -1,6 +1,6 @@
1
1
  # @lssm/example.openbanking-powens
2
2
 
3
- ## 0.0.0-canary-20251217060834
3
+ ## 0.0.0-canary-20251217072406
4
4
 
5
5
  ### Minor Changes
6
6
 
@@ -9,5 +9,6 @@
9
9
  ### Patch Changes
10
10
 
11
11
  - Updated dependencies [66a5dfd]
12
- - @lssm/integration.providers-impls@0.0.0-canary-20251217060834
13
- - @lssm/lib.contracts@0.0.0-canary-20251217060834
12
+ - @lssm/integration.providers-impls@0.0.0-canary-20251217072406
13
+ - @lssm/lib.contracts@0.0.0-canary-20251217072406
14
+ - @lssm/lib.schema@0.0.0-canary-20251217072406
@@ -1 +1 @@
1
- import"./openbanking-powens.docblock.js";
1
+ import "./openbanking-powens.docblock.js";
@@ -1,14 +1,30 @@
1
- import{registerDocBlocks as e}from"@lssm/lib.contracts/docs";e([{id:`docs.examples.openbanking-powens`,title:`Open Banking — Powens (example)`,summary:`Framework-neutral OAuth callback + webhook handler patterns for Powens, orchestrating canonical sync workflows.`,kind:`reference`,visibility:`public`,route:`/docs/examples/openbanking-powens`,tags:[`openbanking`,`powens`,`integration`,`example`],body:`## What this example shows
2
- - OAuth callback handler: exchange auth code, map powens user, enqueue sync workflow.
3
- - Webhook handler: verify signature, route event → workflow, optionally refresh balances.
1
+ import { registerDocBlocks } from "../libs/contracts/dist/docs/registry.js";
2
+ import "../libs/contracts/dist/docs/index.js";
4
3
 
5
- ## Guardrails
6
- - Secrets via secret providers/env only.
7
- - Verify webhook signatures.
8
- - Keep side effects explicit: enqueue workflows instead of mutating canonical stores inline.`},{id:`docs.examples.openbanking-powens.usage`,title:`Open Banking — Powens — Usage`,summary:`How to integrate the handlers in a fetch-compatible runtime.`,kind:`usage`,visibility:`public`,route:`/docs/examples/openbanking-powens/usage`,tags:[`openbanking`,`usage`],body:`## Usage
9
- - Wire \`powensOAuthCallbackHandler(req)\` at your OAuth redirect route.
10
- - Wire \`powensWebhookHandler(req)\` at your webhook route.
4
+ //#region src/docs/openbanking-powens.docblock.ts
5
+ registerDocBlocks([{
6
+ id: "docs.examples.openbanking-powens",
7
+ title: "Open Banking — Powens (example)",
8
+ summary: "Framework-neutral OAuth callback + webhook handler patterns for Powens, orchestrating canonical sync workflows.",
9
+ kind: "reference",
10
+ visibility: "public",
11
+ route: "/docs/examples/openbanking-powens",
12
+ tags: [
13
+ "openbanking",
14
+ "powens",
15
+ "integration",
16
+ "example"
17
+ ],
18
+ body: `## What this example shows\n- OAuth callback handler: exchange auth code, map powens user, enqueue sync workflow.\n- Webhook handler: verify signature, route event → workflow, optionally refresh balances.\n\n## Guardrails\n- Secrets via secret providers/env only.\n- Verify webhook signatures.\n- Keep side effects explicit: enqueue workflows instead of mutating canonical stores inline.`
19
+ }, {
20
+ id: "docs.examples.openbanking-powens.usage",
21
+ title: "Open Banking — Powens — Usage",
22
+ summary: "How to integrate the handlers in a fetch-compatible runtime.",
23
+ kind: "usage",
24
+ visibility: "public",
25
+ route: "/docs/examples/openbanking-powens/usage",
26
+ tags: ["openbanking", "usage"],
27
+ body: `## Usage\n- Wire \`powensOAuthCallbackHandler(req)\` at your OAuth redirect route.\n- Wire \`powensWebhookHandler(req)\` at your webhook route.\n\n## Notes\n- Replace the fake stores with your app-layer persistence.\n- Enqueue ContractSpec workflows for canonical upserts and telemetry.`
28
+ }]);
11
29
 
12
- ## Notes
13
- - Replace the fake stores with your app-layer persistence.
14
- - Enqueue ContractSpec workflows for canonical upserts and telemetry.`}]);
30
+ //#endregion
package/dist/example.js CHANGED
@@ -1 +1,39 @@
1
- var e={id:`openbanking-powens`,title:`Open Banking — Powens`,summary:`OAuth callback + webhook handler patterns for Powens open banking integration (provider + workflow orchestration).`,tags:[`openbanking`,`powens`,`oauth`,`webhooks`,`integrations`],kind:`integration`,visibility:`public`,docs:{rootDocId:`docs.examples.openbanking-powens`,usageDocId:`docs.examples.openbanking-powens.usage`},entrypoints:{packageName:`@lssm/example.openbanking-powens`,docs:`./docs`},surfaces:{templates:!0,sandbox:{enabled:!0,modes:[`markdown`,`specs`]},studio:{enabled:!0,installable:!0},mcp:{enabled:!0}}};export{e as default};
1
+ //#region src/example.ts
2
+ const example = {
3
+ id: "openbanking-powens",
4
+ title: "Open Banking — Powens",
5
+ summary: "OAuth callback + webhook handler patterns for Powens open banking integration (provider + workflow orchestration).",
6
+ tags: [
7
+ "openbanking",
8
+ "powens",
9
+ "oauth",
10
+ "webhooks",
11
+ "integrations"
12
+ ],
13
+ kind: "integration",
14
+ visibility: "public",
15
+ docs: {
16
+ rootDocId: "docs.examples.openbanking-powens",
17
+ usageDocId: "docs.examples.openbanking-powens.usage"
18
+ },
19
+ entrypoints: {
20
+ packageName: "@lssm/example.openbanking-powens",
21
+ docs: "./docs"
22
+ },
23
+ surfaces: {
24
+ templates: true,
25
+ sandbox: {
26
+ enabled: true,
27
+ modes: ["markdown", "specs"]
28
+ },
29
+ studio: {
30
+ enabled: true,
31
+ installable: true
32
+ },
33
+ mcp: { enabled: true }
34
+ }
35
+ };
36
+ var example_default = example;
37
+
38
+ //#endregion
39
+ export { example_default as default };
@@ -1 +1,63 @@
1
- import{PowensOpenBankingProvider as e}from"@lssm/integration.providers-impls/impls/powens-openbanking";async function t(t){let a=new URL(t.url),o=a.searchParams.get(`code`),s=a.searchParams.get(`state`),c=a.searchParams.get(`user_uuid`);if(!o||!s||!c)return new Response(`Missing Powens OAuth params`,{status:400});let l=await n(s);if(!l)return new Response(`Unknown Powens OAuth state`,{status:404});let u=await r(l.meta.id),d=await new e({clientId:u.clientId,clientSecret:u.clientSecret,apiKey:u.apiKey,environment:l.config.environment,baseUrl:l.config.baseUrl}).listAccounts({tenantId:l.meta.tenantId,connectionId:l.meta.id,userId:c});await l.storePowensUser({tenantUserId:l.meta.tenantUserId,powensUserUuid:c,authCode:o}),await i(`pfo.workflow.sync-openbanking-accounts`,{tenantId:l.meta.tenantId,userUuid:c,connectionId:l.meta.id,previewAccounts:d.accounts});let f=process.env.APP_DASHBOARD_URL??``;return Response.redirect(`${f}/banking/linked?tenant=${l.meta.tenantId}`,302)}async function n(e){return a.connections.find(t=>t.state===e)??null}async function r(e){let t=o[e];if(!t)throw Error(`Missing Powens secrets for ${e}`);return t}async function i(e,t){await s.enqueue({name:e,input:t})}const a={connections:[]},o={},s={enqueue:async e=>{}};export{t as powensOAuthCallbackHandler};
1
+ import { PowensOpenBankingProvider } from "../integrations/providers-impls/dist/impls/powens-openbanking.js";
2
+
3
+ //#region src/handlers/oauth-callback.ts
4
+ /**
5
+ * Example OAuth callback handler for Powens (open banking).
6
+ *
7
+ * This example stays framework-neutral: it operates on the standard `Request`
8
+ * type so it can be used in Next.js, Elysia, or any fetch-compatible runtime.
9
+ */
10
+ async function powensOAuthCallbackHandler(req) {
11
+ const url = new URL(req.url);
12
+ const code = url.searchParams.get("code");
13
+ const state = url.searchParams.get("state");
14
+ const userUuid = url.searchParams.get("user_uuid");
15
+ if (!code || !state || !userUuid) return new Response("Missing Powens OAuth params", { status: 400 });
16
+ const connection = await getConnectionByState(state);
17
+ if (!connection) return new Response("Unknown Powens OAuth state", { status: 404 });
18
+ const secrets = await getPowensSecretsForConnection(connection.meta.id);
19
+ const preview = await new PowensOpenBankingProvider({
20
+ clientId: secrets.clientId,
21
+ clientSecret: secrets.clientSecret,
22
+ apiKey: secrets.apiKey,
23
+ environment: connection.config.environment,
24
+ baseUrl: connection.config.baseUrl
25
+ }).listAccounts({
26
+ tenantId: connection.meta.tenantId,
27
+ connectionId: connection.meta.id,
28
+ userId: userUuid
29
+ });
30
+ await connection.storePowensUser({
31
+ tenantUserId: connection.meta.tenantUserId,
32
+ powensUserUuid: userUuid,
33
+ authCode: code
34
+ });
35
+ await enqueueWorkflow("pfo.workflow.sync-openbanking-accounts", {
36
+ tenantId: connection.meta.tenantId,
37
+ userUuid,
38
+ connectionId: connection.meta.id,
39
+ previewAccounts: preview.accounts
40
+ });
41
+ const redirectBase = process.env.APP_DASHBOARD_URL ?? "";
42
+ return Response.redirect(`${redirectBase}/banking/linked?tenant=${connection.meta.tenantId}`, 302);
43
+ }
44
+ async function getConnectionByState(state) {
45
+ return fakeDatabase.connections.find((conn) => conn.state === state) ?? null;
46
+ }
47
+ async function getPowensSecretsForConnection(connectionId) {
48
+ const secret = fakeSecretStore[connectionId];
49
+ if (!secret) throw new Error(`Missing Powens secrets for ${connectionId}`);
50
+ return secret;
51
+ }
52
+ async function enqueueWorkflow(name, input) {
53
+ await fakeWorkflowQueue.enqueue({
54
+ name,
55
+ input
56
+ });
57
+ }
58
+ const fakeDatabase = { connections: [] };
59
+ const fakeSecretStore = {};
60
+ const fakeWorkflowQueue = { enqueue: async (_payload) => {} };
61
+
62
+ //#endregion
63
+ export { powensOAuthCallbackHandler };
@@ -1 +1,87 @@
1
- import{PowensOpenBankingProvider as e}from"@lssm/integration.providers-impls/impls/powens-openbanking";import{createHmac as t,timingSafeEqual as n}from"node:crypto";async function r(t){let n=t.headers.get(`x-powens-signature`),r=t.headers.get(`x-powens-state`),l=await t.text();if(!n||!r)return new Response(`Missing Powens signature headers`,{status:400});let u=await a(r);if(!u)return new Response(`Unknown Powens state header`,{status:404});let d=await o(u.meta.id);if(!i(l,n,d.webhookSecret))return new Response(`Invalid Powens webhook signature`,{status:401});let f=JSON.parse(l),p=new e({clientId:d.clientId,clientSecret:d.clientSecret,apiKey:d.apiKey,environment:u.config.environment,baseUrl:u.config.baseUrl});switch(f.type){case`connection.updated`:case`user.sync.completed`:await s(`pfo.workflow.sync-openbanking-accounts`,{tenantId:u.meta.tenantId,connectionId:u.meta.id,userUuid:f.user_uuid});break;case`transactions.created`:case`transactions.updated`:await s(`pfo.workflow.sync-openbanking-transactions`,{tenantId:u.meta.tenantId,connectionId:u.meta.id,userUuid:f.user_uuid,accountId:f.account_uuid});break;default:await c(f)}return f.account_uuid&&await p.getBalances({tenantId:u.meta.tenantId,connectionId:u.meta.id,accountId:f.account_uuid}),new Response(`OK`,{status:200})}function i(e,r,i){let a=t(`sha256`,i).update(e).digest(`hex`),o=Buffer.from(a,`hex`),s=Buffer.from(r,`hex`);return o.length===s.length&&n(o,s)}async function a(e){return l.connections.find(t=>t.state===e)??null}async function o(e){let t=u[e];if(!t)throw Error(`Missing Powens secrets for ${e}`);return t}async function s(e,t){await d.enqueue({name:e,input:t})}async function c(e){await f.record({event:`openbanking.webhook.unmapped`,payload:`redacted`})}const l={connections:[]},u={},d={enqueue:async e=>{}},f={record:async e=>{}};export{r as powensWebhookHandler};
1
+ import { PowensOpenBankingProvider } from "../integrations/providers-impls/dist/impls/powens-openbanking.js";
2
+ import { createHmac, timingSafeEqual } from "node:crypto";
3
+
4
+ //#region src/handlers/webhook-handler.ts
5
+ /**
6
+ * Example Powens webhook handler (fetch-compatible).
7
+ *
8
+ * Verifies signature, then enqueues the canonical workflows to keep the ledger
9
+ * in sync. Unknown events are ignored (or can be recorded by the app layer).
10
+ */
11
+ async function powensWebhookHandler(req) {
12
+ const signature = req.headers.get("x-powens-signature");
13
+ const stateHeader = req.headers.get("x-powens-state");
14
+ const payload = await req.text();
15
+ if (!signature || !stateHeader) return new Response("Missing Powens signature headers", { status: 400 });
16
+ const connection = await getConnectionByState(stateHeader);
17
+ if (!connection) return new Response("Unknown Powens state header", { status: 404 });
18
+ const secrets = await getPowensSecretsForConnection(connection.meta.id);
19
+ if (!verifySignature(payload, signature, secrets.webhookSecret)) return new Response("Invalid Powens webhook signature", { status: 401 });
20
+ const event = JSON.parse(payload);
21
+ const provider = new PowensOpenBankingProvider({
22
+ clientId: secrets.clientId,
23
+ clientSecret: secrets.clientSecret,
24
+ apiKey: secrets.apiKey,
25
+ environment: connection.config.environment,
26
+ baseUrl: connection.config.baseUrl
27
+ });
28
+ switch (event.type) {
29
+ case "connection.updated":
30
+ case "user.sync.completed":
31
+ await enqueueWorkflow("pfo.workflow.sync-openbanking-accounts", {
32
+ tenantId: connection.meta.tenantId,
33
+ connectionId: connection.meta.id,
34
+ userUuid: event.user_uuid
35
+ });
36
+ break;
37
+ case "transactions.created":
38
+ case "transactions.updated":
39
+ await enqueueWorkflow("pfo.workflow.sync-openbanking-transactions", {
40
+ tenantId: connection.meta.tenantId,
41
+ connectionId: connection.meta.id,
42
+ userUuid: event.user_uuid,
43
+ accountId: event.account_uuid
44
+ });
45
+ break;
46
+ default: await logUnmappedEvent(event);
47
+ }
48
+ if (event.account_uuid) await provider.getBalances({
49
+ tenantId: connection.meta.tenantId,
50
+ connectionId: connection.meta.id,
51
+ accountId: event.account_uuid
52
+ });
53
+ return new Response("OK", { status: 200 });
54
+ }
55
+ function verifySignature(payload, signature, secret) {
56
+ const digest = createHmac("sha256", secret).update(payload).digest("hex");
57
+ const a = Buffer.from(digest, "hex");
58
+ const b = Buffer.from(signature, "hex");
59
+ return a.length === b.length && timingSafeEqual(a, b);
60
+ }
61
+ async function getConnectionByState(state) {
62
+ return fakeDatabase.connections.find((conn) => conn.state === state) ?? null;
63
+ }
64
+ async function getPowensSecretsForConnection(connectionId) {
65
+ const secret = fakeSecretStore[connectionId];
66
+ if (!secret) throw new Error(`Missing Powens secrets for ${connectionId}`);
67
+ return secret;
68
+ }
69
+ async function enqueueWorkflow(name, input) {
70
+ await fakeWorkflowQueue.enqueue({
71
+ name,
72
+ input
73
+ });
74
+ }
75
+ async function logUnmappedEvent(_event) {
76
+ await fakeTelemetryLogger.record({
77
+ event: "openbanking.webhook.unmapped",
78
+ payload: "redacted"
79
+ });
80
+ }
81
+ const fakeDatabase = { connections: [] };
82
+ const fakeSecretStore = {};
83
+ const fakeWorkflowQueue = { enqueue: async (_payload) => {} };
84
+ const fakeTelemetryLogger = { record: async (_payload) => {} };
85
+
86
+ //#endregion
87
+ export { powensWebhookHandler };
package/dist/index.js CHANGED
@@ -1 +1,6 @@
1
- import e from"./example.js";import{powensOAuthCallbackHandler as t}from"./handlers/oauth-callback.js";import{powensWebhookHandler as n}from"./handlers/webhook-handler.js";import"./docs/index.js";export{e as example,t as powensOAuthCallbackHandler,n as powensWebhookHandler};
1
+ import example_default from "./example.js";
2
+ import { powensOAuthCallbackHandler } from "./handlers/oauth-callback.js";
3
+ import { powensWebhookHandler } from "./handlers/webhook-handler.js";
4
+ import "./docs/index.js";
5
+
6
+ export { example_default as example, powensOAuthCallbackHandler, powensWebhookHandler };
@@ -0,0 +1,171 @@
1
+ import { URL } from "node:url";
2
+
3
+ //#region ../../integrations/providers-impls/dist/impls/powens-client.js
4
+ const POWENS_BASE_URL = {
5
+ sandbox: "https://api-sandbox.powens.com/v2",
6
+ production: "https://api.powens.com/v2"
7
+ };
8
+ var PowensClientError = class extends Error {
9
+ status;
10
+ code;
11
+ requestId;
12
+ response;
13
+ constructor(message, status, code, requestId, response) {
14
+ super(message);
15
+ this.name = "PowensClientError";
16
+ this.status = status;
17
+ this.code = code;
18
+ this.requestId = requestId;
19
+ this.response = response;
20
+ }
21
+ };
22
+ var PowensClient = class {
23
+ clientId;
24
+ clientSecret;
25
+ apiKey;
26
+ fetchImpl;
27
+ logger;
28
+ defaultTimeoutMs;
29
+ token;
30
+ baseUrl;
31
+ constructor(options) {
32
+ this.clientId = options.clientId;
33
+ this.clientSecret = options.clientSecret;
34
+ this.apiKey = options.apiKey;
35
+ this.fetchImpl = options.fetchImpl ?? fetch;
36
+ this.logger = options.logger;
37
+ this.defaultTimeoutMs = options.defaultTimeoutMs ?? 15e3;
38
+ this.baseUrl = options.baseUrl ?? POWENS_BASE_URL[options.environment] ?? POWENS_BASE_URL.production;
39
+ }
40
+ async listAccounts(params) {
41
+ const searchParams = {
42
+ cursor: params.cursor,
43
+ limit: params.limit,
44
+ include_balances: params.includeBalances,
45
+ institution_uuid: params.institutionUuid
46
+ };
47
+ return await this.request({
48
+ method: "GET",
49
+ path: `/users/${encodeURIComponent(params.userUuid)}/accounts`,
50
+ searchParams
51
+ });
52
+ }
53
+ async getAccount(accountUuid) {
54
+ return this.request({
55
+ method: "GET",
56
+ path: `/accounts/${encodeURIComponent(accountUuid)}`
57
+ });
58
+ }
59
+ async listTransactions(params) {
60
+ const searchParams = {
61
+ cursor: params.cursor,
62
+ limit: params.limit,
63
+ from: params.from,
64
+ to: params.to,
65
+ include_pending: params.includePending
66
+ };
67
+ return this.request({
68
+ method: "GET",
69
+ path: `/accounts/${encodeURIComponent(params.accountUuid)}/transactions`,
70
+ searchParams
71
+ });
72
+ }
73
+ async getBalances(accountUuid) {
74
+ return this.request({
75
+ method: "GET",
76
+ path: `/accounts/${encodeURIComponent(accountUuid)}/balances`
77
+ });
78
+ }
79
+ async getConnectionStatus(connectionUuid) {
80
+ return this.request({
81
+ method: "GET",
82
+ path: `/connections/${encodeURIComponent(connectionUuid)}`
83
+ });
84
+ }
85
+ async request(options) {
86
+ const url = new URL(options.path, this.baseUrl);
87
+ if (options.searchParams) for (const [key, value] of Object.entries(options.searchParams)) {
88
+ if (value === void 0 || value === null) continue;
89
+ url.searchParams.set(key, String(value));
90
+ }
91
+ const headers = {
92
+ Accept: "application/json",
93
+ "Content-Type": "application/json",
94
+ ...options.headers
95
+ };
96
+ if (this.apiKey) headers["x-api-key"] = this.apiKey;
97
+ if (!options.skipAuth) headers.Authorization = `Bearer ${await this.ensureAccessToken()}`;
98
+ const controller = new AbortController();
99
+ const timeout = setTimeout(() => controller.abort(), options.timeoutMs ?? this.defaultTimeoutMs);
100
+ try {
101
+ const response = await this.fetchImpl(url, {
102
+ method: options.method,
103
+ headers,
104
+ body: options.body ? JSON.stringify(options.body) : void 0,
105
+ signal: controller.signal
106
+ });
107
+ const requestId = response.headers.get("x-request-id") ?? void 0;
108
+ if (!response.ok) {
109
+ let errorBody;
110
+ try {
111
+ errorBody = await response.json();
112
+ } catch {}
113
+ const errorObject = typeof errorBody === "object" && errorBody !== null ? errorBody : void 0;
114
+ const message = typeof errorObject?.message === "string" ? errorObject.message : `Powens API request failed with status ${response.status}`;
115
+ const code = typeof errorObject?.code === "string" ? errorObject.code : void 0;
116
+ throw new PowensClientError(message, response.status, code, requestId, errorBody);
117
+ }
118
+ if (response.status === 204) return;
119
+ try {
120
+ return await response.json();
121
+ } catch (error) {
122
+ this.logger?.error?.("[PowensClient] Failed to parse JSON response", error);
123
+ throw new PowensClientError("Failed to parse Powens response payload as JSON", response.status, void 0, requestId);
124
+ }
125
+ } catch (error) {
126
+ if (error instanceof PowensClientError) throw error;
127
+ if (error.name === "AbortError") throw new PowensClientError(`Powens API request timed out after ${options.timeoutMs ?? this.defaultTimeoutMs}ms`, 408);
128
+ this.logger?.error?.("[PowensClient] Request failed", error);
129
+ throw new PowensClientError(error.message ?? "Powens API request failed", 500);
130
+ } finally {
131
+ clearTimeout(timeout);
132
+ }
133
+ }
134
+ async ensureAccessToken() {
135
+ if (this.token && Date.now() < this.token.expiresAt - 5e3) return this.token.accessToken;
136
+ this.token = await this.fetchAccessToken();
137
+ return this.token.accessToken;
138
+ }
139
+ async fetchAccessToken() {
140
+ const url = new URL("/oauth/token", this.baseUrl);
141
+ const basicAuth = Buffer.from(`${this.clientId}:${this.clientSecret}`, "utf-8").toString("base64");
142
+ const response = await this.fetchImpl(url, {
143
+ method: "POST",
144
+ headers: {
145
+ Authorization: `Basic ${basicAuth}`,
146
+ "Content-Type": "application/x-www-form-urlencoded",
147
+ Accept: "application/json"
148
+ },
149
+ body: new URLSearchParams({ grant_type: "client_credentials" }).toString()
150
+ });
151
+ if (!response.ok) {
152
+ let errorBody;
153
+ try {
154
+ errorBody = await response.json();
155
+ } catch {}
156
+ const errorObject = typeof errorBody === "object" && errorBody !== null ? errorBody : void 0;
157
+ throw new PowensClientError(typeof errorObject?.error_description === "string" ? errorObject.error_description : "Failed to obtain Powens access token", response.status, void 0, void 0, errorBody);
158
+ }
159
+ const payload = await response.json();
160
+ const expiresAt = Date.now() + payload.expires_in * 1e3;
161
+ this.logger?.debug?.("[PowensClient] Received access token", { expiresIn: payload.expires_in });
162
+ return {
163
+ accessToken: payload.access_token,
164
+ expiresAt,
165
+ scope: payload.scope
166
+ };
167
+ }
168
+ };
169
+
170
+ //#endregion
171
+ export { PowensClient, PowensClientError };