@passlock/node 2.0.1 → 2.0.3
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/dist/effect.d.ts +5 -5
- package/dist/effect.d.ts.map +1 -1
- package/dist/effect.js +2 -2
- package/dist/effect.js.map +1 -1
- package/dist/errors.d.ts +71 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +30 -0
- package/dist/errors.js.map +1 -0
- package/dist/index.d.ts +93 -34
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +87 -30
- package/dist/index.js.map +1 -1
- package/dist/network.d.ts +134 -0
- package/dist/network.d.ts.map +1 -0
- package/dist/network.js +172 -0
- package/dist/network.js.map +1 -0
- package/dist/passkey/passkey.d.ts +132 -0
- package/dist/passkey/passkey.d.ts.map +1 -0
- package/dist/passkey/passkey.js +158 -0
- package/dist/passkey/passkey.js.map +1 -0
- package/dist/principal/principal.d.ts +22 -0
- package/dist/principal/principal.d.ts.map +1 -0
- package/dist/{principal.js → principal/principal.js} +28 -27
- package/dist/principal/principal.js.map +1 -0
- package/dist/safe.d.ts +154 -0
- package/dist/safe.d.ts.map +1 -0
- package/dist/safe.js +147 -0
- package/dist/safe.js.map +1 -0
- package/dist/schemas/errors.d.ts +0 -75
- package/dist/schemas/errors.d.ts.map +1 -1
- package/dist/schemas/errors.js +51 -22
- package/dist/schemas/errors.js.map +1 -1
- package/dist/schemas/passkey.d.ts +41 -19
- package/dist/schemas/passkey.d.ts.map +1 -1
- package/dist/schemas/passkey.js +21 -16
- package/dist/schemas/passkey.js.map +1 -1
- package/dist/schemas/principal.d.ts +46 -6
- package/dist/schemas/principal.d.ts.map +1 -1
- package/dist/schemas/principal.js +6 -6
- package/dist/schemas/principal.js.map +1 -1
- package/dist/schemas/satisfy.d.ts +2 -0
- package/dist/schemas/satisfy.d.ts.map +1 -0
- package/dist/schemas/satisfy.js +2 -0
- package/dist/schemas/satisfy.js.map +1 -0
- package/dist/shared.d.ts +3 -5
- package/dist/shared.d.ts.map +1 -1
- package/dist/shared.js +0 -3
- package/dist/shared.js.map +1 -1
- package/package.json +21 -22
- package/dist/passkey.d.ts +0 -24
- package/dist/passkey.d.ts.map +0 -1
- package/dist/passkey.js +0 -87
- package/dist/passkey.js.map +0 -1
- package/dist/principal.d.ts +0 -18
- package/dist/principal.d.ts.map +0 -1
- package/dist/principal.js.map +0 -1
- package/dist/testUtils.d.ts +0 -11
- package/dist/testUtils.d.ts.map +0 -1
- package/dist/testUtils.js +0 -25
- package/dist/testUtils.js.map +0 -1
- package/dist/unsafe.d.ts +0 -55
- package/dist/unsafe.d.ts.map +0 -1
- package/dist/unsafe.js +0 -52
- package/dist/unsafe.js.map +0 -1
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
import { Context, Effect, Layer } from "effect";
|
|
2
|
+
/**
|
|
3
|
+
* Supported HTTP methods accepted by {@link fetchNetwork}.
|
|
4
|
+
*/
|
|
5
|
+
export type NetworkMethod = "get" | "post" | "delete" | "patch";
|
|
6
|
+
/**
|
|
7
|
+
* Lightweight response model used by this package.
|
|
8
|
+
*
|
|
9
|
+
* Invariants:
|
|
10
|
+
* - `json` is lazy: parsing only happens when the effect is run.
|
|
11
|
+
* - `json` is memoized per response wrapper, so repeated evaluation does not
|
|
12
|
+
* parse the same payload multiple times.
|
|
13
|
+
*/
|
|
14
|
+
export interface NetworkResponse {
|
|
15
|
+
readonly status: number;
|
|
16
|
+
readonly statusText: string;
|
|
17
|
+
readonly statusMessage: string;
|
|
18
|
+
readonly headers: Readonly<Record<string, string>>;
|
|
19
|
+
readonly json: Effect.Effect<unknown, NetworkResponseError>;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Accepted request header input formats.
|
|
23
|
+
*/
|
|
24
|
+
export type NetworkHeaders = Headers | Readonly<Record<string, string>> | Array<[string, string]>;
|
|
25
|
+
/**
|
|
26
|
+
* Optional request options for {@link fetchNetwork}.
|
|
27
|
+
*/
|
|
28
|
+
export interface FetchNetworkOptions {
|
|
29
|
+
readonly headers?: NetworkHeaders;
|
|
30
|
+
}
|
|
31
|
+
declare const NetworkFetch_base: Context.TagClass<NetworkFetch, "@passlock/node/network/Fetch", typeof fetch>;
|
|
32
|
+
/**
|
|
33
|
+
* Effect service tag for the fetch implementation.
|
|
34
|
+
*
|
|
35
|
+
* Supply this service with `Effect.provide` / layers to swap fetch behavior
|
|
36
|
+
* in tests or alternate runtimes.
|
|
37
|
+
*/
|
|
38
|
+
export declare class NetworkFetch extends NetworkFetch_base {
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Default live fetch layer backed by `globalThis.fetch`.
|
|
42
|
+
*/
|
|
43
|
+
export declare const NetworkFetchLive: Layer.Layer<NetworkFetch>;
|
|
44
|
+
/**
|
|
45
|
+
* HTTP status buckets used by {@link matchStatus}.
|
|
46
|
+
*/
|
|
47
|
+
export type NetworkResponseStatusCase = "2xx" | "3xx" | "4xx" | "5xx";
|
|
48
|
+
type MatchStatusHandler<Resp extends NetworkResponse = NetworkResponse, A = unknown, E = unknown> = (response: Resp) => Effect.Effect<A, E, never>;
|
|
49
|
+
type HandlerEffect<Cases extends MatchStatusCases<Resp>, Resp extends NetworkResponse, Case extends NetworkResponseStatusCase | "orElse"> = Case extends keyof Cases ? NonNullable<Cases[Case]> extends MatchStatusHandler<Resp> ? ReturnType<NonNullable<Cases[Case]>> : never : never;
|
|
50
|
+
type MatchStatusEffectUnion<Cases extends MatchStatusCases<Resp>, Resp extends NetworkResponse> = HandlerEffect<Cases, Resp, "2xx"> | HandlerEffect<Cases, Resp, "3xx"> | HandlerEffect<Cases, Resp, "4xx"> | HandlerEffect<Cases, Resp, "5xx"> | HandlerEffect<Cases, Resp, "orElse">;
|
|
51
|
+
type EffectSuccess<T> = T extends Effect.Effect<infer A, unknown, unknown> ? A : never;
|
|
52
|
+
type EffectError<T> = T extends Effect.Effect<unknown, infer E, unknown> ? E : never;
|
|
53
|
+
type EffectContext<T> = T extends Effect.Effect<unknown, unknown, infer R> ? R : never;
|
|
54
|
+
/**
|
|
55
|
+
* Handlers used by {@link matchStatus}.
|
|
56
|
+
*
|
|
57
|
+
* Invariants:
|
|
58
|
+
* - `orElse` is always required.
|
|
59
|
+
* - Specific status family handlers are optional and fall back to `orElse`
|
|
60
|
+
* when missing.
|
|
61
|
+
*/
|
|
62
|
+
export interface MatchStatusCases<Resp extends NetworkResponse = NetworkResponse> {
|
|
63
|
+
readonly "2xx"?: MatchStatusHandler<Resp>;
|
|
64
|
+
readonly "3xx"?: MatchStatusHandler<Resp>;
|
|
65
|
+
readonly "4xx"?: MatchStatusHandler<Resp>;
|
|
66
|
+
readonly "5xx"?: MatchStatusHandler<Resp>;
|
|
67
|
+
readonly orElse: MatchStatusHandler<Resp>;
|
|
68
|
+
}
|
|
69
|
+
declare const NetworkRequestError_base: new <A extends Record<string, any> = {}>(args: import("effect/Types").Equals<A, {}> extends true ? void : { readonly [P in keyof A as P extends "_tag" ? never : P]: A[P]; }) => import("effect/Cause").YieldableError & {
|
|
70
|
+
readonly _tag: "@error/NetworkRequest";
|
|
71
|
+
} & Readonly<A>;
|
|
72
|
+
/**
|
|
73
|
+
* Raised when the underlying fetch call fails before a response is received.
|
|
74
|
+
*/
|
|
75
|
+
export declare class NetworkRequestError extends NetworkRequestError_base<{
|
|
76
|
+
readonly method: NetworkMethod;
|
|
77
|
+
readonly url: string;
|
|
78
|
+
readonly message: string;
|
|
79
|
+
}> {
|
|
80
|
+
}
|
|
81
|
+
declare const NetworkPayloadError_base: new <A extends Record<string, any> = {}>(args: import("effect/Types").Equals<A, {}> extends true ? void : { readonly [P in keyof A as P extends "_tag" ? never : P]: A[P]; }) => import("effect/Cause").YieldableError & {
|
|
82
|
+
readonly _tag: "@error/NetworkPayload";
|
|
83
|
+
} & Readonly<A>;
|
|
84
|
+
/**
|
|
85
|
+
* Raised when request payload serialization fails.
|
|
86
|
+
*/
|
|
87
|
+
export declare class NetworkPayloadError extends NetworkPayloadError_base<{
|
|
88
|
+
readonly method: NetworkMethod;
|
|
89
|
+
readonly message: string;
|
|
90
|
+
}> {
|
|
91
|
+
}
|
|
92
|
+
declare const NetworkResponseError_base: new <A extends Record<string, any> = {}>(args: import("effect/Types").Equals<A, {}> extends true ? void : { readonly [P in keyof A as P extends "_tag" ? never : P]: A[P]; }) => import("effect/Cause").YieldableError & {
|
|
93
|
+
readonly _tag: "@error/NetworkResponse";
|
|
94
|
+
} & Readonly<A>;
|
|
95
|
+
/**
|
|
96
|
+
* Raised when response body parsing fails.
|
|
97
|
+
*/
|
|
98
|
+
export declare class NetworkResponseError extends NetworkResponseError_base<{
|
|
99
|
+
readonly method: NetworkMethod;
|
|
100
|
+
readonly url: string;
|
|
101
|
+
readonly message: string;
|
|
102
|
+
}> {
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Convert an HTTP status code into its status family bucket.
|
|
106
|
+
*
|
|
107
|
+
* Returns `undefined` for codes outside the 2xx-5xx families.
|
|
108
|
+
*/
|
|
109
|
+
export declare const statusToCase: (status: number) => NetworkResponseStatusCase | undefined;
|
|
110
|
+
/**
|
|
111
|
+
* Route a {@link NetworkResponse} to the first matching status-family handler.
|
|
112
|
+
*
|
|
113
|
+
* Invariants:
|
|
114
|
+
* - `2xx`/`3xx`/`4xx`/`5xx` handlers are matched by status family.
|
|
115
|
+
* - `orElse` is always used as a fallback when no specific family handler exists.
|
|
116
|
+
* - The returned effect type is inferred as the union of all handler effects.
|
|
117
|
+
*/
|
|
118
|
+
export declare const matchStatus: <Resp extends NetworkResponse, Cases extends MatchStatusCases<Resp>>(response: Resp, cases: Cases) => Effect.Effect<EffectSuccess<MatchStatusEffectUnion<Cases, Resp>>, EffectError<MatchStatusEffectUnion<Cases, Resp>>, EffectContext<MatchStatusEffectUnion<Cases, Resp>>>;
|
|
119
|
+
/**
|
|
120
|
+
* Convert `Headers` into a readonly plain record.
|
|
121
|
+
*/
|
|
122
|
+
export declare const headersToRecord: (headers: Headers) => Readonly<Record<string, string>>;
|
|
123
|
+
/**
|
|
124
|
+
* Execute a fetch request using the injected {@link NetworkFetch} service.
|
|
125
|
+
*
|
|
126
|
+
* Invariants:
|
|
127
|
+
* - method is always normalized to upper-case before dispatch.
|
|
128
|
+
* - JSON payload (if present) is serialized before request execution.
|
|
129
|
+
* - caller headers override payload-derived headers when keys overlap.
|
|
130
|
+
* - the returned `NetworkResponse.json` remains lazy.
|
|
131
|
+
*/
|
|
132
|
+
export declare const fetchNetwork: (url: string | URL, method: NetworkMethod, payload?: unknown, options?: FetchNetworkOptions) => Effect.Effect<NetworkResponse, NetworkRequestError | NetworkPayloadError, NetworkFetch>;
|
|
133
|
+
export {};
|
|
134
|
+
//# sourceMappingURL=network.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"network.d.ts","sourceRoot":"","sources":["../src/network.ts"],"names":[],"mappings":"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"}
|
package/dist/network.js
ADDED
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import { Context, Data, Effect, Layer } from "effect";
|
|
2
|
+
/**
|
|
3
|
+
* Effect service tag for the fetch implementation.
|
|
4
|
+
*
|
|
5
|
+
* Supply this service with `Effect.provide` / layers to swap fetch behavior
|
|
6
|
+
* in tests or alternate runtimes.
|
|
7
|
+
*/
|
|
8
|
+
export class NetworkFetch extends Context.Tag("@passlock/node/network/Fetch")() {
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Default live fetch layer backed by `globalThis.fetch`.
|
|
12
|
+
*/
|
|
13
|
+
export const NetworkFetchLive = Layer.succeed(NetworkFetch, globalThis.fetch);
|
|
14
|
+
/**
|
|
15
|
+
* Raised when the underlying fetch call fails before a response is received.
|
|
16
|
+
*/
|
|
17
|
+
export class NetworkRequestError extends Data.TaggedError("@error/NetworkRequest") {
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Raised when request payload serialization fails.
|
|
21
|
+
*/
|
|
22
|
+
export class NetworkPayloadError extends Data.TaggedError("@error/NetworkPayload") {
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Raised when response body parsing fails.
|
|
26
|
+
*/
|
|
27
|
+
export class NetworkResponseError extends Data.TaggedError("@error/NetworkResponse") {
|
|
28
|
+
}
|
|
29
|
+
const formatMessage = (cause, fallback) => cause instanceof Error ? cause.message : fallback;
|
|
30
|
+
/**
|
|
31
|
+
* Convert an HTTP status code into its status family bucket.
|
|
32
|
+
*
|
|
33
|
+
* Returns `undefined` for codes outside the 2xx-5xx families.
|
|
34
|
+
*/
|
|
35
|
+
export const statusToCase = (status) => {
|
|
36
|
+
if (status >= 200 && status <= 299)
|
|
37
|
+
return "2xx";
|
|
38
|
+
if (status >= 300 && status <= 399)
|
|
39
|
+
return "3xx";
|
|
40
|
+
if (status >= 400 && status <= 499)
|
|
41
|
+
return "4xx";
|
|
42
|
+
if (status >= 500 && status <= 599)
|
|
43
|
+
return "5xx";
|
|
44
|
+
return undefined;
|
|
45
|
+
};
|
|
46
|
+
const resolveStatusHandler = (response, cases) => {
|
|
47
|
+
const statusCase = statusToCase(response.status);
|
|
48
|
+
if (!statusCase)
|
|
49
|
+
return cases.orElse;
|
|
50
|
+
const handler = cases[statusCase];
|
|
51
|
+
return (handler ?? cases.orElse);
|
|
52
|
+
};
|
|
53
|
+
/**
|
|
54
|
+
* Route a {@link NetworkResponse} to the first matching status-family handler.
|
|
55
|
+
*
|
|
56
|
+
* Invariants:
|
|
57
|
+
* - `2xx`/`3xx`/`4xx`/`5xx` handlers are matched by status family.
|
|
58
|
+
* - `orElse` is always used as a fallback when no specific family handler exists.
|
|
59
|
+
* - The returned effect type is inferred as the union of all handler effects.
|
|
60
|
+
*/
|
|
61
|
+
export const matchStatus = (response, cases) => Effect.suspend(() => {
|
|
62
|
+
const handler = resolveStatusHandler(response, cases);
|
|
63
|
+
return handler(response);
|
|
64
|
+
});
|
|
65
|
+
/**
|
|
66
|
+
* Serialize an optional payload into a JSON request body.
|
|
67
|
+
*
|
|
68
|
+
* Invariants:
|
|
69
|
+
* - `GET` requests cannot carry a payload and fail with `NetworkPayloadError`.
|
|
70
|
+
* - when a payload exists, `content-type: application/json` is emitted.
|
|
71
|
+
*/
|
|
72
|
+
const serializePayload = (method, payload) => {
|
|
73
|
+
if (payload === undefined) {
|
|
74
|
+
return Effect.succeed(undefined);
|
|
75
|
+
}
|
|
76
|
+
if (method === "get") {
|
|
77
|
+
return Effect.fail(new NetworkPayloadError({
|
|
78
|
+
method,
|
|
79
|
+
message: "GET requests do not support a request body",
|
|
80
|
+
}));
|
|
81
|
+
}
|
|
82
|
+
return Effect.try({
|
|
83
|
+
try: () => ({
|
|
84
|
+
body: JSON.stringify(payload),
|
|
85
|
+
headers: { "content-type": "application/json" },
|
|
86
|
+
}),
|
|
87
|
+
catch: (cause) => new NetworkPayloadError({
|
|
88
|
+
method,
|
|
89
|
+
message: formatMessage(cause, "Unable to serialize payload to JSON"),
|
|
90
|
+
}),
|
|
91
|
+
});
|
|
92
|
+
};
|
|
93
|
+
/**
|
|
94
|
+
* Convert `Headers` into a readonly plain record.
|
|
95
|
+
*/
|
|
96
|
+
export const headersToRecord = (headers) => Object.fromEntries(headers.entries());
|
|
97
|
+
const normalizeHeaders = (headers) => {
|
|
98
|
+
if (headers instanceof Headers) {
|
|
99
|
+
return new Headers(headers);
|
|
100
|
+
}
|
|
101
|
+
if (Array.isArray(headers)) {
|
|
102
|
+
return new Headers(headers);
|
|
103
|
+
}
|
|
104
|
+
return new Headers(headers);
|
|
105
|
+
};
|
|
106
|
+
const mergeHeaders = (...headers) => Object.assign({}, ...headers);
|
|
107
|
+
/**
|
|
108
|
+
* Build a lazy, memoized JSON parser effect from a response.
|
|
109
|
+
*
|
|
110
|
+
* Invariants:
|
|
111
|
+
* - parsing happens on effect execution, not during response construction.
|
|
112
|
+
* - parsing uses `response.clone()` so the original response body remains untouched.
|
|
113
|
+
*/
|
|
114
|
+
const makeJsonEffect = (response, context) => {
|
|
115
|
+
let cached;
|
|
116
|
+
return Effect.tryPromise({
|
|
117
|
+
try: () => {
|
|
118
|
+
cached = cached ?? response.clone().json();
|
|
119
|
+
return cached;
|
|
120
|
+
},
|
|
121
|
+
catch: (cause) => new NetworkResponseError({
|
|
122
|
+
method: context.method,
|
|
123
|
+
url: context.url,
|
|
124
|
+
message: formatMessage(cause, "Unable to parse response JSON"),
|
|
125
|
+
}),
|
|
126
|
+
});
|
|
127
|
+
};
|
|
128
|
+
const toNetworkResponse = (response, context) => ({
|
|
129
|
+
status: response.status,
|
|
130
|
+
statusText: response.statusText,
|
|
131
|
+
statusMessage: response.statusText,
|
|
132
|
+
headers: headersToRecord(response.headers),
|
|
133
|
+
json: makeJsonEffect(response, context),
|
|
134
|
+
});
|
|
135
|
+
/**
|
|
136
|
+
* Execute a fetch request using the injected {@link NetworkFetch} service.
|
|
137
|
+
*
|
|
138
|
+
* Invariants:
|
|
139
|
+
* - method is always normalized to upper-case before dispatch.
|
|
140
|
+
* - JSON payload (if present) is serialized before request execution.
|
|
141
|
+
* - caller headers override payload-derived headers when keys overlap.
|
|
142
|
+
* - the returned `NetworkResponse.json` remains lazy.
|
|
143
|
+
*/
|
|
144
|
+
export const fetchNetwork = (url, method, payload, options) => Effect.gen(function* () {
|
|
145
|
+
const requestUrl = String(url);
|
|
146
|
+
const context = { method, url: requestUrl };
|
|
147
|
+
const fetchImpl = yield* NetworkFetch;
|
|
148
|
+
const serialized = yield* serializePayload(method, payload);
|
|
149
|
+
const providedHeaders = options?.headers
|
|
150
|
+
? headersToRecord(normalizeHeaders(options.headers))
|
|
151
|
+
: {};
|
|
152
|
+
const requestHeaders = mergeHeaders(serialized?.headers ?? {}, providedHeaders);
|
|
153
|
+
const hasHeaders = Object.keys(requestHeaders).length > 0;
|
|
154
|
+
const response = yield* Effect.tryPromise({
|
|
155
|
+
try: () => fetchImpl(requestUrl, {
|
|
156
|
+
method: method.toUpperCase(),
|
|
157
|
+
...(hasHeaders ? { headers: requestHeaders } : {}),
|
|
158
|
+
...(serialized
|
|
159
|
+
? {
|
|
160
|
+
body: serialized.body,
|
|
161
|
+
}
|
|
162
|
+
: {}),
|
|
163
|
+
}),
|
|
164
|
+
catch: (cause) => new NetworkRequestError({
|
|
165
|
+
method,
|
|
166
|
+
url: requestUrl,
|
|
167
|
+
message: formatMessage(cause, "Network request failed"),
|
|
168
|
+
}),
|
|
169
|
+
});
|
|
170
|
+
return toNetworkResponse(response, context);
|
|
171
|
+
});
|
|
172
|
+
//# sourceMappingURL=network.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"network.js","sourceRoot":"","sources":["../src/network.ts"],"names":[],"mappings":"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","sourcesContent":["import { Context, Data, Effect, Layer } from \"effect\"\n\n/**\n * Supported HTTP methods accepted by {@link fetchNetwork}.\n */\nexport type NetworkMethod = \"get\" | \"post\" | \"delete\" | \"patch\"\n\n/**\n * Lightweight response model used by this package.\n *\n * Invariants:\n * - `json` is lazy: parsing only happens when the effect is run.\n * - `json` is memoized per response wrapper, so repeated evaluation does not\n * parse the same payload multiple times.\n */\nexport interface NetworkResponse {\n readonly status: number\n readonly statusText: string\n readonly statusMessage: string\n readonly headers: Readonly<Record<string, string>>\n readonly json: Effect.Effect<unknown, NetworkResponseError>\n}\n\n/**\n * Accepted request header input formats.\n */\nexport type NetworkHeaders =\n | Headers\n | Readonly<Record<string, string>>\n | Array<[string, string]>\n\n/**\n * Optional request options for {@link fetchNetwork}.\n */\nexport interface FetchNetworkOptions {\n readonly headers?: NetworkHeaders\n}\n\n/**\n * Effect service tag for the fetch implementation.\n *\n * Supply this service with `Effect.provide` / layers to swap fetch behavior\n * in tests or alternate runtimes.\n */\nexport class NetworkFetch extends Context.Tag(\"@passlock/node/network/Fetch\")<\n NetworkFetch,\n typeof fetch\n>() {}\n\n/**\n * Default live fetch layer backed by `globalThis.fetch`.\n */\nexport const NetworkFetchLive: Layer.Layer<NetworkFetch> = Layer.succeed(\n NetworkFetch,\n globalThis.fetch\n)\n\n/**\n * HTTP status buckets used by {@link matchStatus}.\n */\nexport type NetworkResponseStatusCase = \"2xx\" | \"3xx\" | \"4xx\" | \"5xx\"\n\ntype MatchStatusHandler<\n Resp extends NetworkResponse = NetworkResponse,\n A = unknown,\n E = unknown,\n> = (response: Resp) => Effect.Effect<A, E, never>\n\ntype HandlerEffect<\n Cases extends MatchStatusCases<Resp>,\n Resp extends NetworkResponse,\n Case extends NetworkResponseStatusCase | \"orElse\",\n> = Case extends keyof Cases\n ? NonNullable<Cases[Case]> extends MatchStatusHandler<Resp>\n ? ReturnType<NonNullable<Cases[Case]>>\n : never\n : never\n\ntype MatchStatusEffectUnion<\n Cases extends MatchStatusCases<Resp>,\n Resp extends NetworkResponse,\n> =\n | HandlerEffect<Cases, Resp, \"2xx\">\n | HandlerEffect<Cases, Resp, \"3xx\">\n | HandlerEffect<Cases, Resp, \"4xx\">\n | HandlerEffect<Cases, Resp, \"5xx\">\n | HandlerEffect<Cases, Resp, \"orElse\">\n\ntype MatchStatusResolvedHandler<\n Cases extends MatchStatusCases<Resp>,\n Resp extends NetworkResponse,\n> = NonNullable<\n Cases[\"2xx\"] | Cases[\"3xx\"] | Cases[\"4xx\"] | Cases[\"5xx\"] | Cases[\"orElse\"]\n>\n\ntype EffectSuccess<T> =\n T extends Effect.Effect<infer A, unknown, unknown> ? A : never\n\ntype EffectError<T> =\n T extends Effect.Effect<unknown, infer E, unknown> ? E : never\n\ntype EffectContext<T> =\n T extends Effect.Effect<unknown, unknown, infer R> ? R : never\n\n/**\n * Handlers used by {@link matchStatus}.\n *\n * Invariants:\n * - `orElse` is always required.\n * - Specific status family handlers are optional and fall back to `orElse`\n * when missing.\n */\nexport interface MatchStatusCases<\n Resp extends NetworkResponse = NetworkResponse,\n> {\n readonly \"2xx\"?: MatchStatusHandler<Resp>\n readonly \"3xx\"?: MatchStatusHandler<Resp>\n readonly \"4xx\"?: MatchStatusHandler<Resp>\n readonly \"5xx\"?: MatchStatusHandler<Resp>\n readonly orElse: MatchStatusHandler<Resp>\n}\n\ntype RequestContext = {\n readonly method: NetworkMethod\n readonly url: string\n}\n\n/**\n * Raised when the underlying fetch call fails before a response is received.\n */\nexport class NetworkRequestError extends Data.TaggedError(\n \"@error/NetworkRequest\"\n)<{\n readonly method: NetworkMethod\n readonly url: string\n readonly message: string\n}> {}\n\n/**\n * Raised when request payload serialization fails.\n */\nexport class NetworkPayloadError extends Data.TaggedError(\n \"@error/NetworkPayload\"\n)<{\n readonly method: NetworkMethod\n readonly message: string\n}> {}\n\n/**\n * Raised when response body parsing fails.\n */\nexport class NetworkResponseError extends Data.TaggedError(\n \"@error/NetworkResponse\"\n)<{\n readonly method: NetworkMethod\n readonly url: string\n readonly message: string\n}> {}\n\nconst formatMessage = (cause: unknown, fallback: string): string =>\n cause instanceof Error ? cause.message : fallback\n\n/**\n * Convert an HTTP status code into its status family bucket.\n *\n * Returns `undefined` for codes outside the 2xx-5xx families.\n */\nexport const statusToCase = (\n status: number\n): NetworkResponseStatusCase | undefined => {\n if (status >= 200 && status <= 299) return \"2xx\"\n if (status >= 300 && status <= 399) return \"3xx\"\n if (status >= 400 && status <= 499) return \"4xx\"\n if (status >= 500 && status <= 599) return \"5xx\"\n\n return undefined\n}\n\nconst resolveStatusHandler = <\n Resp extends NetworkResponse,\n Cases extends MatchStatusCases<Resp>,\n>(\n response: Resp,\n cases: Cases\n): MatchStatusResolvedHandler<Cases, Resp> => {\n const statusCase = statusToCase(response.status)\n if (!statusCase)\n return cases.orElse as MatchStatusResolvedHandler<Cases, Resp>\n\n const handler = cases[statusCase]\n return (handler ?? cases.orElse) as MatchStatusResolvedHandler<Cases, Resp>\n}\n\n/**\n * Route a {@link NetworkResponse} to the first matching status-family handler.\n *\n * Invariants:\n * - `2xx`/`3xx`/`4xx`/`5xx` handlers are matched by status family.\n * - `orElse` is always used as a fallback when no specific family handler exists.\n * - The returned effect type is inferred as the union of all handler effects.\n */\nexport const matchStatus = <\n Resp extends NetworkResponse,\n Cases extends MatchStatusCases<Resp>,\n>(\n response: Resp,\n cases: Cases\n): Effect.Effect<\n EffectSuccess<MatchStatusEffectUnion<Cases, Resp>>,\n EffectError<MatchStatusEffectUnion<Cases, Resp>>,\n EffectContext<MatchStatusEffectUnion<Cases, Resp>>\n> =>\n Effect.suspend(() => {\n const handler = resolveStatusHandler(response, cases)\n return handler(response)\n }) as Effect.Effect<\n EffectSuccess<MatchStatusEffectUnion<Cases, Resp>>,\n EffectError<MatchStatusEffectUnion<Cases, Resp>>,\n EffectContext<MatchStatusEffectUnion<Cases, Resp>>\n >\n\n/**\n * Serialize an optional payload into a JSON request body.\n *\n * Invariants:\n * - `GET` requests cannot carry a payload and fail with `NetworkPayloadError`.\n * - when a payload exists, `content-type: application/json` is emitted.\n */\nconst serializePayload = (\n method: NetworkMethod,\n payload?: unknown\n): Effect.Effect<\n | {\n readonly body: string\n readonly headers: Readonly<Record<string, string>>\n }\n | undefined,\n NetworkPayloadError\n> => {\n if (payload === undefined) {\n return Effect.succeed(undefined)\n }\n\n if (method === \"get\") {\n return Effect.fail(\n new NetworkPayloadError({\n method,\n message: \"GET requests do not support a request body\",\n })\n )\n }\n\n return Effect.try({\n try: () => ({\n body: JSON.stringify(payload),\n headers: { \"content-type\": \"application/json\" },\n }),\n catch: (cause) =>\n new NetworkPayloadError({\n method,\n message: formatMessage(cause, \"Unable to serialize payload to JSON\"),\n }),\n })\n}\n\n/**\n * Convert `Headers` into a readonly plain record.\n */\nexport const headersToRecord = (\n headers: Headers\n): Readonly<Record<string, string>> => Object.fromEntries(headers.entries())\n\nconst normalizeHeaders = (headers: NetworkHeaders): Headers => {\n if (headers instanceof Headers) {\n return new Headers(headers)\n }\n\n if (Array.isArray(headers)) {\n return new Headers(headers)\n }\n\n return new Headers(headers)\n}\n\nconst mergeHeaders = (\n ...headers: ReadonlyArray<Readonly<Record<string, string>>>\n): Readonly<Record<string, string>> => Object.assign({}, ...headers)\n\n/**\n * Build a lazy, memoized JSON parser effect from a response.\n *\n * Invariants:\n * - parsing happens on effect execution, not during response construction.\n * - parsing uses `response.clone()` so the original response body remains untouched.\n */\nconst makeJsonEffect = (\n response: Response,\n context: RequestContext\n): Effect.Effect<unknown, NetworkResponseError> => {\n let cached: Promise<unknown> | undefined\n\n return Effect.tryPromise({\n try: () => {\n cached = cached ?? response.clone().json()\n return cached\n },\n catch: (cause) =>\n new NetworkResponseError({\n method: context.method,\n url: context.url,\n message: formatMessage(cause, \"Unable to parse response JSON\"),\n }),\n })\n}\n\nconst toNetworkResponse = (\n response: Response,\n context: RequestContext\n): NetworkResponse => ({\n status: response.status,\n statusText: response.statusText,\n statusMessage: response.statusText,\n headers: headersToRecord(response.headers),\n json: makeJsonEffect(response, context),\n})\n\n/**\n * Execute a fetch request using the injected {@link NetworkFetch} service.\n *\n * Invariants:\n * - method is always normalized to upper-case before dispatch.\n * - JSON payload (if present) is serialized before request execution.\n * - caller headers override payload-derived headers when keys overlap.\n * - the returned `NetworkResponse.json` remains lazy.\n */\nexport const fetchNetwork = (\n url: string | URL,\n method: NetworkMethod,\n payload?: unknown,\n options?: FetchNetworkOptions\n): Effect.Effect<\n NetworkResponse,\n NetworkRequestError | NetworkPayloadError,\n NetworkFetch\n> =>\n Effect.gen(function* () {\n const requestUrl = String(url)\n const context: RequestContext = { method, url: requestUrl }\n const fetchImpl = yield* NetworkFetch\n const serialized = yield* serializePayload(method, payload)\n\n const providedHeaders = options?.headers\n ? headersToRecord(normalizeHeaders(options.headers))\n : {}\n\n const requestHeaders = mergeHeaders(\n serialized?.headers ?? {},\n providedHeaders\n )\n const hasHeaders = Object.keys(requestHeaders).length > 0\n\n const response = yield* Effect.tryPromise({\n try: () =>\n fetchImpl(requestUrl, {\n method: method.toUpperCase(),\n ...(hasHeaders ? { headers: requestHeaders } : {}),\n ...(serialized\n ? {\n body: serialized.body,\n }\n : {}),\n }),\n catch: (cause) =>\n new NetworkRequestError({\n method,\n url: requestUrl,\n message: formatMessage(cause, \"Network request failed\"),\n }),\n })\n\n return toNetworkResponse(response, context)\n })\n"]}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
import { Effect, type Layer, Stream } from "effect";
|
|
2
|
+
import { type NetworkFetch } from "../network.js";
|
|
3
|
+
import { ForbiddenError, NotFoundError } from "../schemas/index.js";
|
|
4
|
+
import * as PasskeySchemas from "../schemas/passkey.js";
|
|
5
|
+
import type { AuthenticatedOptions } from "../shared.js";
|
|
6
|
+
/**
|
|
7
|
+
* WebAuthn specific passkey data
|
|
8
|
+
*/
|
|
9
|
+
export type Credential = {
|
|
10
|
+
id: string;
|
|
11
|
+
userId: string;
|
|
12
|
+
username: string;
|
|
13
|
+
aaguid: string;
|
|
14
|
+
backedUp: boolean;
|
|
15
|
+
counter: number;
|
|
16
|
+
deviceType: PasskeySchemas.CredentialDeviceType;
|
|
17
|
+
transports: ReadonlyArray<PasskeySchemas.Transports>;
|
|
18
|
+
publicKey: Uint8Array<ArrayBufferLike>;
|
|
19
|
+
rpId: string;
|
|
20
|
+
};
|
|
21
|
+
/**
|
|
22
|
+
* Passkeys are usually synced across devices **but only within
|
|
23
|
+
* a specific platform/ecosystem** e.g. a passkey created on Apple
|
|
24
|
+
* devices would typically be synced across devices sharing the same
|
|
25
|
+
* iCloud ID.
|
|
26
|
+
*
|
|
27
|
+
* However, if the user also wants to sign in from their Windows
|
|
28
|
+
* or Android/Chrome devices they will need an additional passkey.
|
|
29
|
+
* Therefore when listing the passkeys registered to a user's account
|
|
30
|
+
* it's a good idea to tell them which platform the passkeys relate to.
|
|
31
|
+
*
|
|
32
|
+
* We've also included links to icons (SVG) so you can give your users
|
|
33
|
+
* a quick visual indication.
|
|
34
|
+
*/
|
|
35
|
+
export type Platform = {
|
|
36
|
+
name?: string | undefined;
|
|
37
|
+
icon?: string | undefined;
|
|
38
|
+
};
|
|
39
|
+
/**
|
|
40
|
+
* The server-side component of a passkey
|
|
41
|
+
*
|
|
42
|
+
* @category Passkeys
|
|
43
|
+
*/
|
|
44
|
+
export type Passkey = {
|
|
45
|
+
_tag: "Passkey";
|
|
46
|
+
/**
|
|
47
|
+
* Not to be confused with the credential.id
|
|
48
|
+
*/
|
|
49
|
+
id: string;
|
|
50
|
+
/**
|
|
51
|
+
* Not to be confused with the credential.userId
|
|
52
|
+
*/
|
|
53
|
+
userId?: string | undefined;
|
|
54
|
+
enabled: boolean;
|
|
55
|
+
credential: Credential;
|
|
56
|
+
platform?: Platform | undefined;
|
|
57
|
+
lastUsed?: number | undefined;
|
|
58
|
+
createdAt: number;
|
|
59
|
+
updatedAt: number;
|
|
60
|
+
};
|
|
61
|
+
export declare const isPasskey: (payload: unknown) => payload is Passkey;
|
|
62
|
+
export type PasskeySummary = {
|
|
63
|
+
readonly _tag: "PasskeySummary";
|
|
64
|
+
readonly id: string;
|
|
65
|
+
readonly userId: string;
|
|
66
|
+
readonly enabled: boolean;
|
|
67
|
+
readonly credential: {
|
|
68
|
+
readonly id: string;
|
|
69
|
+
readonly userId: string;
|
|
70
|
+
};
|
|
71
|
+
readonly lastUsed?: number | undefined;
|
|
72
|
+
readonly createdAt: number;
|
|
73
|
+
};
|
|
74
|
+
export declare const isPasskeySummary: (payload: unknown) => payload is PasskeySummary;
|
|
75
|
+
export type UpdatedPasskeys = {
|
|
76
|
+
_tag: "UpdatedPasskeys";
|
|
77
|
+
updated: ReadonlyArray<Passkey>;
|
|
78
|
+
};
|
|
79
|
+
export declare const isUpdatedPasskeys: (payload: unknown) => payload is UpdatedPasskeys;
|
|
80
|
+
export type FindAllPasskeys = {
|
|
81
|
+
readonly _tag: "FindAllPasskeys";
|
|
82
|
+
readonly cursor: string | null;
|
|
83
|
+
readonly records: ReadonlyArray<PasskeySummary>;
|
|
84
|
+
};
|
|
85
|
+
export declare const isFindAllPasskeys: (payload: unknown) => payload is FindAllPasskeys;
|
|
86
|
+
export type UpdatedPasskeyUsernames = {
|
|
87
|
+
_tag: "UpdatedPasskeyUsernames";
|
|
88
|
+
credentials: ReadonlyArray<{
|
|
89
|
+
rpId: string;
|
|
90
|
+
userId: string;
|
|
91
|
+
username: string;
|
|
92
|
+
displayName: string;
|
|
93
|
+
}>;
|
|
94
|
+
};
|
|
95
|
+
export declare const isUpdatedPasskeyUsernames: (payload: unknown) => payload is UpdatedPasskeyUsernames;
|
|
96
|
+
export interface GetPasskeyOptions extends AuthenticatedOptions {
|
|
97
|
+
passkeyId: string;
|
|
98
|
+
}
|
|
99
|
+
export declare const getPasskey: (options: GetPasskeyOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Effect.Effect<Passkey, NotFoundError | ForbiddenError>;
|
|
100
|
+
export interface DeletePasskeyOptions extends AuthenticatedOptions {
|
|
101
|
+
passkeyId: string;
|
|
102
|
+
}
|
|
103
|
+
export declare const deletePasskey: (options: DeletePasskeyOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Effect.Effect<Passkey, NotFoundError | ForbiddenError>;
|
|
104
|
+
/**
|
|
105
|
+
* @category Passkeys
|
|
106
|
+
*/
|
|
107
|
+
export interface AssignUserOptions extends AuthenticatedOptions {
|
|
108
|
+
passkeyId: string;
|
|
109
|
+
/**
|
|
110
|
+
* Custom User ID to align with your own systems
|
|
111
|
+
*/
|
|
112
|
+
userId: string;
|
|
113
|
+
}
|
|
114
|
+
export declare const assignUser: (options: AssignUserOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Effect.Effect<Passkey, NotFoundError | ForbiddenError>;
|
|
115
|
+
export interface UpdatePasskeyOptions extends AuthenticatedOptions {
|
|
116
|
+
passkeyId: string;
|
|
117
|
+
userId?: string;
|
|
118
|
+
username?: string;
|
|
119
|
+
}
|
|
120
|
+
export declare const updatePasskey: (options: UpdatePasskeyOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Effect.Effect<Passkey, NotFoundError | ForbiddenError>;
|
|
121
|
+
export interface UpdatePasskeyUsernamesOptions extends AuthenticatedOptions {
|
|
122
|
+
userId: string;
|
|
123
|
+
username: string;
|
|
124
|
+
displayName?: string;
|
|
125
|
+
}
|
|
126
|
+
export declare const updatePasskeyUsernames: (options: UpdatePasskeyUsernamesOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Effect.Effect<UpdatedPasskeyUsernames, NotFoundError | ForbiddenError>;
|
|
127
|
+
export declare const listPasskeysStream: (options: AuthenticatedOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Stream.Stream<PasskeySummary, ForbiddenError>;
|
|
128
|
+
export interface ListPasskeyOptions extends AuthenticatedOptions {
|
|
129
|
+
cursor?: string;
|
|
130
|
+
}
|
|
131
|
+
export declare const listPasskeys: (options: ListPasskeyOptions, fetchLayer?: Layer.Layer<NetworkFetch>) => Effect.Effect<FindAllPasskeys, ForbiddenError>;
|
|
132
|
+
//# sourceMappingURL=passkey.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"passkey.d.ts","sourceRoot":"","sources":["../../src/passkey/passkey.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
import { Array, Chunk, Effect, Match, Option, pipe, Schema, Stream, } from "effect";
|
|
2
|
+
import { fetchNetwork, matchStatus, NetworkFetchLive, } from "../network.js";
|
|
3
|
+
import { FindAllPasskeys as FindAllPasskeysSchema, ForbiddenError, NotFoundError, } from "../schemas/index.js";
|
|
4
|
+
import * as PasskeySchemas from "../schemas/passkey.js";
|
|
5
|
+
export const isPasskey = (payload) => Schema.is(PasskeySchemas.Passkey)(payload);
|
|
6
|
+
export const isPasskeySummary = (payload) => Schema.is(PasskeySchemas.PasskeySummary)(payload);
|
|
7
|
+
export const isUpdatedPasskeys = (payload) => Schema.is(PasskeySchemas.UpdatedPasskeys)(payload);
|
|
8
|
+
export const isFindAllPasskeys = (payload) => Schema.is(PasskeySchemas.FindAllPasskeys)(payload);
|
|
9
|
+
export const isUpdatedPasskeyUsernames = (payload) => {
|
|
10
|
+
if (typeof payload !== "object")
|
|
11
|
+
return false;
|
|
12
|
+
if (payload === null)
|
|
13
|
+
return false;
|
|
14
|
+
if (!("_tag" in payload))
|
|
15
|
+
return false;
|
|
16
|
+
if (typeof payload._tag !== "string")
|
|
17
|
+
return false;
|
|
18
|
+
if (payload._tag !== "UpdatedPasskeyUsernames")
|
|
19
|
+
return false;
|
|
20
|
+
return true;
|
|
21
|
+
};
|
|
22
|
+
/* END UpdatedPasskeyUsernames */
|
|
23
|
+
const authorizationHeaders = (apiKey) => ({
|
|
24
|
+
authorization: `Bearer ${apiKey}`,
|
|
25
|
+
});
|
|
26
|
+
const decodeResponseJson = (response, schema) => pipe(response.json, Effect.flatMap(Schema.decodeUnknown(schema)));
|
|
27
|
+
export const getPasskey = (options, fetchLayer = NetworkFetchLive) => pipe(Effect.gen(function* () {
|
|
28
|
+
const baseUrl = options.endpoint ?? "https://api.passlock.dev";
|
|
29
|
+
const { tenancyId, passkeyId } = options;
|
|
30
|
+
const url = new URL(`/${tenancyId}/passkeys/${passkeyId}`, baseUrl);
|
|
31
|
+
const response = yield* fetchNetwork(url, "get", undefined, {
|
|
32
|
+
headers: authorizationHeaders(options.apiKey),
|
|
33
|
+
});
|
|
34
|
+
const encoded = yield* matchStatus(response, {
|
|
35
|
+
"2xx": (res) => decodeResponseJson(res, PasskeySchemas.Passkey),
|
|
36
|
+
orElse: (res) => decodeResponseJson(res, Schema.Union(ForbiddenError, NotFoundError)),
|
|
37
|
+
});
|
|
38
|
+
return yield* pipe(Match.value(encoded), Match.tag("Passkey", (data) => Effect.succeed(data)), Match.tag("@error/Forbidden", (err) => Effect.fail(err)), Match.tag("@error/NotFound", (err) => Effect.fail(err)), Match.exhaustive);
|
|
39
|
+
}), Effect.catchTags({
|
|
40
|
+
"@error/NetworkPayload": (err) => Effect.die(err),
|
|
41
|
+
"@error/NetworkRequest": (err) => Effect.die(err),
|
|
42
|
+
"@error/NetworkResponse": (err) => Effect.die(err),
|
|
43
|
+
ParseError: (err) => Effect.die(err),
|
|
44
|
+
}), Effect.provide(fetchLayer));
|
|
45
|
+
export const deletePasskey = (options, fetchLayer = NetworkFetchLive) => pipe(Effect.gen(function* () {
|
|
46
|
+
const baseUrl = options.endpoint ?? "https://api.passlock.dev";
|
|
47
|
+
const { tenancyId, passkeyId } = options;
|
|
48
|
+
const url = new URL(`/${tenancyId}/passkeys/${passkeyId}`, baseUrl);
|
|
49
|
+
const response = yield* fetchNetwork(url, "delete", undefined, {
|
|
50
|
+
headers: authorizationHeaders(options.apiKey),
|
|
51
|
+
});
|
|
52
|
+
const encoded = yield* matchStatus(response, {
|
|
53
|
+
"2xx": (res) => decodeResponseJson(res, PasskeySchemas.Passkey),
|
|
54
|
+
orElse: (res) => decodeResponseJson(res, Schema.Union(ForbiddenError, NotFoundError)),
|
|
55
|
+
});
|
|
56
|
+
return yield* pipe(Match.value(encoded), Match.tag("Passkey", (deletedPasskey) => Effect.succeed(deletedPasskey)), Match.tag("@error/Forbidden", (err) => Effect.fail(err)), Match.tag("@error/NotFound", (err) => Effect.fail(err)), Match.exhaustive);
|
|
57
|
+
}), Effect.catchTags({
|
|
58
|
+
"@error/NetworkPayload": (err) => Effect.die(err),
|
|
59
|
+
"@error/NetworkRequest": (err) => Effect.die(err),
|
|
60
|
+
"@error/NetworkResponse": (err) => Effect.die(err),
|
|
61
|
+
ParseError: (err) => Effect.die(err),
|
|
62
|
+
}), Effect.provide(fetchLayer));
|
|
63
|
+
// TODO reuse updatePasskey
|
|
64
|
+
export const assignUser = (options, fetchLayer = NetworkFetchLive) => pipe(Effect.gen(function* () {
|
|
65
|
+
const baseUrl = options.endpoint ?? "https://api.passlock.dev";
|
|
66
|
+
const { userId, passkeyId } = options;
|
|
67
|
+
const { tenancyId } = options;
|
|
68
|
+
const url = new URL(`/${tenancyId}/passkeys/${passkeyId}`, baseUrl);
|
|
69
|
+
const response = yield* fetchNetwork(url, "patch", { userId }, {
|
|
70
|
+
headers: authorizationHeaders(options.apiKey),
|
|
71
|
+
});
|
|
72
|
+
const encoded = yield* matchStatus(response, {
|
|
73
|
+
"2xx": (res) => decodeResponseJson(res, PasskeySchemas.Passkey),
|
|
74
|
+
orElse: (res) => decodeResponseJson(res, Schema.Union(NotFoundError, ForbiddenError)),
|
|
75
|
+
});
|
|
76
|
+
return yield* pipe(Match.value(encoded), Match.tag("Passkey", (passkey) => Effect.succeed(passkey)), Match.tag("@error/NotFound", (err) => Effect.fail(err)), Match.tag("@error/Forbidden", (err) => Effect.fail(err)), Match.exhaustive);
|
|
77
|
+
}), Effect.catchTags({
|
|
78
|
+
"@error/NetworkPayload": (err) => Effect.die(err),
|
|
79
|
+
"@error/NetworkRequest": (err) => Effect.die(err),
|
|
80
|
+
"@error/NetworkResponse": (err) => Effect.die(err),
|
|
81
|
+
ParseError: (err) => Effect.die(err),
|
|
82
|
+
}), Effect.provide(fetchLayer));
|
|
83
|
+
export const updatePasskey = (options, fetchLayer = NetworkFetchLive) => pipe(Effect.gen(function* () {
|
|
84
|
+
const baseUrl = options.endpoint ?? "https://api.passlock.dev";
|
|
85
|
+
const { userId, passkeyId, username } = options;
|
|
86
|
+
const { tenancyId } = options;
|
|
87
|
+
const url = new URL(`/${tenancyId}/passkeys/${passkeyId}`, baseUrl);
|
|
88
|
+
const response = yield* fetchNetwork(url, "patch", { userId, username }, {
|
|
89
|
+
headers: authorizationHeaders(options.apiKey),
|
|
90
|
+
});
|
|
91
|
+
const encoded = yield* matchStatus(response, {
|
|
92
|
+
"2xx": (res) => decodeResponseJson(res, PasskeySchemas.Passkey),
|
|
93
|
+
orElse: (res) => decodeResponseJson(res, Schema.Union(NotFoundError, ForbiddenError)),
|
|
94
|
+
});
|
|
95
|
+
return yield* pipe(Match.value(encoded), Match.tag("Passkey", (passkey) => Effect.succeed(passkey)), Match.tag("@error/NotFound", (err) => Effect.fail(err)), Match.tag("@error/Forbidden", (err) => Effect.fail(err)), Match.exhaustive);
|
|
96
|
+
}), Effect.catchTags({
|
|
97
|
+
"@error/NetworkPayload": (err) => Effect.die(err),
|
|
98
|
+
"@error/NetworkRequest": (err) => Effect.die(err),
|
|
99
|
+
"@error/NetworkResponse": (err) => Effect.die(err),
|
|
100
|
+
ParseError: (err) => Effect.die(err),
|
|
101
|
+
}), Effect.provide(fetchLayer));
|
|
102
|
+
const updateUserPasskeys = (options, fetchLayer = NetworkFetchLive) => pipe(Effect.gen(function* () {
|
|
103
|
+
const baseUrl = options.endpoint ?? "https://api.passlock.dev";
|
|
104
|
+
const { userId, username } = options;
|
|
105
|
+
const { tenancyId } = options;
|
|
106
|
+
const url = new URL(`/${tenancyId}/users/${userId}/passkeys/`, baseUrl);
|
|
107
|
+
const response = yield* fetchNetwork(url, "patch", { userId, username }, {
|
|
108
|
+
headers: authorizationHeaders(options.apiKey),
|
|
109
|
+
});
|
|
110
|
+
const encoded = yield* matchStatus(response, {
|
|
111
|
+
"2xx": (res) => decodeResponseJson(res, PasskeySchemas.UpdatedPasskeys),
|
|
112
|
+
orElse: (res) => decodeResponseJson(res, Schema.Union(NotFoundError, ForbiddenError)),
|
|
113
|
+
});
|
|
114
|
+
return yield* pipe(Match.value(encoded), Match.tag("UpdatedPasskeys", (result) => Effect.succeed(result)), Match.tag("@error/NotFound", (err) => Effect.fail(err)), Match.tag("@error/Forbidden", (err) => Effect.fail(err)), Match.exhaustive);
|
|
115
|
+
}), Effect.catchTags({
|
|
116
|
+
"@error/NetworkPayload": (err) => Effect.die(err),
|
|
117
|
+
"@error/NetworkRequest": (err) => Effect.die(err),
|
|
118
|
+
"@error/NetworkResponse": (err) => Effect.die(err),
|
|
119
|
+
ParseError: (err) => Effect.die(err),
|
|
120
|
+
}), Effect.provide(fetchLayer));
|
|
121
|
+
export const updatePasskeyUsernames = (options, fetchLayer = NetworkFetchLive) => pipe(updateUserPasskeys(options, fetchLayer), Effect.map((result) => result.updated), Effect.map(Array.map((passkey) => {
|
|
122
|
+
return {
|
|
123
|
+
rpId: passkey.credential.rpId,
|
|
124
|
+
userId: passkey.credential.userId,
|
|
125
|
+
username: passkey.credential.username,
|
|
126
|
+
displayName: options.displayName ?? passkey.credential.username,
|
|
127
|
+
};
|
|
128
|
+
})), Effect.map((credentials) => ({
|
|
129
|
+
_tag: "UpdatedPasskeyUsernames",
|
|
130
|
+
credentials,
|
|
131
|
+
})));
|
|
132
|
+
/* List Passkeys */
|
|
133
|
+
export const listPasskeysStream = (options, fetchLayer = NetworkFetchLive) => pipe(Stream.paginateChunkEffect(null, (cursor) => pipe(listPasskeys(cursor ? { ...options, cursor } : options, fetchLayer), Effect.map((result) => [
|
|
134
|
+
Chunk.fromIterable(result.records),
|
|
135
|
+
Option.fromNullable(result.cursor),
|
|
136
|
+
]))));
|
|
137
|
+
export const listPasskeys = (options, fetchLayer = NetworkFetchLive) => pipe(Effect.gen(function* () {
|
|
138
|
+
const baseUrl = options.endpoint ?? "https://api.passlock.dev";
|
|
139
|
+
const { tenancyId } = options;
|
|
140
|
+
const url = new URL(`/${tenancyId}/passkeys/`, baseUrl);
|
|
141
|
+
if (options.cursor) {
|
|
142
|
+
url.searchParams.append("cursor", options.cursor);
|
|
143
|
+
}
|
|
144
|
+
const response = yield* fetchNetwork(url, "get", undefined, {
|
|
145
|
+
headers: authorizationHeaders(options.apiKey),
|
|
146
|
+
});
|
|
147
|
+
const encoded = yield* matchStatus(response, {
|
|
148
|
+
"2xx": (res) => decodeResponseJson(res, FindAllPasskeysSchema),
|
|
149
|
+
orElse: (res) => decodeResponseJson(res, ForbiddenError),
|
|
150
|
+
});
|
|
151
|
+
return yield* pipe(Match.value(encoded), Match.tag("FindAllPasskeys", (data) => Effect.succeed(data)), Match.tag("@error/Forbidden", (err) => Effect.fail(err)), Match.exhaustive);
|
|
152
|
+
}), Effect.catchTags({
|
|
153
|
+
"@error/NetworkPayload": (err) => Effect.die(err),
|
|
154
|
+
"@error/NetworkRequest": (err) => Effect.die(err),
|
|
155
|
+
"@error/NetworkResponse": (err) => Effect.die(err),
|
|
156
|
+
ParseError: (err) => Effect.die(err),
|
|
157
|
+
}), Effect.provide(fetchLayer));
|
|
158
|
+
//# sourceMappingURL=passkey.js.map
|