@contractspec/example.openbanking-powens 1.57.0 → 1.58.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.
- package/.turbo/turbo-build.log +23 -43
- package/.turbo/turbo-prebuild.log +1 -0
- package/CHANGELOG.md +13 -0
- package/dist/docs/index.d.ts +2 -1
- package/dist/docs/index.d.ts.map +1 -0
- package/dist/docs/index.js +39 -1
- package/dist/docs/openbanking-powens.docblock.d.ts +2 -1
- package/dist/docs/openbanking-powens.docblock.d.ts.map +1 -0
- package/dist/docs/openbanking-powens.docblock.js +36 -27
- package/dist/example.d.ts +2 -6
- package/dist/example.d.ts.map +1 -1
- package/dist/example.js +32 -44
- package/dist/handlers/oauth-callback.d.ts +1 -4
- package/dist/handlers/oauth-callback.d.ts.map +1 -1
- package/dist/handlers/oauth-callback.js +56 -55
- package/dist/handlers/webhook-handler.d.ts +1 -4
- package/dist/handlers/webhook-handler.d.ts.map +1 -1
- package/dist/handlers/webhook-handler.js +83 -75
- package/dist/index.d.ts +5 -4
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +228 -5
- package/dist/node/docs/index.js +38 -0
- package/dist/node/docs/openbanking-powens.docblock.js +38 -0
- package/dist/node/example.js +33 -0
- package/dist/node/handlers/oauth-callback.js +64 -0
- package/dist/node/handlers/webhook-handler.js +95 -0
- package/dist/node/index.js +228 -0
- package/package.json +62 -26
- package/tsdown.config.js +1 -2
- package/.turbo/turbo-build$colon$bundle.log +0 -44
- package/dist/docs/openbanking-powens.docblock.js.map +0 -1
- package/dist/example.js.map +0 -1
- package/dist/handlers/oauth-callback.js.map +0 -1
- package/dist/handlers/webhook-handler.js.map +0 -1
- package/tsconfig.tsbuildinfo +0 -1
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
// src/docs/openbanking-powens.docblock.ts
|
|
2
|
+
import { registerDocBlocks } from "@contractspec/lib.contracts/docs";
|
|
3
|
+
var blocks = [
|
|
4
|
+
{
|
|
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: ["openbanking", "powens", "integration", "example"],
|
|
12
|
+
body: `## What this example shows
|
|
13
|
+
- OAuth callback handler: exchange auth code, map powens user, enqueue sync workflow.
|
|
14
|
+
- Webhook handler: verify signature, route event → workflow, optionally refresh balances.
|
|
15
|
+
|
|
16
|
+
## Guardrails
|
|
17
|
+
- Secrets via secret providers/env only.
|
|
18
|
+
- Verify webhook signatures.
|
|
19
|
+
- Keep side effects explicit: enqueue workflows instead of mutating canonical stores inline.`
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
id: "docs.examples.openbanking-powens.usage",
|
|
23
|
+
title: "Open Banking — Powens — Usage",
|
|
24
|
+
summary: "How to integrate the handlers in a fetch-compatible runtime.",
|
|
25
|
+
kind: "usage",
|
|
26
|
+
visibility: "public",
|
|
27
|
+
route: "/docs/examples/openbanking-powens/usage",
|
|
28
|
+
tags: ["openbanking", "usage"],
|
|
29
|
+
body: `## Usage
|
|
30
|
+
- Wire \`powensOAuthCallbackHandler(req)\` at your OAuth redirect route.
|
|
31
|
+
- Wire \`powensWebhookHandler(req)\` at your webhook route.
|
|
32
|
+
|
|
33
|
+
## Notes
|
|
34
|
+
- Replace the fake stores with your app-layer persistence.
|
|
35
|
+
- Enqueue ContractSpec workflows for canonical upserts and telemetry.`
|
|
36
|
+
}
|
|
37
|
+
];
|
|
38
|
+
registerDocBlocks(blocks);
|
|
39
|
+
// src/example.ts
|
|
40
|
+
import { defineExample } from "@contractspec/lib.contracts";
|
|
41
|
+
var example = defineExample({
|
|
42
|
+
meta: {
|
|
43
|
+
key: "openbanking-powens",
|
|
44
|
+
version: "1.0.0",
|
|
45
|
+
title: "Open Banking — Powens",
|
|
46
|
+
description: "OAuth callback + webhook handler patterns for Powens open banking integration (provider + workflow orchestration).",
|
|
47
|
+
kind: "integration",
|
|
48
|
+
visibility: "public",
|
|
49
|
+
stability: "experimental",
|
|
50
|
+
owners: ["@platform.core"],
|
|
51
|
+
tags: ["openbanking", "powens", "oauth", "webhooks", "integrations"]
|
|
52
|
+
},
|
|
53
|
+
docs: {
|
|
54
|
+
rootDocId: "docs.examples.openbanking-powens",
|
|
55
|
+
usageDocId: "docs.examples.openbanking-powens.usage"
|
|
56
|
+
},
|
|
57
|
+
entrypoints: {
|
|
58
|
+
packageName: "@contractspec/example.openbanking-powens",
|
|
59
|
+
docs: "./docs"
|
|
60
|
+
},
|
|
61
|
+
surfaces: {
|
|
62
|
+
templates: true,
|
|
63
|
+
sandbox: { enabled: true, modes: ["markdown", "specs"] },
|
|
64
|
+
studio: { enabled: true, installable: true },
|
|
65
|
+
mcp: { enabled: true }
|
|
66
|
+
}
|
|
67
|
+
});
|
|
68
|
+
var example_default = example;
|
|
69
|
+
|
|
70
|
+
// src/handlers/oauth-callback.ts
|
|
71
|
+
import { PowensOpenBankingProvider } from "@contractspec/integration.providers-impls/impls/powens-openbanking";
|
|
72
|
+
async function powensOAuthCallbackHandler(req) {
|
|
73
|
+
const url = new URL(req.url);
|
|
74
|
+
const code = url.searchParams.get("code");
|
|
75
|
+
const state = url.searchParams.get("state");
|
|
76
|
+
const userUuid = url.searchParams.get("user_uuid");
|
|
77
|
+
if (!code || !state || !userUuid) {
|
|
78
|
+
return new Response("Missing Powens OAuth params", { status: 400 });
|
|
79
|
+
}
|
|
80
|
+
const connection = await getConnectionByState(state);
|
|
81
|
+
if (!connection) {
|
|
82
|
+
return new Response("Unknown Powens OAuth state", { status: 404 });
|
|
83
|
+
}
|
|
84
|
+
const secrets = await getPowensSecretsForConnection(connection.meta.id);
|
|
85
|
+
const provider = new PowensOpenBankingProvider({
|
|
86
|
+
clientId: secrets.clientId,
|
|
87
|
+
clientSecret: secrets.clientSecret,
|
|
88
|
+
apiKey: secrets.apiKey,
|
|
89
|
+
environment: connection.config.environment,
|
|
90
|
+
baseUrl: connection.config.baseUrl
|
|
91
|
+
});
|
|
92
|
+
const preview = await provider.listAccounts({
|
|
93
|
+
tenantId: connection.meta.tenantId,
|
|
94
|
+
connectionId: connection.meta.id,
|
|
95
|
+
userId: userUuid
|
|
96
|
+
});
|
|
97
|
+
await connection.storePowensUser({
|
|
98
|
+
tenantUserId: connection.meta.tenantUserId,
|
|
99
|
+
powensUserUuid: userUuid,
|
|
100
|
+
authCode: code
|
|
101
|
+
});
|
|
102
|
+
await enqueueWorkflow("pfo.workflow.sync-openbanking-accounts", {
|
|
103
|
+
tenantId: connection.meta.tenantId,
|
|
104
|
+
userUuid,
|
|
105
|
+
connectionId: connection.meta.id,
|
|
106
|
+
previewAccounts: preview.accounts
|
|
107
|
+
});
|
|
108
|
+
const redirectBase = process.env.APP_DASHBOARD_URL ?? "";
|
|
109
|
+
return Response.redirect(`${redirectBase}/banking/linked?tenant=${connection.meta.tenantId}`, 302);
|
|
110
|
+
}
|
|
111
|
+
async function getConnectionByState(state) {
|
|
112
|
+
const record = fakeDatabase.connections.find((conn) => conn.state === state);
|
|
113
|
+
return record ?? null;
|
|
114
|
+
}
|
|
115
|
+
async function getPowensSecretsForConnection(connectionId) {
|
|
116
|
+
const secret = fakeSecretStore[connectionId];
|
|
117
|
+
if (!secret)
|
|
118
|
+
throw new Error(`Missing Powens secrets for ${connectionId}`);
|
|
119
|
+
return secret;
|
|
120
|
+
}
|
|
121
|
+
async function enqueueWorkflow(name, input) {
|
|
122
|
+
await fakeWorkflowQueue.enqueue({ name, input });
|
|
123
|
+
}
|
|
124
|
+
var fakeDatabase = {
|
|
125
|
+
connections: []
|
|
126
|
+
};
|
|
127
|
+
var fakeSecretStore = {};
|
|
128
|
+
var fakeWorkflowQueue = {
|
|
129
|
+
enqueue: async (_payload) => {}
|
|
130
|
+
};
|
|
131
|
+
|
|
132
|
+
// src/handlers/webhook-handler.ts
|
|
133
|
+
import { createHmac, timingSafeEqual } from "crypto";
|
|
134
|
+
import { PowensOpenBankingProvider as PowensOpenBankingProvider2 } from "@contractspec/integration.providers-impls/impls/powens-openbanking";
|
|
135
|
+
async function powensWebhookHandler(req) {
|
|
136
|
+
const signature = req.headers.get("x-powens-signature");
|
|
137
|
+
const stateHeader = req.headers.get("x-powens-state");
|
|
138
|
+
const payload = await req.text();
|
|
139
|
+
if (!signature || !stateHeader) {
|
|
140
|
+
return new Response("Missing Powens signature headers", { status: 400 });
|
|
141
|
+
}
|
|
142
|
+
const connection = await getConnectionByState2(stateHeader);
|
|
143
|
+
if (!connection) {
|
|
144
|
+
return new Response("Unknown Powens state header", { status: 404 });
|
|
145
|
+
}
|
|
146
|
+
const secrets = await getPowensSecretsForConnection2(connection.meta.id);
|
|
147
|
+
if (!verifySignature(payload, signature, secrets.webhookSecret)) {
|
|
148
|
+
return new Response("Invalid Powens webhook signature", { status: 401 });
|
|
149
|
+
}
|
|
150
|
+
const event = JSON.parse(payload);
|
|
151
|
+
const provider = new PowensOpenBankingProvider2({
|
|
152
|
+
clientId: secrets.clientId,
|
|
153
|
+
clientSecret: secrets.clientSecret,
|
|
154
|
+
apiKey: secrets.apiKey,
|
|
155
|
+
environment: connection.config.environment,
|
|
156
|
+
baseUrl: connection.config.baseUrl
|
|
157
|
+
});
|
|
158
|
+
switch (event.type) {
|
|
159
|
+
case "connection.updated":
|
|
160
|
+
case "user.sync.completed": {
|
|
161
|
+
await enqueueWorkflow2("pfo.workflow.sync-openbanking-accounts", {
|
|
162
|
+
tenantId: connection.meta.tenantId,
|
|
163
|
+
connectionId: connection.meta.id,
|
|
164
|
+
userUuid: event.user_uuid
|
|
165
|
+
});
|
|
166
|
+
break;
|
|
167
|
+
}
|
|
168
|
+
case "transactions.created":
|
|
169
|
+
case "transactions.updated": {
|
|
170
|
+
await enqueueWorkflow2("pfo.workflow.sync-openbanking-transactions", {
|
|
171
|
+
tenantId: connection.meta.tenantId,
|
|
172
|
+
connectionId: connection.meta.id,
|
|
173
|
+
userUuid: event.user_uuid,
|
|
174
|
+
accountId: event.account_uuid
|
|
175
|
+
});
|
|
176
|
+
break;
|
|
177
|
+
}
|
|
178
|
+
default:
|
|
179
|
+
await logUnmappedEvent(event);
|
|
180
|
+
}
|
|
181
|
+
if (event.account_uuid) {
|
|
182
|
+
await provider.getBalances({
|
|
183
|
+
tenantId: connection.meta.tenantId,
|
|
184
|
+
connectionId: connection.meta.id,
|
|
185
|
+
accountId: event.account_uuid
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
return new Response("OK", { status: 200 });
|
|
189
|
+
}
|
|
190
|
+
function verifySignature(payload, signature, secret) {
|
|
191
|
+
const digest = createHmac("sha256", secret).update(payload).digest("hex");
|
|
192
|
+
const a = Buffer.from(digest, "hex");
|
|
193
|
+
const b = Buffer.from(signature, "hex");
|
|
194
|
+
return a.length === b.length && timingSafeEqual(a, b);
|
|
195
|
+
}
|
|
196
|
+
async function getConnectionByState2(state) {
|
|
197
|
+
return fakeDatabase2.connections.find((conn) => conn.state === state) ?? null;
|
|
198
|
+
}
|
|
199
|
+
async function getPowensSecretsForConnection2(connectionId) {
|
|
200
|
+
const secret = fakeSecretStore2[connectionId];
|
|
201
|
+
if (!secret)
|
|
202
|
+
throw new Error(`Missing Powens secrets for ${connectionId}`);
|
|
203
|
+
return secret;
|
|
204
|
+
}
|
|
205
|
+
async function enqueueWorkflow2(name, input) {
|
|
206
|
+
await fakeWorkflowQueue2.enqueue({ name, input });
|
|
207
|
+
}
|
|
208
|
+
async function logUnmappedEvent(_event) {
|
|
209
|
+
await fakeTelemetryLogger.record({
|
|
210
|
+
event: "openbanking.webhook.unmapped",
|
|
211
|
+
payload: "redacted"
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
var fakeDatabase2 = {
|
|
215
|
+
connections: []
|
|
216
|
+
};
|
|
217
|
+
var fakeSecretStore2 = {};
|
|
218
|
+
var fakeWorkflowQueue2 = {
|
|
219
|
+
enqueue: async (_payload) => {}
|
|
220
|
+
};
|
|
221
|
+
var fakeTelemetryLogger = {
|
|
222
|
+
record: async (_payload) => {}
|
|
223
|
+
};
|
|
224
|
+
export {
|
|
225
|
+
powensWebhookHandler,
|
|
226
|
+
powensOAuthCallbackHandler,
|
|
227
|
+
example_default as example
|
|
228
|
+
};
|
package/package.json
CHANGED
|
@@ -1,51 +1,87 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@contractspec/example.openbanking-powens",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.58.0",
|
|
4
4
|
"description": "OpenBanking Powens example: OAuth callback + webhook handler patterns (provider + workflows).",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"types": "./dist/index.d.ts",
|
|
7
7
|
"exports": {
|
|
8
|
-
".": "./
|
|
9
|
-
"./docs": "./
|
|
10
|
-
"./docs/
|
|
11
|
-
"./
|
|
12
|
-
"./
|
|
13
|
-
"./handlers/
|
|
14
|
-
"
|
|
8
|
+
".": "./src/index.ts",
|
|
9
|
+
"./docs": "./src/docs/index.ts",
|
|
10
|
+
"./docs/index": "./src/docs/index.ts",
|
|
11
|
+
"./docs/openbanking-powens.docblock": "./src/docs/openbanking-powens.docblock.ts",
|
|
12
|
+
"./example": "./src/example.ts",
|
|
13
|
+
"./handlers/oauth-callback": "./src/handlers/oauth-callback.ts",
|
|
14
|
+
"./handlers/webhook-handler": "./src/handlers/webhook-handler.ts"
|
|
15
15
|
},
|
|
16
16
|
"scripts": {
|
|
17
17
|
"publish:pkg": "bun publish --tolerate-republish --ignore-scripts --verbose",
|
|
18
18
|
"publish:pkg:canary": "bun publish:pkg --tag canary",
|
|
19
|
-
"build": "bun build:
|
|
20
|
-
"build:bundle": "
|
|
21
|
-
"build:types": "
|
|
22
|
-
"dev": "bun
|
|
19
|
+
"build": "bun run prebuild && bun run build:bundle && bun run build:types",
|
|
20
|
+
"build:bundle": "contractspec-bun-build transpile",
|
|
21
|
+
"build:types": "contractspec-bun-build types",
|
|
22
|
+
"dev": "contractspec-bun-build dev",
|
|
23
23
|
"clean": "rimraf dist .turbo",
|
|
24
24
|
"lint": "bun lint:fix",
|
|
25
25
|
"lint:fix": "eslint src --fix",
|
|
26
26
|
"lint:check": "eslint src",
|
|
27
|
-
"test": "bun test"
|
|
27
|
+
"test": "bun test",
|
|
28
|
+
"prebuild": "contractspec-bun-build prebuild",
|
|
29
|
+
"typecheck": "tsc --noEmit"
|
|
28
30
|
},
|
|
29
31
|
"dependencies": {
|
|
30
|
-
"@contractspec/integration.providers-impls": "1.
|
|
31
|
-
"@contractspec/lib.contracts": "1.
|
|
32
|
+
"@contractspec/integration.providers-impls": "1.58.0",
|
|
33
|
+
"@contractspec/lib.contracts": "1.58.0"
|
|
32
34
|
},
|
|
33
35
|
"devDependencies": {
|
|
34
|
-
"@contractspec/tool.
|
|
35
|
-
"
|
|
36
|
-
"
|
|
37
|
-
"typescript": "^5.9.3"
|
|
36
|
+
"@contractspec/tool.typescript": "1.58.0",
|
|
37
|
+
"typescript": "^5.9.3",
|
|
38
|
+
"@contractspec/tool.bun": "1.57.0"
|
|
38
39
|
},
|
|
39
40
|
"publishConfig": {
|
|
40
41
|
"access": "public",
|
|
41
42
|
"exports": {
|
|
42
|
-
".":
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
"
|
|
43
|
+
".": {
|
|
44
|
+
"types": "./dist/index.d.ts",
|
|
45
|
+
"bun": "./dist/index.js",
|
|
46
|
+
"node": "./dist/node/index.mjs",
|
|
47
|
+
"default": "./dist/index.js"
|
|
48
|
+
},
|
|
49
|
+
"./docs": {
|
|
50
|
+
"types": "./dist/docs/index.d.ts",
|
|
51
|
+
"bun": "./dist/docs/index.js",
|
|
52
|
+
"node": "./dist/node/docs/index.mjs",
|
|
53
|
+
"default": "./dist/docs/index.js"
|
|
54
|
+
},
|
|
55
|
+
"./docs/index": {
|
|
56
|
+
"types": "./dist/docs/index.d.ts",
|
|
57
|
+
"bun": "./dist/docs/index.js",
|
|
58
|
+
"node": "./dist/node/docs/index.mjs",
|
|
59
|
+
"default": "./dist/docs/index.js"
|
|
60
|
+
},
|
|
61
|
+
"./docs/openbanking-powens.docblock": {
|
|
62
|
+
"types": "./dist/docs/openbanking-powens.docblock.d.ts",
|
|
63
|
+
"bun": "./dist/docs/openbanking-powens.docblock.js",
|
|
64
|
+
"node": "./dist/node/docs/openbanking-powens.docblock.mjs",
|
|
65
|
+
"default": "./dist/docs/openbanking-powens.docblock.js"
|
|
66
|
+
},
|
|
67
|
+
"./example": {
|
|
68
|
+
"types": "./dist/example.d.ts",
|
|
69
|
+
"bun": "./dist/example.js",
|
|
70
|
+
"node": "./dist/node/example.mjs",
|
|
71
|
+
"default": "./dist/example.js"
|
|
72
|
+
},
|
|
73
|
+
"./handlers/oauth-callback": {
|
|
74
|
+
"types": "./dist/handlers/oauth-callback.d.ts",
|
|
75
|
+
"bun": "./dist/handlers/oauth-callback.js",
|
|
76
|
+
"node": "./dist/node/handlers/oauth-callback.mjs",
|
|
77
|
+
"default": "./dist/handlers/oauth-callback.js"
|
|
78
|
+
},
|
|
79
|
+
"./handlers/webhook-handler": {
|
|
80
|
+
"types": "./dist/handlers/webhook-handler.d.ts",
|
|
81
|
+
"bun": "./dist/handlers/webhook-handler.js",
|
|
82
|
+
"node": "./dist/node/handlers/webhook-handler.mjs",
|
|
83
|
+
"default": "./dist/handlers/webhook-handler.js"
|
|
84
|
+
}
|
|
49
85
|
},
|
|
50
86
|
"registry": "https://registry.npmjs.org/"
|
|
51
87
|
},
|
package/tsdown.config.js
CHANGED
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
$ tsdown
|
|
2
|
-
[34mℹ[39m tsdown [2mv0.20.3[22m powered by rolldown [2mv1.0.0-rc.3[22m
|
|
3
|
-
[34mℹ[39m config file: [4m/home/runner/work/contractspec/contractspec/packages/examples/openbanking-powens/tsdown.config.js[24m
|
|
4
|
-
[34mℹ[39m entry: [34msrc/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[39m
|
|
5
|
-
[34mℹ[39m target: [34mesnext[39m
|
|
6
|
-
[34mℹ[39m tsconfig: [34mtsconfig.json[39m
|
|
7
|
-
[34mℹ[39m Build start
|
|
8
|
-
[34mℹ[39m Cleaning 21 files
|
|
9
|
-
[34mℹ[39m [2mdist/[22m[1mhandlers/webhook-handler.js[22m [2m3.32 kB[22m [2m│ gzip: 1.23 kB[22m
|
|
10
|
-
[34mℹ[39m [2mdist/[22m[1mhandlers/oauth-callback.js[22m [2m2.42 kB[22m [2m│ gzip: 1.02 kB[22m
|
|
11
|
-
[34mℹ[39m [2mdist/[22m[1mdocs/openbanking-powens.docblock.js[22m [2m1.57 kB[22m [2m│ gzip: 0.76 kB[22m
|
|
12
|
-
[34mℹ[39m [2mdist/[22m[1mexample.js[22m [2m1.03 kB[22m [2m│ gzip: 0.54 kB[22m
|
|
13
|
-
[34mℹ[39m [2mdist/[22m[1mindex.js[22m [2m0.28 kB[22m [2m│ gzip: 0.15 kB[22m
|
|
14
|
-
[34mℹ[39m [2mdist/[22m[1mdocs/index.js[22m [2m0.04 kB[22m [2m│ gzip: 0.06 kB[22m
|
|
15
|
-
[34mℹ[39m [2mdist/[22mhandlers/webhook-handler.js.map [2m6.40 kB[22m [2m│ gzip: 2.17 kB[22m
|
|
16
|
-
[34mℹ[39m [2mdist/[22mhandlers/oauth-callback.js.map [2m4.79 kB[22m [2m│ gzip: 1.80 kB[22m
|
|
17
|
-
[34mℹ[39m [2mdist/[22mdocs/openbanking-powens.docblock.js.map [2m2.07 kB[22m [2m│ gzip: 0.94 kB[22m
|
|
18
|
-
[34mℹ[39m [2mdist/[22mexample.js.map [2m1.50 kB[22m [2m│ gzip: 0.74 kB[22m
|
|
19
|
-
[34mℹ[39m [2mdist/[22mhandlers/webhook-handler.d.ts.map [2m0.17 kB[22m [2m│ gzip: 0.15 kB[22m
|
|
20
|
-
[34mℹ[39m [2mdist/[22mhandlers/oauth-callback.d.ts.map [2m0.17 kB[22m [2m│ gzip: 0.15 kB[22m
|
|
21
|
-
[34mℹ[39m [2mdist/[22mexample.d.ts.map [2m0.13 kB[22m [2m│ gzip: 0.13 kB[22m
|
|
22
|
-
[34mℹ[39m [2mdist/[22m[32m[1mexample.d.ts[22m[39m [2m0.25 kB[22m [2m│ gzip: 0.17 kB[22m
|
|
23
|
-
[34mℹ[39m [2mdist/[22m[32m[1mindex.d.ts[22m[39m [2m0.25 kB[22m [2m│ gzip: 0.13 kB[22m
|
|
24
|
-
[34mℹ[39m [2mdist/[22m[32m[1mhandlers/oauth-callback.d.ts[22m[39m [2m0.22 kB[22m [2m│ gzip: 0.17 kB[22m
|
|
25
|
-
[34mℹ[39m [2mdist/[22m[32m[1mhandlers/webhook-handler.d.ts[22m[39m [2m0.21 kB[22m [2m│ gzip: 0.16 kB[22m
|
|
26
|
-
[34mℹ[39m [2mdist/[22m[32m[1mdocs/index.d.ts[22m[39m [2m0.01 kB[22m [2m│ gzip: 0.03 kB[22m
|
|
27
|
-
[34mℹ[39m [2mdist/[22m[32m[1mdocs/openbanking-powens.docblock.d.ts[22m[39m [2m0.01 kB[22m [2m│ gzip: 0.03 kB[22m
|
|
28
|
-
[34mℹ[39m 19 files, total: 24.84 kB
|
|
29
|
-
src/handlers/webhook-handler.ts (7:44) [33m[UNRESOLVED_IMPORT] Warning:[0m Could not resolve 'crypto' in src/handlers/webhook-handler.ts
|
|
30
|
-
[38;5;246m╭[0m[38;5;246m─[0m[38;5;246m[[0m src/handlers/webhook-handler.ts:7:45 [38;5;246m][0m
|
|
31
|
-
[38;5;246m│[0m
|
|
32
|
-
[38;5;246m7 │[0m [38;5;249mi[0m[38;5;249mm[0m[38;5;249mp[0m[38;5;249mo[0m[38;5;249mr[0m[38;5;249mt[0m[38;5;249m [0m[38;5;249m{[0m[38;5;249m [0m[38;5;249mc[0m[38;5;249mr[0m[38;5;249me[0m[38;5;249ma[0m[38;5;249mt[0m[38;5;249me[0m[38;5;249mH[0m[38;5;249mm[0m[38;5;249ma[0m[38;5;249mc[0m[38;5;249m,[0m[38;5;249m [0m[38;5;249mt[0m[38;5;249mi[0m[38;5;249mm[0m[38;5;249mi[0m[38;5;249mn[0m[38;5;249mg[0m[38;5;249mS[0m[38;5;249ma[0m[38;5;249mf[0m[38;5;249me[0m[38;5;249mE[0m[38;5;249mq[0m[38;5;249mu[0m[38;5;249ma[0m[38;5;249ml[0m[38;5;249m [0m[38;5;249m}[0m[38;5;249m [0m[38;5;249mf[0m[38;5;249mr[0m[38;5;249mo[0m[38;5;249mm[0m[38;5;249m [0m'crypto'[38;5;249m;[0m
|
|
33
|
-
[38;5;240m │[0m ────┬───
|
|
34
|
-
[38;5;240m │[0m ╰───── Module not found, treating it as an external dependency
|
|
35
|
-
[38;5;240m │[0m
|
|
36
|
-
[38;5;240m │[0m [38;5;115mHelp[0m: The "main" field here was ignored. Main fields must be configured explicitly when using the "neutral" platform.
|
|
37
|
-
[38;5;246m───╯[0m
|
|
38
|
-
|
|
39
|
-
[33m[PLUGIN_TIMINGS] Warning:[0m Your build spent significant time in plugins. Here is a breakdown:
|
|
40
|
-
- tsdown:external (62%)
|
|
41
|
-
- rolldown-plugin-dts:generate (37%)
|
|
42
|
-
See https://rolldown.rs/options/checks#plugintimings for more details.
|
|
43
|
-
|
|
44
|
-
[32m✔[39m Build complete in [32m14025ms[39m
|
|
@@ -1 +0,0 @@
|
|
|
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"}
|
package/dist/example.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"example.js","names":[],"sources":["../src/example.ts"],"sourcesContent":["import { defineExample } from '@contractspec/lib.contracts';\n\nconst example = defineExample({\n meta: {\n key: 'openbanking-powens',\n version: '1.0.0',\n title: 'Open Banking — Powens',\n description:\n 'OAuth callback + webhook handler patterns for Powens open banking integration (provider + workflow orchestration).',\n kind: 'integration',\n visibility: 'public',\n stability: 'experimental',\n owners: ['@platform.core'],\n tags: ['openbanking', 'powens', 'oauth', 'webhooks', 'integrations'],\n },\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});\n\nexport default example;\n"],"mappings":";;;AAEA,MAAM,UAAU,cAAc;CAC5B,MAAM;EACJ,KAAK;EACL,SAAS;EACT,OAAO;EACP,aACE;EACF,MAAM;EACN,YAAY;EACZ,WAAW;EACX,QAAQ,CAAC,iBAAiB;EAC1B,MAAM;GAAC;GAAe;GAAU;GAAS;GAAY;GAAe;EACrE;CACD,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,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"oauth-callback.js","names":[],"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,MAAM,kBAAwD,EAAE;AAEhE,MAAM,oBAAoB,EACxB,SAAS,OAAO,aAAsC,IAGvD"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"webhook-handler.js","names":[],"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 '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,MAAM,kBAAwD,EAAE;AAEhE,MAAM,oBAAoB,EACxB,SAAS,OAAO,aAAsC,IAGvD;AAED,MAAM,sBAAsB,EAC1B,QAAQ,OAAO,aAAsC,IAGtD"}
|