@contractspec/example.openbanking-powens 1.44.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,39 @@
1
+ $ tsdown
2
+ ℹ tsdown v0.18.3 powered by rolldown v1.0.0-beta.57
3
+ ℹ config file: /home/runner/work/contractspec/contractspec/packages/examples/openbanking-powens/tsdown.config.js
4
+ ℹ entry: src/example.ts, src/index.ts, src/docs/index.ts, src/docs/openbanking-powens.docblock.ts, src/handlers/oauth-callback.ts, src/handlers/webhook-handler.ts
5
+ ℹ target: esnext
6
+ ℹ tsconfig: tsconfig.json
7
+ ℹ Build start
8
+ ℹ Cleaning 21 files
9
+ ℹ dist/handlers/webhook-handler.js 3.33 kB │ gzip: 1.23 kB
10
+ ℹ dist/handlers/oauth-callback.js 2.42 kB │ gzip: 1.02 kB
11
+ ℹ dist/docs/openbanking-powens.docblock.js 1.57 kB │ gzip: 0.76 kB
12
+ ℹ dist/example.js 0.88 kB │ gzip: 0.47 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/handlers/webhook-handler.js.map 6.46 kB │ gzip: 2.18 kB
16
+ ℹ dist/handlers/oauth-callback.js.map 4.84 kB │ gzip: 1.81 kB
17
+ ℹ dist/docs/openbanking-powens.docblock.js.map 2.07 kB │ gzip: 0.94 kB
18
+ ℹ dist/example.js.map 1.25 kB │ gzip: 0.64 kB
19
+ ℹ dist/handlers/webhook-handler.d.ts.map 0.17 kB │ gzip: 0.14 kB
20
+ ℹ dist/handlers/oauth-callback.d.ts.map 0.17 kB │ gzip: 0.15 kB
21
+ ℹ dist/example.d.ts.map 0.14 kB │ gzip: 0.13 kB
22
+ ℹ dist/example.d.ts 1.12 kB │ gzip: 0.47 kB
23
+ ℹ dist/index.d.ts 0.25 kB │ gzip: 0.13 kB
24
+ ℹ dist/handlers/oauth-callback.d.ts 0.22 kB │ gzip: 0.17 kB
25
+ ℹ dist/handlers/webhook-handler.d.ts 0.21 kB │ gzip: 0.16 kB
26
+ ℹ dist/docs/index.d.ts 0.01 kB │ gzip: 0.03 kB
27
+ ℹ dist/docs/openbanking-powens.docblock.d.ts 0.01 kB │ gzip: 0.03 kB
28
+ ℹ 19 files, total: 25.46 kB
29
+ src/handlers/webhook-handler.ts (7:44) [UNRESOLVED_IMPORT] Warning: Could not resolve 'node:crypto' in src/handlers/webhook-handler.ts
30
+ ╭─[ src/handlers/webhook-handler.ts:7:45 ]
31
+ │
32
+ 7 │ import { createHmac, timingSafeEqual } from 'node:crypto';
33
+  │ ──────┬──────
34
+  │ ╰──────── Module not found, treating it as an external dependency
35
+  │
36
+  │ Help: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
37
+ ───╯
38
+
39
+ ✔ Build complete in 8760ms
@@ -0,0 +1,40 @@
1
+ $ bun build:types && bun build:bundle
2
+ $ tsc --noEmit
3
+ $ tsdown
4
+ ℹ tsdown v0.18.3 powered by rolldown v1.0.0-beta.57
5
+ ℹ config file: /home/runner/work/contractspec/contractspec/packages/examples/openbanking-powens/tsdown.config.js
6
+ ℹ entry: src/example.ts, src/index.ts, src/docs/index.ts, src/docs/openbanking-powens.docblock.ts, src/handlers/oauth-callback.ts, src/handlers/webhook-handler.ts
7
+ ℹ target: esnext
8
+ ℹ tsconfig: tsconfig.json
9
+ ℹ Build start
10
+ ℹ dist/handlers/webhook-handler.js 3.33 kB │ gzip: 1.23 kB
11
+ ℹ dist/handlers/oauth-callback.js 2.42 kB │ gzip: 1.02 kB
12
+ ℹ dist/docs/openbanking-powens.docblock.js 1.57 kB │ gzip: 0.76 kB
13
+ src/handlers/webhook-handler.ts (7:44) [UNRESOLVED_IMPORT] Warning: Could not resolve 'node:crypto' in src/handlers/webhook-handler.ts
14
+ ╭─[ src/handlers/webhook-handler.ts:7:45 ]
15
+ │
16
+ ℹ dist/example.js 0.88 kB │ gzip: 0.47 kB
17
+ 7 │ import { createHmac, timingSafeEqual } from 'node:crypto';
18
+ ℹ dist/index.js 0.30 kB │ gzip: 0.16 kB
19
+ ℹ dist/docs/index.js 0.04 kB │ gzip: 0.06 kB
20
+ ℹ dist/handlers/webhook-handler.js.map 6.46 kB │ gzip: 2.18 kB
21
+ ℹ dist/handlers/oauth-callback.js.map 4.84 kB │ gzip: 1.81 kB
22
+  │ ──────┬──────
23
+ ℹ dist/docs/openbanking-powens.docblock.js.map 2.07 kB │ gzip: 0.94 kB
24
+  │ ╰──────── Module not found, treating it as an external dependency
25
+  │
26
+ ℹ dist/example.js.map 1.25 kB │ gzip: 0.64 kB
27
+ ℹ dist/handlers/webhook-handler.d.ts.map 0.17 kB │ gzip: 0.14 kB
28
+ ℹ dist/handlers/oauth-callback.d.ts.map 0.17 kB │ gzip: 0.15 kB
29
+  │ Help: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
30
+ ℹ dist/example.d.ts.map 0.14 kB │ gzip: 0.13 kB
31
+ ℹ dist/example.d.ts 1.12 kB │ gzip: 0.47 kB
32
+ ℹ dist/index.d.ts 0.25 kB │ gzip: 0.13 kB
33
+ ───╯
34
+
35
+ ℹ dist/handlers/oauth-callback.d.ts 0.22 kB │ gzip: 0.17 kB
36
+ ℹ dist/handlers/webhook-handler.d.ts 0.21 kB │ gzip: 0.16 kB
37
+ ℹ dist/docs/index.d.ts 0.01 kB │ gzip: 0.03 kB
38
+ ℹ dist/docs/openbanking-powens.docblock.d.ts 0.01 kB │ gzip: 0.03 kB
39
+ ℹ 19 files, total: 25.46 kB
40
+ ✔ Build complete in 9385ms
package/CHANGELOG.md ADDED
@@ -0,0 +1,178 @@
1
+ # @contractspec/example.openbanking-powens
2
+
3
+ ## 1.44.0
4
+
5
+ ### Minor Changes
6
+
7
+ - 5f3a868: chore: isolate branding to contractspec.io
8
+
9
+ ### Patch Changes
10
+
11
+ - Updated dependencies [5f3a868]
12
+ - @contractspec/integration.providers-impls@1.44.0
13
+ - @contractspec/lib.contracts@1.44.0
14
+ - @contractspec/lib.schema@1.44.0
15
+
16
+ ## 1.43.4
17
+
18
+ ### Patch Changes
19
+
20
+ - 9216062: fix: cross-platform compatibility
21
+ - Updated dependencies [9216062]
22
+ - @contractspec/integration.providers-impls@1.43.4
23
+ - @contractspec/lib.contracts@1.43.4
24
+ - @contractspec/lib.schema@1.43.3
25
+
26
+ ## 1.43.3
27
+
28
+ ### Patch Changes
29
+
30
+ - 24d9759: improve documentation
31
+ - Updated dependencies [24d9759]
32
+ - @contractspec/integration.providers-impls@1.43.3
33
+ - @contractspec/lib.contracts@1.43.3
34
+ - @contractspec/lib.schema@1.43.2
35
+
36
+ ## 1.43.2
37
+
38
+ ### Patch Changes
39
+
40
+ - e147271: fix: improve stability
41
+ - Updated dependencies [e147271]
42
+ - @contractspec/integration.providers-impls@1.43.2
43
+ - @contractspec/lib.contracts@1.43.2
44
+ - @contractspec/lib.schema@1.43.1
45
+
46
+ ## 1.43.1
47
+
48
+ ### Patch Changes
49
+
50
+ - Updated dependencies [f28fdad]
51
+ - @contractspec/lib.contracts@1.43.1
52
+ - @contractspec/integration.providers-impls@1.43.1
53
+
54
+ ## 1.43.0
55
+
56
+ ### Minor Changes
57
+
58
+ - 042d072: feat: schema declaration using json schema, including zod
59
+
60
+ ### Patch Changes
61
+
62
+ - Updated dependencies [042d072]
63
+ - @contractspec/integration.providers-impls@1.43.0
64
+ - @contractspec/lib.contracts@1.43.0
65
+ - @contractspec/lib.schema@1.43.0
66
+
67
+ ## 1.42.10
68
+
69
+ ### Patch Changes
70
+
71
+ - 1e6a0f1: fix: mcp server
72
+ - Updated dependencies [1e6a0f1]
73
+ - @contractspec/integration.providers-impls@1.42.10
74
+ - @contractspec/lib.contracts@1.42.10
75
+ - @contractspec/lib.schema@1.42.10
76
+
77
+ ## 1.42.9
78
+
79
+ ### Patch Changes
80
+
81
+ - 9281db7: fix ModelRegistry
82
+ - Updated dependencies [9281db7]
83
+ - @contractspec/integration.providers-impls@1.42.9
84
+ - @contractspec/lib.contracts@1.42.9
85
+ - @contractspec/lib.schema@1.42.9
86
+
87
+ ## 1.42.8
88
+
89
+ ### Patch Changes
90
+
91
+ - e07b5ac: fix
92
+ - Updated dependencies [e07b5ac]
93
+ - @contractspec/integration.providers-impls@1.42.8
94
+ - @contractspec/lib.contracts@1.42.8
95
+ - @contractspec/lib.schema@1.42.8
96
+
97
+ ## 1.42.7
98
+
99
+ ### Patch Changes
100
+
101
+ - e9b575d: fix release
102
+ - Updated dependencies [e9b575d]
103
+ - @contractspec/integration.providers-impls@1.42.7
104
+ - @contractspec/lib.contracts@1.42.7
105
+ - @contractspec/lib.schema@1.42.7
106
+
107
+ ## 1.42.6
108
+
109
+ ### Patch Changes
110
+
111
+ - 1500242: fix tooling
112
+ - Updated dependencies [1500242]
113
+ - @contractspec/integration.providers-impls@1.42.6
114
+ - @contractspec/lib.contracts@1.42.6
115
+ - @contractspec/lib.schema@1.42.6
116
+
117
+ ## 1.42.5
118
+
119
+ ### Patch Changes
120
+
121
+ - 1299719: fix vscode
122
+ - Updated dependencies [1299719]
123
+ - @contractspec/integration.providers-impls@1.42.5
124
+ - @contractspec/lib.contracts@1.42.5
125
+ - @contractspec/lib.schema@1.42.5
126
+
127
+ ## 1.42.4
128
+
129
+ ### Patch Changes
130
+
131
+ - ac28b99: fix: generate from openapi
132
+ - Updated dependencies [ac28b99]
133
+ - @contractspec/integration.providers-impls@1.42.4
134
+ - @contractspec/lib.contracts@1.42.4
135
+ - @contractspec/lib.schema@1.42.4
136
+
137
+ ## 1.42.3
138
+
139
+ ### Patch Changes
140
+
141
+ - 3f5d015: fix(tooling): cicd
142
+ - Updated dependencies [3f5d015]
143
+ - @contractspec/integration.providers-impls@1.42.3
144
+ - @contractspec/lib.contracts@1.42.3
145
+ - @contractspec/lib.schema@1.42.3
146
+
147
+ ## 1.42.2
148
+
149
+ ### Patch Changes
150
+
151
+ - 1f9ac4c: fix
152
+ - Updated dependencies [1f9ac4c]
153
+ - @contractspec/integration.providers-impls@1.42.2
154
+ - @contractspec/lib.contracts@1.42.2
155
+ - @contractspec/lib.schema@1.42.2
156
+
157
+ ## 1.42.1
158
+
159
+ ### Patch Changes
160
+
161
+ - f043995: Fix release
162
+ - Updated dependencies [f043995]
163
+ - @contractspec/integration.providers-impls@1.42.1
164
+ - @contractspec/lib.contracts@1.42.1
165
+ - @contractspec/lib.schema@1.42.1
166
+
167
+ ## 1.42.0
168
+
169
+ ### Minor Changes
170
+
171
+ - 8eefd9c: initial release
172
+
173
+ ### Patch Changes
174
+
175
+ - Updated dependencies [8eefd9c]
176
+ - @contractspec/integration.providers-impls@1.42.0
177
+ - @contractspec/lib.contracts@1.42.0
178
+ - @contractspec/lib.schema@1.42.0
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2025 Chaman Ventures, SASU
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
@@ -0,0 +1 @@
1
+ export { };
@@ -0,0 +1 @@
1
+ import "./openbanking-powens.docblock.js";
@@ -0,0 +1 @@
1
+ export { };
@@ -0,0 +1,30 @@
1
+ import { registerDocBlocks } from "@contractspec/lib.contracts/docs";
2
+
3
+ //#region src/docs/openbanking-powens.docblock.ts
4
+ registerDocBlocks([{
5
+ id: "docs.examples.openbanking-powens",
6
+ title: "Open Banking — Powens (example)",
7
+ summary: "Framework-neutral OAuth callback + webhook handler patterns for Powens, orchestrating canonical sync workflows.",
8
+ kind: "reference",
9
+ visibility: "public",
10
+ route: "/docs/examples/openbanking-powens",
11
+ tags: [
12
+ "openbanking",
13
+ "powens",
14
+ "integration",
15
+ "example"
16
+ ],
17
+ 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.`
18
+ }, {
19
+ id: "docs.examples.openbanking-powens.usage",
20
+ title: "Open Banking — Powens — Usage",
21
+ summary: "How to integrate the handlers in a fetch-compatible runtime.",
22
+ kind: "usage",
23
+ visibility: "public",
24
+ route: "/docs/examples/openbanking-powens/usage",
25
+ tags: ["openbanking", "usage"],
26
+ 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.`
27
+ }]);
28
+
29
+ //#endregion
30
+ //# sourceMappingURL=openbanking-powens.docblock.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openbanking-powens.docblock.js","names":[],"sources":["../../src/docs/openbanking-powens.docblock.ts"],"sourcesContent":["import type { DocBlock } from '@contractspec/lib.contracts/docs';\nimport { registerDocBlocks } from '@contractspec/lib.contracts/docs';\n\nconst blocks: DocBlock[] = [\n {\n id: 'docs.examples.openbanking-powens',\n title: 'Open Banking — Powens (example)',\n summary:\n 'Framework-neutral OAuth callback + webhook handler patterns for Powens, orchestrating canonical sync workflows.',\n kind: 'reference',\n visibility: 'public',\n route: '/docs/examples/openbanking-powens',\n tags: ['openbanking', 'powens', 'integration', 'example'],\n 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.`,\n },\n {\n id: 'docs.examples.openbanking-powens.usage',\n title: 'Open Banking — Powens — Usage',\n summary: 'How to integrate the handlers in a fetch-compatible runtime.',\n kind: 'usage',\n visibility: 'public',\n route: '/docs/examples/openbanking-powens/usage',\n tags: ['openbanking', 'usage'],\n 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.`,\n },\n];\n\nregisterDocBlocks(blocks);\n"],"mappings":";;;AA2BA,kBAxB2B,CACzB;CACE,IAAI;CACJ,OAAO;CACP,SACE;CACF,MAAM;CACN,YAAY;CACZ,OAAO;CACP,MAAM;EAAC;EAAe;EAAU;EAAe;EAAU;CACzD,MAAM;CACP,EACD;CACE,IAAI;CACJ,OAAO;CACP,SAAS;CACT,MAAM;CACN,YAAY;CACZ,OAAO;CACP,MAAM,CAAC,eAAe,QAAQ;CAC9B,MAAM;CACP,CACF,CAEwB"}
@@ -0,0 +1,34 @@
1
+ //#region src/example.d.ts
2
+ declare const example: {
3
+ readonly id: "openbanking-powens";
4
+ readonly title: "Open Banking — Powens";
5
+ readonly summary: "OAuth callback + webhook handler patterns for Powens open banking integration (provider + workflow orchestration).";
6
+ readonly tags: readonly ["openbanking", "powens", "oauth", "webhooks", "integrations"];
7
+ readonly kind: "integration";
8
+ readonly visibility: "public";
9
+ readonly docs: {
10
+ readonly rootDocId: "docs.examples.openbanking-powens";
11
+ readonly usageDocId: "docs.examples.openbanking-powens.usage";
12
+ };
13
+ readonly entrypoints: {
14
+ readonly packageName: "@contractspec/example.openbanking-powens";
15
+ readonly docs: "./docs";
16
+ };
17
+ readonly surfaces: {
18
+ readonly templates: true;
19
+ readonly sandbox: {
20
+ readonly enabled: true;
21
+ readonly modes: readonly ["markdown", "specs"];
22
+ };
23
+ readonly studio: {
24
+ readonly enabled: true;
25
+ readonly installable: true;
26
+ };
27
+ readonly mcp: {
28
+ readonly enabled: true;
29
+ };
30
+ };
31
+ };
32
+ //#endregion
33
+ export { example as default };
34
+ //# sourceMappingURL=example.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"example.d.ts","names":[],"sources":["../src/example.ts"],"sourcesContent":[],"mappings":";cAAM;EAAA,SAAA,EAsBI,EAAA,oBAAA"}
@@ -0,0 +1,40 @@
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: "@contractspec/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 };
40
+ //# sourceMappingURL=example.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"example.js","names":[],"sources":["../src/example.ts"],"sourcesContent":["const example = {\n id: 'openbanking-powens',\n title: 'Open Banking — Powens',\n summary:\n 'OAuth callback + webhook handler patterns for Powens open banking integration (provider + workflow orchestration).',\n tags: ['openbanking', 'powens', 'oauth', 'webhooks', 'integrations'],\n kind: 'integration',\n visibility: 'public',\n docs: {\n rootDocId: 'docs.examples.openbanking-powens',\n usageDocId: 'docs.examples.openbanking-powens.usage',\n },\n entrypoints: {\n packageName: '@contractspec/example.openbanking-powens',\n docs: './docs',\n },\n surfaces: {\n templates: true,\n sandbox: { enabled: true, modes: ['markdown', 'specs'] },\n studio: { enabled: true, installable: true },\n mcp: { enabled: true },\n },\n} as const;\n\nexport default example;\n"],"mappings":";AAAA,MAAM,UAAU;CACd,IAAI;CACJ,OAAO;CACP,SACE;CACF,MAAM;EAAC;EAAe;EAAU;EAAS;EAAY;EAAe;CACpE,MAAM;CACN,YAAY;CACZ,MAAM;EACJ,WAAW;EACX,YAAY;EACb;CACD,aAAa;EACX,aAAa;EACb,MAAM;EACP;CACD,UAAU;EACR,WAAW;EACX,SAAS;GAAE,SAAS;GAAM,OAAO,CAAC,YAAY,QAAQ;GAAE;EACxD,QAAQ;GAAE,SAAS;GAAM,aAAa;GAAM;EAC5C,KAAK,EAAE,SAAS,MAAM;EACvB;CACF;AAED,sBAAe"}
@@ -0,0 +1,5 @@
1
+ //#region src/handlers/oauth-callback.d.ts
2
+ declare function powensOAuthCallbackHandler(req: Request): Promise<Response>;
3
+ //#endregion
4
+ export { powensOAuthCallbackHandler };
5
+ //# sourceMappingURL=oauth-callback.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"oauth-callback.d.ts","names":[],"sources":["../../src/handlers/oauth-callback.ts"],"sourcesContent":[],"mappings":";iBASsB,0BAAA,MAAgC,UAAO,QAAA"}
@@ -0,0 +1,64 @@
1
+ import { PowensOpenBankingProvider } from "@contractspec/integration.providers-impls/impls/powens-openbanking";
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 };
64
+ //# sourceMappingURL=oauth-callback.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"oauth-callback.js","names":["fakeSecretStore: Record<string, ExamplePowensSecrets>"],"sources":["../../src/handlers/oauth-callback.ts"],"sourcesContent":["/**\n * Example OAuth callback handler for Powens (open banking).\n *\n * This example stays framework-neutral: it operates on the standard `Request`\n * type so it can be used in Next.js, Elysia, or any fetch-compatible runtime.\n */\nimport { PowensOpenBankingProvider } from '@contractspec/integration.providers-impls/impls/powens-openbanking';\nimport type { PowensEnvironment } from '@contractspec/integration.providers-impls/impls/powens-client';\n\nexport async function powensOAuthCallbackHandler(req: Request) {\n const url = new URL(req.url);\n const code = url.searchParams.get('code');\n const state = url.searchParams.get('state');\n const userUuid = url.searchParams.get('user_uuid');\n\n if (!code || !state || !userUuid) {\n return new Response('Missing Powens OAuth params', { status: 400 });\n }\n\n const connection = await getConnectionByState(state);\n if (!connection) {\n return new Response('Unknown Powens OAuth state', { status: 404 });\n }\n\n const secrets = await getPowensSecretsForConnection(connection.meta.id);\n\n const provider = new PowensOpenBankingProvider({\n clientId: secrets.clientId,\n clientSecret: secrets.clientSecret,\n apiKey: secrets.apiKey,\n environment: connection.config.environment as PowensEnvironment,\n baseUrl: connection.config.baseUrl as string | undefined,\n });\n\n const preview = await provider.listAccounts({\n tenantId: connection.meta.tenantId,\n connectionId: connection.meta.id,\n userId: userUuid,\n });\n\n await connection.storePowensUser({\n tenantUserId: connection.meta.tenantUserId,\n powensUserUuid: userUuid,\n authCode: code,\n });\n\n await enqueueWorkflow('pfo.workflow.sync-openbanking-accounts', {\n tenantId: connection.meta.tenantId,\n userUuid,\n connectionId: connection.meta.id,\n previewAccounts: preview.accounts,\n });\n\n const redirectBase = process.env.APP_DASHBOARD_URL ?? '';\n return Response.redirect(\n `${redirectBase}/banking/linked?tenant=${connection.meta.tenantId}`,\n 302\n );\n}\n\ninterface ExamplePowensSecrets {\n clientId: string;\n clientSecret: string;\n apiKey?: string;\n}\n\ninterface ExampleIntegrationConnection {\n meta: {\n id: string;\n tenantId: string;\n tenantUserId: string;\n };\n config: {\n environment: PowensEnvironment;\n baseUrl?: string;\n };\n storePowensUser(input: {\n tenantUserId: string;\n powensUserUuid: string;\n authCode: string;\n }): Promise<void>;\n}\n\nasync function getConnectionByState(\n state: string\n): Promise<ExampleIntegrationConnection | null> {\n const record = fakeDatabase.connections.find((conn) => conn.state === state);\n return record ?? null;\n}\n\nasync function getPowensSecretsForConnection(\n connectionId: string\n): Promise<ExamplePowensSecrets> {\n const secret = fakeSecretStore[connectionId];\n if (!secret) throw new Error(`Missing Powens secrets for ${connectionId}`);\n return secret;\n}\n\nasync function enqueueWorkflow(name: string, input: Record<string, unknown>) {\n await fakeWorkflowQueue.enqueue({ name, input });\n}\n\nconst fakeDatabase = {\n connections: [] as (ExampleIntegrationConnection & { state: string })[],\n};\n\nconst fakeSecretStore: Record<string, ExamplePowensSecrets> = {};\n\nconst fakeWorkflowQueue = {\n enqueue: async (_payload: Record<string, unknown>) => {\n /* no-op */\n },\n};\n"],"mappings":";;;;;;;;;AASA,eAAsB,2BAA2B,KAAc;CAC7D,MAAM,MAAM,IAAI,IAAI,IAAI,IAAI;CAC5B,MAAM,OAAO,IAAI,aAAa,IAAI,OAAO;CACzC,MAAM,QAAQ,IAAI,aAAa,IAAI,QAAQ;CAC3C,MAAM,WAAW,IAAI,aAAa,IAAI,YAAY;AAElD,KAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,SACtB,QAAO,IAAI,SAAS,+BAA+B,EAAE,QAAQ,KAAK,CAAC;CAGrE,MAAM,aAAa,MAAM,qBAAqB,MAAM;AACpD,KAAI,CAAC,WACH,QAAO,IAAI,SAAS,8BAA8B,EAAE,QAAQ,KAAK,CAAC;CAGpE,MAAM,UAAU,MAAM,8BAA8B,WAAW,KAAK,GAAG;CAUvE,MAAM,UAAU,MARC,IAAI,0BAA0B;EAC7C,UAAU,QAAQ;EAClB,cAAc,QAAQ;EACtB,QAAQ,QAAQ;EAChB,aAAa,WAAW,OAAO;EAC/B,SAAS,WAAW,OAAO;EAC5B,CAAC,CAE6B,aAAa;EAC1C,UAAU,WAAW,KAAK;EAC1B,cAAc,WAAW,KAAK;EAC9B,QAAQ;EACT,CAAC;AAEF,OAAM,WAAW,gBAAgB;EAC/B,cAAc,WAAW,KAAK;EAC9B,gBAAgB;EAChB,UAAU;EACX,CAAC;AAEF,OAAM,gBAAgB,0CAA0C;EAC9D,UAAU,WAAW,KAAK;EAC1B;EACA,cAAc,WAAW,KAAK;EAC9B,iBAAiB,QAAQ;EAC1B,CAAC;CAEF,MAAM,eAAe,QAAQ,IAAI,qBAAqB;AACtD,QAAO,SAAS,SACd,GAAG,aAAa,yBAAyB,WAAW,KAAK,YACzD,IACD;;AA0BH,eAAe,qBACb,OAC8C;AAE9C,QADe,aAAa,YAAY,MAAM,SAAS,KAAK,UAAU,MAAM,IAC3D;;AAGnB,eAAe,8BACb,cAC+B;CAC/B,MAAM,SAAS,gBAAgB;AAC/B,KAAI,CAAC,OAAQ,OAAM,IAAI,MAAM,8BAA8B,eAAe;AAC1E,QAAO;;AAGT,eAAe,gBAAgB,MAAc,OAAgC;AAC3E,OAAM,kBAAkB,QAAQ;EAAE;EAAM;EAAO,CAAC;;AAGlD,MAAM,eAAe,EACnB,aAAa,EAAE,EAChB;AAED,MAAMA,kBAAwD,EAAE;AAEhE,MAAM,oBAAoB,EACxB,SAAS,OAAO,aAAsC,IAGvD"}
@@ -0,0 +1,5 @@
1
+ //#region src/handlers/webhook-handler.d.ts
2
+ declare function powensWebhookHandler(req: Request): Promise<Response>;
3
+ //#endregion
4
+ export { powensWebhookHandler };
5
+ //# sourceMappingURL=webhook-handler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webhook-handler.d.ts","names":[],"sources":["../../src/handlers/webhook-handler.ts"],"sourcesContent":[],"mappings":";iBAUsB,oBAAA,MAA0B,UAAO,QAAA"}
@@ -0,0 +1,88 @@
1
+ import { PowensOpenBankingProvider } from "@contractspec/integration.providers-impls/impls/powens-openbanking";
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 };
88
+ //# sourceMappingURL=webhook-handler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webhook-handler.js","names":["fakeSecretStore: Record<string, ExamplePowensSecrets>"],"sources":["../../src/handlers/webhook-handler.ts"],"sourcesContent":["/**\n * Example Powens webhook handler (fetch-compatible).\n *\n * Verifies signature, then enqueues the canonical workflows to keep the ledger\n * in sync. Unknown events are ignored (or can be recorded by the app layer).\n */\nimport { createHmac, timingSafeEqual } from 'node:crypto';\nimport { PowensOpenBankingProvider } from '@contractspec/integration.providers-impls/impls/powens-openbanking';\nimport type { PowensEnvironment } from '@contractspec/integration.providers-impls/impls/powens-client';\n\nexport async function powensWebhookHandler(req: Request) {\n const signature = req.headers.get('x-powens-signature');\n const stateHeader = req.headers.get('x-powens-state');\n const payload = await req.text();\n\n if (!signature || !stateHeader) {\n return new Response('Missing Powens signature headers', { status: 400 });\n }\n\n const connection = await getConnectionByState(stateHeader);\n if (!connection) {\n return new Response('Unknown Powens state header', { status: 404 });\n }\n\n const secrets = await getPowensSecretsForConnection(connection.meta.id);\n if (!verifySignature(payload, signature, secrets.webhookSecret)) {\n return new Response('Invalid Powens webhook signature', { status: 401 });\n }\n\n const event = JSON.parse(payload) as PowensWebhookEvent;\n const provider = new PowensOpenBankingProvider({\n clientId: secrets.clientId,\n clientSecret: secrets.clientSecret,\n apiKey: secrets.apiKey,\n environment: connection.config.environment as PowensEnvironment,\n baseUrl: connection.config.baseUrl as string | undefined,\n });\n\n switch (event.type) {\n case 'connection.updated':\n case 'user.sync.completed': {\n await enqueueWorkflow('pfo.workflow.sync-openbanking-accounts', {\n tenantId: connection.meta.tenantId,\n connectionId: connection.meta.id,\n userUuid: event.user_uuid,\n });\n break;\n }\n case 'transactions.created':\n case 'transactions.updated': {\n await enqueueWorkflow('pfo.workflow.sync-openbanking-transactions', {\n tenantId: connection.meta.tenantId,\n connectionId: connection.meta.id,\n userUuid: event.user_uuid,\n accountId: event.account_uuid,\n });\n break;\n }\n default:\n await logUnmappedEvent(event);\n }\n\n if (event.account_uuid) {\n await provider.getBalances({\n tenantId: connection.meta.tenantId,\n connectionId: connection.meta.id,\n accountId: event.account_uuid,\n });\n }\n\n return new Response('OK', { status: 200 });\n}\n\ninterface PowensWebhookEvent {\n type: string;\n user_uuid: string;\n connection_uuid: string;\n account_uuid?: string;\n}\n\ninterface ExamplePowensSecrets {\n clientId: string;\n clientSecret: string;\n apiKey?: string;\n webhookSecret: string;\n}\n\ninterface ExampleIntegrationConnection {\n meta: {\n id: string;\n tenantId: string;\n };\n config: {\n environment: PowensEnvironment;\n baseUrl?: string;\n };\n}\n\nfunction verifySignature(payload: string, signature: string, secret: string) {\n const digest = createHmac('sha256', secret).update(payload).digest('hex');\n const a = Buffer.from(digest, 'hex');\n const b = Buffer.from(signature, 'hex');\n return a.length === b.length && timingSafeEqual(a, b);\n}\n\nasync function getConnectionByState(\n state: string\n): Promise<ExampleIntegrationConnection | null> {\n return fakeDatabase.connections.find((conn) => conn.state === state) ?? null;\n}\n\nasync function getPowensSecretsForConnection(\n connectionId: string\n): Promise<ExamplePowensSecrets> {\n const secret = fakeSecretStore[connectionId];\n if (!secret) throw new Error(`Missing Powens secrets for ${connectionId}`);\n return secret;\n}\n\nasync function enqueueWorkflow(name: string, input: Record<string, unknown>) {\n await fakeWorkflowQueue.enqueue({ name, input });\n}\n\nasync function logUnmappedEvent(_event: PowensWebhookEvent) {\n await fakeTelemetryLogger.record({\n event: 'openbanking.webhook.unmapped',\n payload: 'redacted',\n });\n}\n\nconst fakeDatabase = {\n connections: [] as (ExampleIntegrationConnection & { state: string })[],\n};\n\nconst fakeSecretStore: Record<string, ExamplePowensSecrets> = {};\n\nconst fakeWorkflowQueue = {\n enqueue: async (_payload: Record<string, unknown>) => {\n /* no-op */\n },\n};\n\nconst fakeTelemetryLogger = {\n record: async (_payload: Record<string, unknown>) => {\n /* no-op */\n },\n};\n"],"mappings":";;;;;;;;;;AAUA,eAAsB,qBAAqB,KAAc;CACvD,MAAM,YAAY,IAAI,QAAQ,IAAI,qBAAqB;CACvD,MAAM,cAAc,IAAI,QAAQ,IAAI,iBAAiB;CACrD,MAAM,UAAU,MAAM,IAAI,MAAM;AAEhC,KAAI,CAAC,aAAa,CAAC,YACjB,QAAO,IAAI,SAAS,oCAAoC,EAAE,QAAQ,KAAK,CAAC;CAG1E,MAAM,aAAa,MAAM,qBAAqB,YAAY;AAC1D,KAAI,CAAC,WACH,QAAO,IAAI,SAAS,+BAA+B,EAAE,QAAQ,KAAK,CAAC;CAGrE,MAAM,UAAU,MAAM,8BAA8B,WAAW,KAAK,GAAG;AACvE,KAAI,CAAC,gBAAgB,SAAS,WAAW,QAAQ,cAAc,CAC7D,QAAO,IAAI,SAAS,oCAAoC,EAAE,QAAQ,KAAK,CAAC;CAG1E,MAAM,QAAQ,KAAK,MAAM,QAAQ;CACjC,MAAM,WAAW,IAAI,0BAA0B;EAC7C,UAAU,QAAQ;EAClB,cAAc,QAAQ;EACtB,QAAQ,QAAQ;EAChB,aAAa,WAAW,OAAO;EAC/B,SAAS,WAAW,OAAO;EAC5B,CAAC;AAEF,SAAQ,MAAM,MAAd;EACE,KAAK;EACL,KAAK;AACH,SAAM,gBAAgB,0CAA0C;IAC9D,UAAU,WAAW,KAAK;IAC1B,cAAc,WAAW,KAAK;IAC9B,UAAU,MAAM;IACjB,CAAC;AACF;EAEF,KAAK;EACL,KAAK;AACH,SAAM,gBAAgB,8CAA8C;IAClE,UAAU,WAAW,KAAK;IAC1B,cAAc,WAAW,KAAK;IAC9B,UAAU,MAAM;IAChB,WAAW,MAAM;IAClB,CAAC;AACF;EAEF,QACE,OAAM,iBAAiB,MAAM;;AAGjC,KAAI,MAAM,aACR,OAAM,SAAS,YAAY;EACzB,UAAU,WAAW,KAAK;EAC1B,cAAc,WAAW,KAAK;EAC9B,WAAW,MAAM;EAClB,CAAC;AAGJ,QAAO,IAAI,SAAS,MAAM,EAAE,QAAQ,KAAK,CAAC;;AA4B5C,SAAS,gBAAgB,SAAiB,WAAmB,QAAgB;CAC3E,MAAM,SAAS,WAAW,UAAU,OAAO,CAAC,OAAO,QAAQ,CAAC,OAAO,MAAM;CACzE,MAAM,IAAI,OAAO,KAAK,QAAQ,MAAM;CACpC,MAAM,IAAI,OAAO,KAAK,WAAW,MAAM;AACvC,QAAO,EAAE,WAAW,EAAE,UAAU,gBAAgB,GAAG,EAAE;;AAGvD,eAAe,qBACb,OAC8C;AAC9C,QAAO,aAAa,YAAY,MAAM,SAAS,KAAK,UAAU,MAAM,IAAI;;AAG1E,eAAe,8BACb,cAC+B;CAC/B,MAAM,SAAS,gBAAgB;AAC/B,KAAI,CAAC,OAAQ,OAAM,IAAI,MAAM,8BAA8B,eAAe;AAC1E,QAAO;;AAGT,eAAe,gBAAgB,MAAc,OAAgC;AAC3E,OAAM,kBAAkB,QAAQ;EAAE;EAAM;EAAO,CAAC;;AAGlD,eAAe,iBAAiB,QAA4B;AAC1D,OAAM,oBAAoB,OAAO;EAC/B,OAAO;EACP,SAAS;EACV,CAAC;;AAGJ,MAAM,eAAe,EACnB,aAAa,EAAE,EAChB;AAED,MAAMA,kBAAwD,EAAE;AAEhE,MAAM,oBAAoB,EACxB,SAAS,OAAO,aAAsC,IAGvD;AAED,MAAM,sBAAsB,EAC1B,QAAQ,OAAO,aAAsC,IAGtD"}
@@ -0,0 +1,4 @@
1
+ import example from "./example.js";
2
+ import { powensOAuthCallbackHandler } from "./handlers/oauth-callback.js";
3
+ import { powensWebhookHandler } from "./handlers/webhook-handler.js";
4
+ export { example, powensOAuthCallbackHandler, powensWebhookHandler };
package/dist/index.js ADDED
@@ -0,0 +1,6 @@
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 };