-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmod.ts
235 lines (212 loc) · 5.72 KB
/
mod.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
const ERREUR_DATA = Symbol("ERREUR_DATA");
/**
* Readonly version of ErreurStore
*/
export interface TReadonlyErreurStore<Data> {
[ERREUR_DATA]: Data;
has(error: unknown): boolean;
get(error: unknown): Data | undefined;
}
/**
* ErreurStore created by createErreurStore
*/
export interface TErreurStore<Data> extends TReadonlyErreurStore<Data> {
/**
* Readonly version of this store
*/
asReadonly: TReadonlyErreurStore<Data>;
/**
* Mark an error with some data
* @param error the error to mark
* @param data the data to associate with the error
*/
set(error: Error, data: Data): void;
/**
* Mark an error with some data and throw it
* @param error the error to mark, if this is not an instance of Error, it will be converted to an Error
* @param data
*/
setAndThrow(error: unknown, data: Data): never;
/**
* Mark an error with some data and return it
* @param error the error to mark, if this is not an instance of Error, it will be converted to an Error
* @param data
*/
setAndReturn(error: unknown, data: Data): Error;
}
/**
* ErreurStore created by createVoidErreurStore
*/
export interface TVoidErreurStore extends TReadonlyErreurStore<true> {
/**
* Readonly version of this store
*/
asReadonly: TReadonlyErreurStore<true>;
/**
* Mark an error
* @param error
*/
set(error: Error): void;
/**
* Mark an error and throw it
* @param error
*/
setAndThrow(error: unknown): never;
/**
* Mark an error and return it
* @param error
*/
setAndReturn(error: unknown): Error;
}
export type TErreurStoreBase = TErreurStore<unknown> | TVoidErreurStore;
export type TReadableErreurStoreBase = TReadonlyErreurStore<unknown>;
export type TReadableErreurStoreRecord = Record<
string,
TReadableErreurStoreBase
>;
/**
* Create a new ErreurStore, this function expects a generic type to be passed, this type will be the type of the data associated with the error
* @returns a new ErreurStore
*/
export function createErreurStore<Data>(): TErreurStore<Data> {
return createErreurStoreInternal<Data>(undefined);
}
/**
* Create a new VoidErreurStore, this store does not have any data associated with the error
* @returns
*/
export function createVoidErreurStore(): TVoidErreurStore {
return createErreurStoreInternal<true>(true) as TVoidErreurStore;
}
function createErreurStoreInternal<Data>(
// deno-lint-ignore no-explicit-any
defaultValue: any,
): TErreurStore<Data> {
const storage = new WeakMap<Error, Data>();
return {
[ERREUR_DATA]: true as Data,
set,
setAndThrow,
setAndReturn,
has,
get,
asReadonly: {
[ERREUR_DATA]: true as Data,
has,
get,
},
};
function has(error: Error) {
return storage.has(error);
}
function get(error: Error) {
return storage.get(error);
}
function set(error: Error, data: Data = defaultValue) {
if (!(error instanceof Error)) {
throw new Error("Error should be an instance of Error");
}
if (storage.has(error)) {
throw new Error("Error already exists in store");
}
storage.set(error, data);
}
function setAndThrow(error: unknown, data: Data = defaultValue): never {
const err = toError(error);
set(err, data);
throw err;
}
function setAndReturn(error: unknown, data: Data = defaultValue): Error {
const err = toError(error);
set(err, data);
return err;
}
}
/**
* Match the first error in the list stores and return the associated data
* @param error
* @param stores
* @returns
*/
export function matchFirstErreur<
Stores extends readonly TReadableErreurStoreBase[],
>(
error: unknown,
stores: Stores,
): Stores[number][typeof ERREUR_DATA] | undefined {
if (!(error instanceof Error)) {
return undefined;
}
for (const store of stores) {
if (store.has(error)) {
return store.get(error) as Stores[number][typeof ERREUR_DATA];
}
}
return undefined;
}
export type TMatchResult<StoresRec extends TReadableErreurStoreRecord> = {
[K in keyof StoresRec]: StoresRec[K][typeof ERREUR_DATA] | undefined;
};
/**
* Return an object with the data associated with the error for each store (if any)
* @param error
* @param stores
* @returns
*/
export function matchErreurs<StoresRec extends TReadableErreurStoreRecord>(
error: Error,
stores: StoresRec,
): { [K in keyof StoresRec]: StoresRec[K][typeof ERREUR_DATA] | undefined } {
const result: TMatchResult<StoresRec> = {} as TMatchResult<StoresRec>;
if (!(error instanceof Error)) {
for (const key in stores) {
result[key] = undefined;
}
return result;
}
for (const key in stores) {
result[key] = stores[key].get(error);
}
return result;
}
/**
* Convert any value to an Error
* @param value
* @returns
*/
export function toError(value: unknown): Error {
if (value instanceof Error) {
return value;
}
if (typeof value === "string") {
return new Error(value);
}
if (value === null) {
return new Error("ErrorFromNull");
}
if (value === undefined) {
return new Error("ErrorFromUndefined");
}
if (typeof value === "object" && value !== null) {
// object
const message = `ErrorFromObject: ${objectToMessage(value)}`;
const cause = "cause" in value && value.cause instanceof Error
? value.cause
: undefined;
return new Error(message, { cause });
}
const typeCapitalized = (typeof value).charAt(0).toUpperCase() +
(typeof value).slice(1);
const message = `ErrorFrom${typeCapitalized}: ${String(value)}`;
return new Error(message);
}
function objectToMessage(obj: object): string {
if ("message" in obj && typeof obj.message === "string") {
return obj.message;
}
try {
return JSON.stringify(obj);
} catch (_error) {
return `[object]`;
}
}