@betterportal/framework 0.0.1
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/README.md +13 -0
- package/lib/adapters/h3.d.ts +51 -0
- package/lib/adapters/h3.d.ts.map +1 -0
- package/lib/adapters/h3.js +1120 -0
- package/lib/adapters/h3.js.map +1 -0
- package/lib/codegen/cli.d.ts +3 -0
- package/lib/codegen/cli.d.ts.map +1 -0
- package/lib/codegen/cli.js +82 -0
- package/lib/codegen/cli.js.map +1 -0
- package/lib/codegen/emitter.d.ts +7 -0
- package/lib/codegen/emitter.d.ts.map +1 -0
- package/lib/codegen/emitter.js +453 -0
- package/lib/codegen/emitter.js.map +1 -0
- package/lib/codegen/init.d.ts +3 -0
- package/lib/codegen/init.d.ts.map +1 -0
- package/lib/codegen/init.js +90 -0
- package/lib/codegen/init.js.map +1 -0
- package/lib/codegen/scanner.d.ts +56 -0
- package/lib/codegen/scanner.d.ts.map +1 -0
- package/lib/codegen/scanner.js +484 -0
- package/lib/codegen/scanner.js.map +1 -0
- package/lib/codegen/validate.d.ts +14 -0
- package/lib/codegen/validate.d.ts.map +1 -0
- package/lib/codegen/validate.js +166 -0
- package/lib/codegen/validate.js.map +1 -0
- package/lib/contracts/auth.d.ts +160 -0
- package/lib/contracts/auth.d.ts.map +1 -0
- package/lib/contracts/auth.js +123 -0
- package/lib/contracts/auth.js.map +1 -0
- package/lib/contracts/binding.d.ts +169 -0
- package/lib/contracts/binding.d.ts.map +1 -0
- package/lib/contracts/binding.js +69 -0
- package/lib/contracts/binding.js.map +1 -0
- package/lib/contracts/common.d.ts +23 -0
- package/lib/contracts/common.d.ts.map +1 -0
- package/lib/contracts/common.js +18 -0
- package/lib/contracts/common.js.map +1 -0
- package/lib/contracts/config.d.ts +93 -0
- package/lib/contracts/config.d.ts.map +1 -0
- package/lib/contracts/config.js +62 -0
- package/lib/contracts/config.js.map +1 -0
- package/lib/contracts/controlPlane.d.ts +63 -0
- package/lib/contracts/controlPlane.d.ts.map +1 -0
- package/lib/contracts/controlPlane.js +2 -0
- package/lib/contracts/controlPlane.js.map +1 -0
- package/lib/contracts/json.d.ts +9 -0
- package/lib/contracts/json.d.ts.map +1 -0
- package/lib/contracts/json.js +6 -0
- package/lib/contracts/json.js.map +1 -0
- package/lib/contracts/manifest.d.ts +158 -0
- package/lib/contracts/manifest.d.ts.map +1 -0
- package/lib/contracts/manifest.js +40 -0
- package/lib/contracts/manifest.js.map +1 -0
- package/lib/contracts/observability.d.ts +77 -0
- package/lib/contracts/observability.d.ts.map +1 -0
- package/lib/contracts/observability.js +99 -0
- package/lib/contracts/observability.js.map +1 -0
- package/lib/contracts/platformConfig.d.ts +635 -0
- package/lib/contracts/platformConfig.d.ts.map +1 -0
- package/lib/contracts/platformConfig.js +256 -0
- package/lib/contracts/platformConfig.js.map +1 -0
- package/lib/contracts/registry.d.ts +104 -0
- package/lib/contracts/registry.d.ts.map +1 -0
- package/lib/contracts/registry.js +2 -0
- package/lib/contracts/registry.js.map +1 -0
- package/lib/contracts/route.d.ts +199 -0
- package/lib/contracts/route.d.ts.map +1 -0
- package/lib/contracts/route.js +26 -0
- package/lib/contracts/route.js.map +1 -0
- package/lib/contracts/serviceConfig.d.ts +88 -0
- package/lib/contracts/serviceConfig.d.ts.map +1 -0
- package/lib/contracts/serviceConfig.js +45 -0
- package/lib/contracts/serviceConfig.js.map +1 -0
- package/lib/contracts/streaming.d.ts +76 -0
- package/lib/contracts/streaming.d.ts.map +1 -0
- package/lib/contracts/streaming.js +31 -0
- package/lib/contracts/streaming.js.map +1 -0
- package/lib/contracts/view.d.ts +149 -0
- package/lib/contracts/view.d.ts.map +1 -0
- package/lib/contracts/view.js +82 -0
- package/lib/contracts/view.js.map +1 -0
- package/lib/controlPlane/store.d.ts +24 -0
- package/lib/controlPlane/store.d.ts.map +1 -0
- package/lib/controlPlane/store.js +70 -0
- package/lib/controlPlane/store.js.map +1 -0
- package/lib/controlPlane/sync.d.ts +8 -0
- package/lib/controlPlane/sync.d.ts.map +1 -0
- package/lib/controlPlane/sync.js +24 -0
- package/lib/controlPlane/sync.js.map +1 -0
- package/lib/controlPlane/types.d.ts +15 -0
- package/lib/controlPlane/types.d.ts.map +1 -0
- package/lib/controlPlane/types.js +2 -0
- package/lib/controlPlane/types.js.map +1 -0
- package/lib/index.d.ts +40 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +40 -0
- package/lib/index.js.map +1 -0
- package/lib/runtime/auth/envelope.d.ts +32 -0
- package/lib/runtime/auth/envelope.d.ts.map +1 -0
- package/lib/runtime/auth/envelope.js +123 -0
- package/lib/runtime/auth/envelope.js.map +1 -0
- package/lib/runtime/auth/issuer.d.ts +44 -0
- package/lib/runtime/auth/issuer.d.ts.map +1 -0
- package/lib/runtime/auth/issuer.js +82 -0
- package/lib/runtime/auth/issuer.js.map +1 -0
- package/lib/runtime/auth/jwks.d.ts +7 -0
- package/lib/runtime/auth/jwks.d.ts.map +1 -0
- package/lib/runtime/auth/jwks.js +69 -0
- package/lib/runtime/auth/jwks.js.map +1 -0
- package/lib/runtime/auth/keypair.d.ts +21 -0
- package/lib/runtime/auth/keypair.d.ts.map +1 -0
- package/lib/runtime/auth/keypair.js +50 -0
- package/lib/runtime/auth/keypair.js.map +1 -0
- package/lib/runtime/auth/tokens.d.ts +25 -0
- package/lib/runtime/auth/tokens.d.ts.map +1 -0
- package/lib/runtime/auth/tokens.js +137 -0
- package/lib/runtime/auth/tokens.js.map +1 -0
- package/lib/runtime/auth/verifier.d.ts +45 -0
- package/lib/runtime/auth/verifier.d.ts.map +1 -0
- package/lib/runtime/auth/verifier.js +76 -0
- package/lib/runtime/auth/verifier.js.map +1 -0
- package/lib/runtime/bpHeaders.d.ts +10 -0
- package/lib/runtime/bpHeaders.d.ts.map +1 -0
- package/lib/runtime/bpHeaders.js +53 -0
- package/lib/runtime/bpHeaders.js.map +1 -0
- package/lib/runtime/configProvider.d.ts +41 -0
- package/lib/runtime/configProvider.d.ts.map +1 -0
- package/lib/runtime/configProvider.js +232 -0
- package/lib/runtime/configProvider.js.map +1 -0
- package/lib/runtime/configStore.d.ts +34 -0
- package/lib/runtime/configStore.d.ts.map +1 -0
- package/lib/runtime/configStore.js +197 -0
- package/lib/runtime/configStore.js.map +1 -0
- package/lib/runtime/configTicket.d.ts +49 -0
- package/lib/runtime/configTicket.d.ts.map +1 -0
- package/lib/runtime/configTicket.js +168 -0
- package/lib/runtime/configTicket.js.map +1 -0
- package/lib/runtime/h3.d.ts +28 -0
- package/lib/runtime/h3.d.ts.map +1 -0
- package/lib/runtime/h3.js +199 -0
- package/lib/runtime/h3.js.map +1 -0
- package/lib/runtime/handler.d.ts +55 -0
- package/lib/runtime/handler.d.ts.map +1 -0
- package/lib/runtime/handler.js +51 -0
- package/lib/runtime/handler.js.map +1 -0
- package/lib/runtime/http.d.ts +13 -0
- package/lib/runtime/http.d.ts.map +1 -0
- package/lib/runtime/http.js +114 -0
- package/lib/runtime/http.js.map +1 -0
- package/lib/runtime/jsonSchema.d.ts +4 -0
- package/lib/runtime/jsonSchema.d.ts.map +1 -0
- package/lib/runtime/jsonSchema.js +28 -0
- package/lib/runtime/jsonSchema.js.map +1 -0
- package/lib/runtime/manifest.d.ts +3 -0
- package/lib/runtime/manifest.d.ts.map +1 -0
- package/lib/runtime/manifest.js +5 -0
- package/lib/runtime/manifest.js.map +1 -0
- package/lib/runtime/media.d.ts +20 -0
- package/lib/runtime/media.d.ts.map +1 -0
- package/lib/runtime/media.js +70 -0
- package/lib/runtime/media.js.map +1 -0
- package/lib/runtime/registry.d.ts +67 -0
- package/lib/runtime/registry.d.ts.map +1 -0
- package/lib/runtime/registry.js +290 -0
- package/lib/runtime/registry.js.map +1 -0
- package/lib/runtime/serviceConfig.d.ts +38 -0
- package/lib/runtime/serviceConfig.d.ts.map +1 -0
- package/lib/runtime/serviceConfig.js +152 -0
- package/lib/runtime/serviceConfig.js.map +1 -0
- package/lib/runtime/statusViews.d.ts +23 -0
- package/lib/runtime/statusViews.d.ts.map +1 -0
- package/lib/runtime/statusViews.js +48 -0
- package/lib/runtime/statusViews.js.map +1 -0
- package/lib/runtime/stream.d.ts +41 -0
- package/lib/runtime/stream.d.ts.map +1 -0
- package/lib/runtime/stream.js +92 -0
- package/lib/runtime/stream.js.map +1 -0
- package/lib/runtime/streamHandler.d.ts +48 -0
- package/lib/runtime/streamHandler.d.ts.map +1 -0
- package/lib/runtime/streamHandler.js +49 -0
- package/lib/runtime/streamHandler.js.map +1 -0
- package/lib/runtime/tenantResolution.d.ts +4 -0
- package/lib/runtime/tenantResolution.d.ts.map +1 -0
- package/lib/runtime/tenantResolution.js +19 -0
- package/lib/runtime/tenantResolution.js.map +1 -0
- package/lib/runtime/uuid.d.ts +6 -0
- package/lib/runtime/uuid.d.ts.map +1 -0
- package/lib/runtime/uuid.js +27 -0
- package/lib/runtime/uuid.js.map +1 -0
- package/lib/runtime/view.d.ts +48 -0
- package/lib/runtime/view.d.ts.map +1 -0
- package/lib/runtime/view.js +111 -0
- package/lib/runtime/view.js.map +1 -0
- package/package.json +56 -0
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { PluginManifestSchema } from "../contracts/manifest.js";
|
|
2
|
+
export async function importBindingManifest(store, bindingId, provider) {
|
|
3
|
+
const binding = store.getBinding(bindingId);
|
|
4
|
+
if (!binding) {
|
|
5
|
+
throw new Error(`Binding ${bindingId} does not exist`);
|
|
6
|
+
}
|
|
7
|
+
const manifest = PluginManifestSchema.parse(await provider(binding));
|
|
8
|
+
const updatedBinding = {
|
|
9
|
+
...binding,
|
|
10
|
+
importedManifestVersion: manifest.version,
|
|
11
|
+
lastSyncAtIso: new Date().toISOString()
|
|
12
|
+
};
|
|
13
|
+
store.upsertBinding(updatedBinding);
|
|
14
|
+
return store.recordImportedManifest(bindingId, manifest);
|
|
15
|
+
}
|
|
16
|
+
export async function syncAllBindingManifests(store, provider) {
|
|
17
|
+
const snapshot = store.getSnapshot();
|
|
18
|
+
const results = [];
|
|
19
|
+
for (const binding of snapshot.bindings) {
|
|
20
|
+
results.push(await importBindingManifest(store, binding.bindingId, provider));
|
|
21
|
+
}
|
|
22
|
+
return results;
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=sync.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.js","sourceRoot":"","sources":["../../src/controlPlane/sync.ts"],"names":[],"mappings":"AACA,OAAO,EAAkB,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAMhF,MAAM,CAAC,KAAK,UAAU,qBAAqB,CACzC,KAAoC,EACpC,SAAiB,EACjB,QAA0B;IAE1B,MAAM,OAAO,GAAG,KAAK,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;IAC5C,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,MAAM,IAAI,KAAK,CAAC,WAAW,SAAS,iBAAiB,CAAC,CAAC;IACzD,CAAC;IAED,MAAM,QAAQ,GAAG,oBAAoB,CAAC,KAAK,CAAC,MAAM,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;IACrE,MAAM,cAAc,GAAkB;QACpC,GAAG,OAAO;QACV,uBAAuB,EAAE,QAAQ,CAAC,OAAO;QACzC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;KACxC,CAAC;IAEF,KAAK,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC;IACpC,OAAO,KAAK,CAAC,sBAAsB,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;AAC3D,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,uBAAuB,CAC3C,KAAoC,EACpC,QAA0B;IAE1B,MAAM,QAAQ,GAAG,KAAK,CAAC,WAAW,EAAE,CAAC;IACrC,MAAM,OAAO,GAA6B,EAAE,CAAC;IAE7C,KAAK,MAAM,OAAO,IAAI,QAAQ,CAAC,QAAQ,EAAE,CAAC;QACxC,OAAO,CAAC,IAAI,CAAC,MAAM,qBAAqB,CAAC,KAAK,EAAE,OAAO,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC,CAAC;IAChF,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { App, BindingRecord, ServiceCatalogEntry, Tenant } from "../contracts/binding.js";
|
|
2
|
+
import { PluginManifest } from "../contracts/manifest.js";
|
|
3
|
+
export interface ImportedManifestRecord {
|
|
4
|
+
bindingId: string;
|
|
5
|
+
manifest: PluginManifest;
|
|
6
|
+
importedAtIso: string;
|
|
7
|
+
}
|
|
8
|
+
export interface ControlPlaneSnapshot {
|
|
9
|
+
catalog: ReadonlyArray<ServiceCatalogEntry>;
|
|
10
|
+
tenants: ReadonlyArray<Tenant>;
|
|
11
|
+
apps: ReadonlyArray<App>;
|
|
12
|
+
bindings: ReadonlyArray<BindingRecord>;
|
|
13
|
+
importedManifests: ReadonlyArray<ImportedManifestRecord>;
|
|
14
|
+
}
|
|
15
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/controlPlane/types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,aAAa,EAAE,mBAAmB,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAC;AAC1F,OAAO,EAAE,cAAc,EAAE,MAAM,0BAA0B,CAAC;AAE1D,MAAM,WAAW,sBAAsB;IACrC,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,cAAc,CAAC;IACzB,aAAa,EAAE,MAAM,CAAC;CACvB;AAED,MAAM,WAAW,oBAAoB;IACnC,OAAO,EAAE,aAAa,CAAC,mBAAmB,CAAC,CAAC;IAC5C,OAAO,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC;IAC/B,IAAI,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC;IACzB,QAAQ,EAAE,aAAa,CAAC,aAAa,CAAC,CAAC;IACvC,iBAAiB,EAAE,aAAa,CAAC,sBAAsB,CAAC,CAAC;CAC1D"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/controlPlane/types.ts"],"names":[],"mappings":""}
|
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
export * from "./contracts/auth.js";
|
|
2
|
+
export * from "./contracts/binding.js";
|
|
3
|
+
export * from "./contracts/common.js";
|
|
4
|
+
export * from "./contracts/config.js";
|
|
5
|
+
export * from "./contracts/json.js";
|
|
6
|
+
export * from "./contracts/manifest.js";
|
|
7
|
+
export * from "./contracts/observability.js";
|
|
8
|
+
export * from "./contracts/platformConfig.js";
|
|
9
|
+
export * from "./contracts/registry.js";
|
|
10
|
+
export * from "./contracts/route.js";
|
|
11
|
+
export * from "./contracts/serviceConfig.js";
|
|
12
|
+
export * from "./contracts/streaming.js";
|
|
13
|
+
export * from "./contracts/view.js";
|
|
14
|
+
export * from "./controlPlane/store.js";
|
|
15
|
+
export * from "./controlPlane/sync.js";
|
|
16
|
+
export * from "./controlPlane/types.js";
|
|
17
|
+
export * from "./runtime/configProvider.js";
|
|
18
|
+
export * from "./runtime/h3.js";
|
|
19
|
+
export * from "./runtime/jsonSchema.js";
|
|
20
|
+
export * from "./runtime/http.js";
|
|
21
|
+
export * from "./runtime/manifest.js";
|
|
22
|
+
export * from "./runtime/media.js";
|
|
23
|
+
export * from "./runtime/registry.js";
|
|
24
|
+
export * from "./runtime/serviceConfig.js";
|
|
25
|
+
export * from "./runtime/tenantResolution.js";
|
|
26
|
+
export * from "./runtime/view.js";
|
|
27
|
+
export * from "./runtime/handler.js";
|
|
28
|
+
export * from "./runtime/streamHandler.js";
|
|
29
|
+
export * from "./runtime/configStore.js";
|
|
30
|
+
export * from "./runtime/configTicket.js";
|
|
31
|
+
export * from "./runtime/auth/envelope.js";
|
|
32
|
+
export * from "./runtime/auth/issuer.js";
|
|
33
|
+
export * from "./runtime/auth/jwks.js";
|
|
34
|
+
export * from "./runtime/auth/keypair.js";
|
|
35
|
+
export * from "./runtime/auth/tokens.js";
|
|
36
|
+
export * from "./runtime/auth/verifier.js";
|
|
37
|
+
export * from "./contracts/controlPlane.js";
|
|
38
|
+
export * from "./runtime/uuid.js";
|
|
39
|
+
export * from "./adapters/h3.js";
|
|
40
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,qBAAqB,CAAC;AACpC,cAAc,wBAAwB,CAAC;AACvC,cAAc,uBAAuB,CAAC;AACtC,cAAc,uBAAuB,CAAC;AACtC,cAAc,qBAAqB,CAAC;AACpC,cAAc,yBAAyB,CAAC;AACxC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,yBAAyB,CAAC;AACxC,cAAc,sBAAsB,CAAC;AACrC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,0BAA0B,CAAC;AACzC,cAAc,qBAAqB,CAAC;AACpC,cAAc,yBAAyB,CAAC;AACxC,cAAc,wBAAwB,CAAC;AACvC,cAAc,yBAAyB,CAAC;AACxC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,iBAAiB,CAAC;AAChC,cAAc,yBAAyB,CAAC;AACxC,cAAc,mBAAmB,CAAC;AAClC,cAAc,uBAAuB,CAAC;AACtC,cAAc,oBAAoB,CAAC;AACnC,cAAc,uBAAuB,CAAC;AACtC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,mBAAmB,CAAC;AAClC,cAAc,sBAAsB,CAAC;AACrC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,0BAA0B,CAAC;AACzC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,4BAA4B,CAAC;AAC3C,cAAc,0BAA0B,CAAC;AACzC,cAAc,wBAAwB,CAAC;AACvC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,0BAA0B,CAAC;AACzC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,6BAA6B,CAAC;AAC5C,cAAc,mBAAmB,CAAC;AAClC,cAAc,kBAAkB,CAAC"}
|
package/lib/index.js
ADDED
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
export * from "./contracts/auth.js";
|
|
2
|
+
export * from "./contracts/binding.js";
|
|
3
|
+
export * from "./contracts/common.js";
|
|
4
|
+
export * from "./contracts/config.js";
|
|
5
|
+
export * from "./contracts/json.js";
|
|
6
|
+
export * from "./contracts/manifest.js";
|
|
7
|
+
export * from "./contracts/observability.js";
|
|
8
|
+
export * from "./contracts/platformConfig.js";
|
|
9
|
+
export * from "./contracts/registry.js";
|
|
10
|
+
export * from "./contracts/route.js";
|
|
11
|
+
export * from "./contracts/serviceConfig.js";
|
|
12
|
+
export * from "./contracts/streaming.js";
|
|
13
|
+
export * from "./contracts/view.js";
|
|
14
|
+
export * from "./controlPlane/store.js";
|
|
15
|
+
export * from "./controlPlane/sync.js";
|
|
16
|
+
export * from "./controlPlane/types.js";
|
|
17
|
+
export * from "./runtime/configProvider.js";
|
|
18
|
+
export * from "./runtime/h3.js";
|
|
19
|
+
export * from "./runtime/jsonSchema.js";
|
|
20
|
+
export * from "./runtime/http.js";
|
|
21
|
+
export * from "./runtime/manifest.js";
|
|
22
|
+
export * from "./runtime/media.js";
|
|
23
|
+
export * from "./runtime/registry.js";
|
|
24
|
+
export * from "./runtime/serviceConfig.js";
|
|
25
|
+
export * from "./runtime/tenantResolution.js";
|
|
26
|
+
export * from "./runtime/view.js";
|
|
27
|
+
export * from "./runtime/handler.js";
|
|
28
|
+
export * from "./runtime/streamHandler.js";
|
|
29
|
+
export * from "./runtime/configStore.js";
|
|
30
|
+
export * from "./runtime/configTicket.js";
|
|
31
|
+
export * from "./runtime/auth/envelope.js";
|
|
32
|
+
export * from "./runtime/auth/issuer.js";
|
|
33
|
+
export * from "./runtime/auth/jwks.js";
|
|
34
|
+
export * from "./runtime/auth/keypair.js";
|
|
35
|
+
export * from "./runtime/auth/tokens.js";
|
|
36
|
+
export * from "./runtime/auth/verifier.js";
|
|
37
|
+
export * from "./contracts/controlPlane.js";
|
|
38
|
+
export * from "./runtime/uuid.js";
|
|
39
|
+
export * from "./adapters/h3.js";
|
|
40
|
+
//# sourceMappingURL=index.js.map
|
package/lib/index.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,qBAAqB,CAAC;AACpC,cAAc,wBAAwB,CAAC;AACvC,cAAc,uBAAuB,CAAC;AACtC,cAAc,uBAAuB,CAAC;AACtC,cAAc,qBAAqB,CAAC;AACpC,cAAc,yBAAyB,CAAC;AACxC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,yBAAyB,CAAC;AACxC,cAAc,sBAAsB,CAAC;AACrC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,0BAA0B,CAAC;AACzC,cAAc,qBAAqB,CAAC;AACpC,cAAc,yBAAyB,CAAC;AACxC,cAAc,wBAAwB,CAAC;AACvC,cAAc,yBAAyB,CAAC;AACxC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,iBAAiB,CAAC;AAChC,cAAc,yBAAyB,CAAC;AACxC,cAAc,mBAAmB,CAAC;AAClC,cAAc,uBAAuB,CAAC;AACtC,cAAc,oBAAoB,CAAC;AACnC,cAAc,uBAAuB,CAAC;AACtC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,mBAAmB,CAAC;AAClC,cAAc,sBAAsB,CAAC;AACrC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,0BAA0B,CAAC;AACzC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,4BAA4B,CAAC;AAC3C,cAAc,0BAA0B,CAAC;AACzC,cAAc,wBAAwB,CAAC;AACvC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,0BAA0B,CAAC;AACzC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,6BAA6B,CAAC;AAC5C,cAAc,mBAAmB,CAAC;AAClC,cAAc,kBAAkB,CAAC"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { type CpEnvelopeClaims, type SetupTokenClaims } from "../../contracts/auth.js";
|
|
2
|
+
import { type JwksLookupOptions } from "./jwks.js";
|
|
3
|
+
export interface SignCpEnvelopeOptions {
|
|
4
|
+
privateKeyPem: string;
|
|
5
|
+
kid: string;
|
|
6
|
+
claims: Omit<CpEnvelopeClaims, "exp" | "iat" | "jti"> & {
|
|
7
|
+
expiresInSeconds: number;
|
|
8
|
+
jti?: string;
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
export interface SignSetupTokenOptions {
|
|
12
|
+
privateKeyPem: string;
|
|
13
|
+
kid: string;
|
|
14
|
+
claims: Omit<SetupTokenClaims, "exp" | "iat" | "jti"> & {
|
|
15
|
+
expiresInSeconds: number;
|
|
16
|
+
jti?: string;
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
export interface VerifyEnvelopeOptions {
|
|
20
|
+
/** JWKS lookup details for the issuer (CP). */
|
|
21
|
+
jwks?: JwksLookupOptions;
|
|
22
|
+
/** Or an in-process key resolver (preferred for installer flow). */
|
|
23
|
+
keyResolver?: (kid: string) => Promise<string>;
|
|
24
|
+
expectedIssuer: string;
|
|
25
|
+
expectedAudience?: string;
|
|
26
|
+
clockToleranceSeconds?: number;
|
|
27
|
+
}
|
|
28
|
+
export declare function signCpEnvelope(options: SignCpEnvelopeOptions): string;
|
|
29
|
+
export declare function signSetupToken(options: SignSetupTokenOptions): string;
|
|
30
|
+
export declare function verifyCpEnvelope(token: string, options: VerifyEnvelopeOptions): Promise<CpEnvelopeClaims>;
|
|
31
|
+
export declare function verifySetupToken(token: string, options: VerifyEnvelopeOptions): Promise<SetupTokenClaims>;
|
|
32
|
+
//# sourceMappingURL=envelope.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"envelope.d.ts","sourceRoot":"","sources":["../../../src/runtime/auth/envelope.ts"],"names":[],"mappings":"AACA,OAAO,EAGL,KAAK,gBAAgB,EACrB,KAAK,gBAAgB,EACtB,MAAM,yBAAyB,CAAC;AAEjC,OAAO,EAAuB,KAAK,iBAAiB,EAAE,MAAM,WAAW,CAAC;AAMxE,MAAM,WAAW,qBAAqB;IACpC,aAAa,EAAE,MAAM,CAAC;IACtB,GAAG,EAAE,MAAM,CAAC;IACZ,MAAM,EAAE,IAAI,CAAC,gBAAgB,EAAE,KAAK,GAAG,KAAK,GAAG,KAAK,CAAC,GAAG;QACtD,gBAAgB,EAAE,MAAM,CAAC;QACzB,GAAG,CAAC,EAAE,MAAM,CAAC;KACd,CAAC;CACH;AAED,MAAM,WAAW,qBAAqB;IACpC,aAAa,EAAE,MAAM,CAAC;IACtB,GAAG,EAAE,MAAM,CAAC;IACZ,MAAM,EAAE,IAAI,CAAC,gBAAgB,EAAE,KAAK,GAAG,KAAK,GAAG,KAAK,CAAC,GAAG;QACtD,gBAAgB,EAAE,MAAM,CAAC;QACzB,GAAG,CAAC,EAAE,MAAM,CAAC;KACd,CAAC;CACH;AAED,MAAM,WAAW,qBAAqB;IACpC,+CAA+C;IAC/C,IAAI,CAAC,EAAE,iBAAiB,CAAC;IACzB,oEAAoE;IACpE,WAAW,CAAC,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,CAAC;IAC/C,cAAc,EAAE,MAAM,CAAC;IACvB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,qBAAqB,CAAC,EAAE,MAAM,CAAC;CAChC;AAED,wBAAgB,cAAc,CAAC,OAAO,EAAE,qBAAqB,GAAG,MAAM,CAgBrE;AAED,wBAAgB,cAAc,CAAC,OAAO,EAAE,qBAAqB,GAAG,MAAM,CAgBrE;AAED,wBAAsB,gBAAgB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAM/G;AAED,wBAAsB,gBAAgB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAM/G"}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import jwt from "jsonwebtoken";
|
|
2
|
+
import { CpEnvelopeClaimsSchema, SetupTokenClaimsSchema } from "../../contracts/auth.js";
|
|
3
|
+
import { uuidv7 } from "../uuid.js";
|
|
4
|
+
import { getSigningKeyForKid } from "./jwks.js";
|
|
5
|
+
const ALLOWED_ALGORITHM = "RS256";
|
|
6
|
+
const ALLOWED_TYP = "JWT";
|
|
7
|
+
const KID_PATTERN = /^[A-Za-z0-9_-]+$/;
|
|
8
|
+
export function signCpEnvelope(options) {
|
|
9
|
+
const now = Math.floor(Date.now() / 1000);
|
|
10
|
+
const { expiresInSeconds, jti, ...rest } = options.claims;
|
|
11
|
+
const fullClaims = {
|
|
12
|
+
...rest,
|
|
13
|
+
iat: now,
|
|
14
|
+
exp: now + expiresInSeconds,
|
|
15
|
+
jti: jti ?? uuidv7()
|
|
16
|
+
};
|
|
17
|
+
const validated = CpEnvelopeClaimsSchema.parse(fullClaims);
|
|
18
|
+
return jwt.sign(validated, options.privateKeyPem, {
|
|
19
|
+
algorithm: ALLOWED_ALGORITHM,
|
|
20
|
+
keyid: options.kid,
|
|
21
|
+
header: { alg: ALLOWED_ALGORITHM, typ: ALLOWED_TYP, kid: options.kid }
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
export function signSetupToken(options) {
|
|
25
|
+
const now = Math.floor(Date.now() / 1000);
|
|
26
|
+
const { expiresInSeconds, jti, ...rest } = options.claims;
|
|
27
|
+
const fullClaims = {
|
|
28
|
+
...rest,
|
|
29
|
+
iat: now,
|
|
30
|
+
exp: now + expiresInSeconds,
|
|
31
|
+
jti: jti ?? uuidv7()
|
|
32
|
+
};
|
|
33
|
+
const validated = SetupTokenClaimsSchema.parse(fullClaims);
|
|
34
|
+
return jwt.sign(validated, options.privateKeyPem, {
|
|
35
|
+
algorithm: ALLOWED_ALGORITHM,
|
|
36
|
+
keyid: options.kid,
|
|
37
|
+
header: { alg: ALLOWED_ALGORITHM, typ: ALLOWED_TYP, kid: options.kid }
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
export async function verifyCpEnvelope(token, options) {
|
|
41
|
+
const claims = await verifyTypedToken(token, options);
|
|
42
|
+
if (claims.tokenType !== "cp-envelope") {
|
|
43
|
+
throw new Error(`Expected cp-envelope token, got ${String(claims.tokenType)}`);
|
|
44
|
+
}
|
|
45
|
+
return CpEnvelopeClaimsSchema.parse(claims);
|
|
46
|
+
}
|
|
47
|
+
export async function verifySetupToken(token, options) {
|
|
48
|
+
const claims = await verifyTypedToken(token, options);
|
|
49
|
+
if (claims.tokenType !== "setup") {
|
|
50
|
+
throw new Error(`Expected setup token, got ${String(claims.tokenType)}`);
|
|
51
|
+
}
|
|
52
|
+
return SetupTokenClaimsSchema.parse(claims);
|
|
53
|
+
}
|
|
54
|
+
async function verifyTypedToken(token, options) {
|
|
55
|
+
if (typeof token !== "string" || token.length === 0) {
|
|
56
|
+
throw new Error("Token is empty");
|
|
57
|
+
}
|
|
58
|
+
const parts = token.split(".");
|
|
59
|
+
if (parts.length !== 3) {
|
|
60
|
+
throw new Error("Token must have exactly three parts");
|
|
61
|
+
}
|
|
62
|
+
const [encodedHeader] = parts;
|
|
63
|
+
const header = parseHeader(encodedHeader);
|
|
64
|
+
if (header.alg !== ALLOWED_ALGORITHM) {
|
|
65
|
+
throw new Error(`Algorithm not allowed: ${String(header.alg)}`);
|
|
66
|
+
}
|
|
67
|
+
if (header.typ !== ALLOWED_TYP) {
|
|
68
|
+
throw new Error(`Token typ not allowed: ${String(header.typ)}`);
|
|
69
|
+
}
|
|
70
|
+
if (typeof header.kid !== "string" || !KID_PATTERN.test(header.kid)) {
|
|
71
|
+
throw new Error("Invalid kid");
|
|
72
|
+
}
|
|
73
|
+
if ("jku" in header || "x5u" in header) {
|
|
74
|
+
throw new Error("Token header contains untrusted reference (jku/x5u)");
|
|
75
|
+
}
|
|
76
|
+
const publicKeyPem = options.keyResolver
|
|
77
|
+
? await options.keyResolver(header.kid)
|
|
78
|
+
: await getSigningKeyForKid(requireJwks(options), header.kid);
|
|
79
|
+
let verified;
|
|
80
|
+
try {
|
|
81
|
+
verified = jwt.verify(token, publicKeyPem, {
|
|
82
|
+
algorithms: [ALLOWED_ALGORITHM],
|
|
83
|
+
issuer: options.expectedIssuer,
|
|
84
|
+
audience: options.expectedAudience,
|
|
85
|
+
clockTolerance: options.clockToleranceSeconds ?? 0,
|
|
86
|
+
complete: false
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
catch (error) {
|
|
90
|
+
throw new Error(`Envelope verification failed: ${error.message}`);
|
|
91
|
+
}
|
|
92
|
+
if (!verified || typeof verified !== "object") {
|
|
93
|
+
throw new Error("Verifier returned non-object claims");
|
|
94
|
+
}
|
|
95
|
+
return verified;
|
|
96
|
+
}
|
|
97
|
+
function parseHeader(encodedHeader) {
|
|
98
|
+
let json;
|
|
99
|
+
try {
|
|
100
|
+
json = Buffer.from(encodedHeader, "base64url").toString("utf8");
|
|
101
|
+
}
|
|
102
|
+
catch {
|
|
103
|
+
throw new Error("Token header is not valid base64url");
|
|
104
|
+
}
|
|
105
|
+
let parsed;
|
|
106
|
+
try {
|
|
107
|
+
parsed = JSON.parse(json);
|
|
108
|
+
}
|
|
109
|
+
catch {
|
|
110
|
+
throw new Error("Token header is not valid JSON");
|
|
111
|
+
}
|
|
112
|
+
if (!parsed || typeof parsed !== "object") {
|
|
113
|
+
throw new Error("Token header is not an object");
|
|
114
|
+
}
|
|
115
|
+
return parsed;
|
|
116
|
+
}
|
|
117
|
+
function requireJwks(options) {
|
|
118
|
+
if (!options.jwks) {
|
|
119
|
+
throw new Error("verifyEnvelope requires either jwks or keyResolver");
|
|
120
|
+
}
|
|
121
|
+
return options.jwks;
|
|
122
|
+
}
|
|
123
|
+
//# sourceMappingURL=envelope.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"envelope.js","sourceRoot":"","sources":["../../../src/runtime/auth/envelope.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import type { JwtClaims, TokenType } from "../../contracts/auth.js";
|
|
2
|
+
import type { JwtVerifier } from "../../contracts/route.js";
|
|
3
|
+
import type { RsaKeyPair } from "./keypair.js";
|
|
4
|
+
export interface BpTokenIssuerOptions {
|
|
5
|
+
keyPair: RsaKeyPair;
|
|
6
|
+
issuer: string;
|
|
7
|
+
audience: string;
|
|
8
|
+
accessTokenSeconds: number;
|
|
9
|
+
refreshTokenSeconds?: number;
|
|
10
|
+
}
|
|
11
|
+
export interface BpTokenUser {
|
|
12
|
+
sub: string;
|
|
13
|
+
tenantId: string;
|
|
14
|
+
appId: string;
|
|
15
|
+
roles?: string[];
|
|
16
|
+
authProvider?: string;
|
|
17
|
+
providerSubject?: string;
|
|
18
|
+
provider?: JwtClaims["provider"];
|
|
19
|
+
name?: string;
|
|
20
|
+
email?: string;
|
|
21
|
+
picture?: string;
|
|
22
|
+
}
|
|
23
|
+
export interface BpIssuedTokenPair {
|
|
24
|
+
tokenId: string;
|
|
25
|
+
accessToken: string;
|
|
26
|
+
accessTokenExpiresInSeconds: number;
|
|
27
|
+
refreshToken?: string;
|
|
28
|
+
refreshTokenExpiresInSeconds?: number;
|
|
29
|
+
}
|
|
30
|
+
export interface BpRefreshTokenValidation {
|
|
31
|
+
refreshToken: string;
|
|
32
|
+
tenantId: string;
|
|
33
|
+
appId: string;
|
|
34
|
+
}
|
|
35
|
+
export declare function createBpTokenIssuer(options: BpTokenIssuerOptions): {
|
|
36
|
+
signAccessToken(input: BpTokenUser): string;
|
|
37
|
+
issueTokenPair(input: BpTokenUser, pairOptions?: {
|
|
38
|
+
includeRefreshToken?: boolean;
|
|
39
|
+
}): BpIssuedTokenPair;
|
|
40
|
+
verifyRefreshToken(input: BpRefreshTokenValidation): Promise<JwtClaims>;
|
|
41
|
+
verifier(expectedTokenType?: TokenType): JwtVerifier;
|
|
42
|
+
};
|
|
43
|
+
export type BpTokenIssuer = ReturnType<typeof createBpTokenIssuer>;
|
|
44
|
+
//# sourceMappingURL=issuer.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"issuer.d.ts","sourceRoot":"","sources":["../../../src/runtime/auth/issuer.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAC;AACpE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAE5D,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAG/C,MAAM,WAAW,oBAAoB;IACnC,OAAO,EAAE,UAAU,CAAC;IACpB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,mBAAmB,CAAC,EAAE,MAAM,CAAC;CAC9B;AAED,MAAM,WAAW,WAAW;IAC1B,GAAG,EAAE,MAAM,CAAC;IACZ,QAAQ,EAAE,MAAM,CAAC;IACjB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,EAAE,SAAS,CAAC,UAAU,CAAC,CAAC;IACjC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,2BAA2B,EAAE,MAAM,CAAC;IACpC,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,4BAA4B,CAAC,EAAE,MAAM,CAAC;CACvC;AAED,MAAM,WAAW,wBAAwB;IACvC,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,KAAK,EAAE,MAAM,CAAC;CACf;AAED,wBAAgB,mBAAmB,CAAC,OAAO,EAAE,oBAAoB;2BA8BtC,WAAW,GAAG,MAAM;0BAIrB,WAAW,gBAAe;QAAE,mBAAmB,CAAC,EAAE,OAAO,CAAA;KAAE,GAAmC,iBAAiB;8BAsBrG,wBAAwB,GAAG,OAAO,CAAC,SAAS,CAAC;iCAajD,SAAS,GAAc,WAAW;EAajE;AAED,MAAM,MAAM,aAAa,GAAG,UAAU,CAAC,OAAO,mBAAmB,CAAC,CAAC"}
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import { uuidv7 } from "../uuid.js";
|
|
2
|
+
import { signJwt, verifyJwt } from "./tokens.js";
|
|
3
|
+
export function createBpTokenIssuer(options) {
|
|
4
|
+
const keyResolver = async (kid) => {
|
|
5
|
+
if (kid !== options.keyPair.kid)
|
|
6
|
+
throw new Error("Unknown signing key");
|
|
7
|
+
return options.keyPair.publicKeyPem;
|
|
8
|
+
};
|
|
9
|
+
const signToken = (input, tokenType, expiresInSeconds, tokenId = uuidv7()) => signJwt({
|
|
10
|
+
privateKeyPem: options.keyPair.privateKeyPem,
|
|
11
|
+
kid: options.keyPair.kid,
|
|
12
|
+
claims: {
|
|
13
|
+
iss: options.issuer,
|
|
14
|
+
aud: options.audience,
|
|
15
|
+
sub: input.sub,
|
|
16
|
+
tenantId: input.tenantId,
|
|
17
|
+
appId: input.appId,
|
|
18
|
+
roles: tokenType === "refresh" ? [] : (input.roles ?? []),
|
|
19
|
+
realm: "runtime",
|
|
20
|
+
tokenType,
|
|
21
|
+
authProvider: input.authProvider,
|
|
22
|
+
providerSubject: input.providerSubject,
|
|
23
|
+
provider: input.provider,
|
|
24
|
+
name: input.name,
|
|
25
|
+
email: input.email,
|
|
26
|
+
picture: input.picture,
|
|
27
|
+
expiresInSeconds,
|
|
28
|
+
jti: tokenId
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
return {
|
|
32
|
+
signAccessToken(input) {
|
|
33
|
+
return signToken(input, "access", options.accessTokenSeconds);
|
|
34
|
+
},
|
|
35
|
+
issueTokenPair(input, pairOptions = { includeRefreshToken: true }) {
|
|
36
|
+
const tokenId = uuidv7();
|
|
37
|
+
const accessToken = signToken(input, "access", options.accessTokenSeconds, tokenId);
|
|
38
|
+
if (!pairOptions.includeRefreshToken) {
|
|
39
|
+
return {
|
|
40
|
+
tokenId,
|
|
41
|
+
accessToken,
|
|
42
|
+
accessTokenExpiresInSeconds: options.accessTokenSeconds
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
if (!options.refreshTokenSeconds) {
|
|
46
|
+
throw new Error("refreshTokenSeconds is required to issue refresh tokens");
|
|
47
|
+
}
|
|
48
|
+
return {
|
|
49
|
+
tokenId,
|
|
50
|
+
accessToken,
|
|
51
|
+
accessTokenExpiresInSeconds: options.accessTokenSeconds,
|
|
52
|
+
refreshToken: signToken(input, "refresh", options.refreshTokenSeconds, tokenId),
|
|
53
|
+
refreshTokenExpiresInSeconds: options.refreshTokenSeconds
|
|
54
|
+
};
|
|
55
|
+
},
|
|
56
|
+
async verifyRefreshToken(input) {
|
|
57
|
+
const claims = await verifyJwt(input.refreshToken, {
|
|
58
|
+
keyResolver,
|
|
59
|
+
expectedIssuer: options.issuer,
|
|
60
|
+
expectedAudience: options.audience,
|
|
61
|
+
expectedTokenType: "refresh"
|
|
62
|
+
});
|
|
63
|
+
if (claims.tenantId !== input.tenantId || claims.appId !== input.appId) {
|
|
64
|
+
throw new Error("Refresh token bound to a different tenant/app");
|
|
65
|
+
}
|
|
66
|
+
return claims;
|
|
67
|
+
},
|
|
68
|
+
verifier(expectedTokenType = "access") {
|
|
69
|
+
return {
|
|
70
|
+
verify(token) {
|
|
71
|
+
return verifyJwt(token, {
|
|
72
|
+
keyResolver,
|
|
73
|
+
expectedIssuer: options.issuer,
|
|
74
|
+
expectedAudience: options.audience,
|
|
75
|
+
expectedTokenType
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
//# sourceMappingURL=issuer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"issuer.js","sourceRoot":"","sources":["../../../src/runtime/auth/issuer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
export interface JwksLookupOptions {
|
|
2
|
+
jwksUri: string;
|
|
3
|
+
issuer: string;
|
|
4
|
+
}
|
|
5
|
+
export declare function getSigningKeyForKid(options: JwksLookupOptions, kid: string): Promise<string>;
|
|
6
|
+
export declare function clearJwksCache(): void;
|
|
7
|
+
//# sourceMappingURL=jwks.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"jwks.d.ts","sourceRoot":"","sources":["../../../src/runtime/auth/jwks.ts"],"names":[],"mappings":"AAaA,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;CAChB;AAqDD,wBAAsB,mBAAmB,CACvC,OAAO,EAAE,iBAAiB,EAC1B,GAAG,EAAE,MAAM,GACV,OAAO,CAAC,MAAM,CAAC,CAcjB;AAED,wBAAgB,cAAc,IAAI,IAAI,CAErC"}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { createPublicKey } from "node:crypto";
|
|
2
|
+
const KID_PATTERN = /^[A-Za-z0-9_-]+$/;
|
|
3
|
+
const clientCache = new Map();
|
|
4
|
+
const CACHE_TTL_MS = 30 * 60 * 1000;
|
|
5
|
+
async function getJwksKeys(options) {
|
|
6
|
+
const cacheKey = `${options.issuer}|${options.jwksUri}`;
|
|
7
|
+
const now = Date.now();
|
|
8
|
+
const existing = clientCache.get(cacheKey);
|
|
9
|
+
if (existing && now - existing.lastUsed < CACHE_TTL_MS) {
|
|
10
|
+
existing.lastUsed = now;
|
|
11
|
+
return existing.keys;
|
|
12
|
+
}
|
|
13
|
+
const response = await fetch(options.jwksUri, {
|
|
14
|
+
headers: { accept: "application/json" },
|
|
15
|
+
signal: AbortSignal.timeout(5000)
|
|
16
|
+
});
|
|
17
|
+
const contentType = response.headers.get("content-type") ?? "";
|
|
18
|
+
const text = await response.text();
|
|
19
|
+
if (!response.ok) {
|
|
20
|
+
throw new Error(`JWKS fetch failed: ${options.jwksUri} HTTP ${response.status} ${text.slice(0, 160)}`);
|
|
21
|
+
}
|
|
22
|
+
if (!contentType.includes("application/json") && !contentType.includes("application/jwk-set+json")) {
|
|
23
|
+
throw new Error(`JWKS endpoint returned non-JSON: ${options.jwksUri} content-type=${contentType || "(missing)"} body=${text.slice(0, 160)}`);
|
|
24
|
+
}
|
|
25
|
+
let parsed;
|
|
26
|
+
try {
|
|
27
|
+
parsed = JSON.parse(text);
|
|
28
|
+
}
|
|
29
|
+
catch (error) {
|
|
30
|
+
throw new Error(`JWKS endpoint returned invalid JSON: ${options.jwksUri}: ${error.message}`);
|
|
31
|
+
}
|
|
32
|
+
const rawKeys = parsed.keys;
|
|
33
|
+
if (!Array.isArray(rawKeys)) {
|
|
34
|
+
throw new Error(`JWKS endpoint missing keys array: ${options.jwksUri}`);
|
|
35
|
+
}
|
|
36
|
+
const keys = new Map();
|
|
37
|
+
for (const rawKey of rawKeys) {
|
|
38
|
+
const kid = rawKey.kid;
|
|
39
|
+
if (typeof kid !== "string" || kid.length === 0)
|
|
40
|
+
continue;
|
|
41
|
+
try {
|
|
42
|
+
const pem = createPublicKey({ key: rawKey, format: "jwk" }).export({ type: "spki", format: "pem" });
|
|
43
|
+
keys.set(kid, pem);
|
|
44
|
+
}
|
|
45
|
+
catch {
|
|
46
|
+
// Skip unusable keys; lookup below reports the missing kid.
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
clientCache.set(cacheKey, { keys, jwksUri: options.jwksUri, lastUsed: now });
|
|
50
|
+
return keys;
|
|
51
|
+
}
|
|
52
|
+
export async function getSigningKeyForKid(options, kid) {
|
|
53
|
+
if (typeof kid !== "string" || kid.length === 0 || kid.length > 256) {
|
|
54
|
+
throw new Error("Invalid kid: empty or too long");
|
|
55
|
+
}
|
|
56
|
+
if (!KID_PATTERN.test(kid)) {
|
|
57
|
+
throw new Error(`Invalid kid: must match ${KID_PATTERN.source}`);
|
|
58
|
+
}
|
|
59
|
+
const keys = await getJwksKeys(options);
|
|
60
|
+
const key = keys.get(kid);
|
|
61
|
+
if (!key) {
|
|
62
|
+
throw new Error(`JWKS key not found for kid ${kid}: ${options.jwksUri}`);
|
|
63
|
+
}
|
|
64
|
+
return key;
|
|
65
|
+
}
|
|
66
|
+
export function clearJwksCache() {
|
|
67
|
+
clientCache.clear();
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=jwks.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"jwks.js","sourceRoot":"","sources":["../../../src/runtime/auth/jwks.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export interface RsaKeyPair {
|
|
2
|
+
privateKeyPem: string;
|
|
3
|
+
publicKeyPem: string;
|
|
4
|
+
kid: string;
|
|
5
|
+
}
|
|
6
|
+
export interface GenerateKeyPairOptions {
|
|
7
|
+
kid?: string;
|
|
8
|
+
modulusLength?: 2048 | 3072 | 4096;
|
|
9
|
+
}
|
|
10
|
+
export declare function generateKeyPair(options?: GenerateKeyPairOptions): RsaKeyPair;
|
|
11
|
+
export declare function loadOrGenerateKeyPair(filePath: string, options?: GenerateKeyPairOptions): RsaKeyPair;
|
|
12
|
+
export interface JwkRsaPublic {
|
|
13
|
+
kty: "RSA";
|
|
14
|
+
use: "sig";
|
|
15
|
+
alg: "RS256";
|
|
16
|
+
kid: string;
|
|
17
|
+
n: string;
|
|
18
|
+
e: string;
|
|
19
|
+
}
|
|
20
|
+
export declare function publicKeyToJwk(publicKeyPem: string, kid: string): JwkRsaPublic;
|
|
21
|
+
//# sourceMappingURL=keypair.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"keypair.d.ts","sourceRoot":"","sources":["../../../src/runtime/auth/keypair.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,UAAU;IACzB,aAAa,EAAE,MAAM,CAAC;IACtB,YAAY,EAAE,MAAM,CAAC;IACrB,GAAG,EAAE,MAAM,CAAC;CACb;AAED,MAAM,WAAW,sBAAsB;IACrC,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,aAAa,CAAC,EAAE,IAAI,GAAG,IAAI,GAAG,IAAI,CAAC;CACpC;AAED,wBAAgB,eAAe,CAAC,OAAO,GAAE,sBAA2B,GAAG,UAAU,CAUhF;AAED,wBAAgB,qBAAqB,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,GAAE,sBAA2B,GAAG,UAAU,CAcxG;AAED,MAAM,WAAW,YAAY;IAC3B,GAAG,EAAE,KAAK,CAAC;IACX,GAAG,EAAE,KAAK,CAAC;IACX,GAAG,EAAE,OAAO,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,CAAC,EAAE,MAAM,CAAC;IACV,CAAC,EAAE,MAAM,CAAC;CACX;AAED,wBAAgB,cAAc,CAAC,YAAY,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,YAAY,CAc9E"}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { generateKeyPairSync, createPublicKey } from "node:crypto";
|
|
2
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync } from "node:fs";
|
|
3
|
+
import { dirname } from "node:path";
|
|
4
|
+
export function generateKeyPair(options = {}) {
|
|
5
|
+
const modulusLength = options.modulusLength ?? 2048;
|
|
6
|
+
const { privateKey, publicKey } = generateKeyPairSync("rsa", {
|
|
7
|
+
modulusLength,
|
|
8
|
+
publicKeyEncoding: { type: "spki", format: "pem" },
|
|
9
|
+
privateKeyEncoding: { type: "pkcs8", format: "pem" }
|
|
10
|
+
});
|
|
11
|
+
const kid = options.kid ?? deriveKid(publicKey);
|
|
12
|
+
return { privateKeyPem: privateKey, publicKeyPem: publicKey, kid };
|
|
13
|
+
}
|
|
14
|
+
export function loadOrGenerateKeyPair(filePath, options = {}) {
|
|
15
|
+
if (existsSync(filePath)) {
|
|
16
|
+
const raw = readFileSync(filePath, "utf8");
|
|
17
|
+
const parsed = JSON.parse(raw);
|
|
18
|
+
if (typeof parsed.privateKeyPem !== "string" || typeof parsed.publicKeyPem !== "string" || typeof parsed.kid !== "string") {
|
|
19
|
+
throw new Error(`Keypair file ${filePath} is malformed`);
|
|
20
|
+
}
|
|
21
|
+
return parsed;
|
|
22
|
+
}
|
|
23
|
+
const pair = generateKeyPair(options);
|
|
24
|
+
mkdirSync(dirname(filePath), { recursive: true });
|
|
25
|
+
writeFileSync(filePath, JSON.stringify(pair, null, 2), { mode: 0o600 });
|
|
26
|
+
return pair;
|
|
27
|
+
}
|
|
28
|
+
export function publicKeyToJwk(publicKeyPem, kid) {
|
|
29
|
+
const keyObject = createPublicKey(publicKeyPem);
|
|
30
|
+
const jwk = keyObject.export({ format: "jwk" });
|
|
31
|
+
if (jwk.kty !== "RSA" || typeof jwk.n !== "string" || typeof jwk.e !== "string") {
|
|
32
|
+
throw new Error("Public key is not RSA");
|
|
33
|
+
}
|
|
34
|
+
return {
|
|
35
|
+
kty: "RSA",
|
|
36
|
+
use: "sig",
|
|
37
|
+
alg: "RS256",
|
|
38
|
+
kid,
|
|
39
|
+
n: jwk.n,
|
|
40
|
+
e: jwk.e
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
function deriveKid(publicKeyPem) {
|
|
44
|
+
const keyObject = createPublicKey(publicKeyPem);
|
|
45
|
+
const jwk = keyObject.export({ format: "jwk" });
|
|
46
|
+
if (!jwk.n)
|
|
47
|
+
throw new Error("Cannot derive kid from non-RSA key");
|
|
48
|
+
return jwk.n.replace(/[^A-Za-z0-9_-]/g, "").slice(0, 16) || "default";
|
|
49
|
+
}
|
|
50
|
+
//# sourceMappingURL=keypair.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"keypair.js","sourceRoot":"","sources":["../../../src/runtime/auth/keypair.ts"],"names":[],"mappings":"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"}
|