|
| 1 | +// Forked from unjs/defu (MIT) |
| 2 | +export namespace MergeConfig { |
| 3 | + export type Input = Record<string | number | symbol, any>; |
| 4 | + export type IgnoredInput = boolean | number | null | any[] | Record<never, any> | undefined; |
| 5 | + |
| 6 | + export type Merger = <T extends Input, K extends keyof T>(object: T, key: keyof T, value: T[K], namespace: string) => any; |
| 7 | + |
| 8 | + type nullish = null | undefined | void; |
| 9 | + |
| 10 | + export type MergeObjects<Destination extends Input, Defaults extends Input> = Destination extends Defaults |
| 11 | + ? Destination |
| 12 | + : Omit<Destination, keyof Destination & keyof Defaults> & |
| 13 | + Omit<Defaults, keyof Destination & keyof Defaults> & { |
| 14 | + -readonly [Key in keyof Destination & keyof Defaults]: Destination[Key] extends nullish |
| 15 | + ? Defaults[Key] extends nullish |
| 16 | + ? nullish |
| 17 | + : Defaults[Key] |
| 18 | + : Defaults[Key] extends nullish |
| 19 | + ? Destination[Key] |
| 20 | + : Merge<Destination[Key], Defaults[Key]>; // eslint-disable-line no-use-before-define |
| 21 | + }; |
| 22 | + |
| 23 | + export type Merged<S extends Input, D extends Array<Input | IgnoredInput>> = D extends [infer F, ...infer Rest] |
| 24 | + ? F extends Input |
| 25 | + ? Rest extends Array<Input | IgnoredInput> |
| 26 | + ? Merged<MergeObjects<S, F>, Rest> |
| 27 | + : MergeObjects<S, F> |
| 28 | + : F extends IgnoredInput |
| 29 | + ? Rest extends Array<Input | IgnoredInput> |
| 30 | + ? Merged<S, Rest> |
| 31 | + : S |
| 32 | + : S |
| 33 | + : S; |
| 34 | + |
| 35 | + export type MergeFn = <Source extends Input, Defaults extends Array<Input | IgnoredInput>>(source: Source, ...defaults: Defaults) => Merged<Source, Defaults>; |
| 36 | + |
| 37 | + export interface Instance extends MergeFn { |
| 38 | + // fn: MergeFn; |
| 39 | + // arrayFn: MergeFn; |
| 40 | + create: (merger?: Merger) => MergeFn; |
| 41 | + } |
| 42 | + |
| 43 | + export type MergeArrays<Destination, Source> = |
| 44 | + Destination extends Array<infer DestinationType> |
| 45 | + ? Source extends Array<infer SourceType> |
| 46 | + ? Array<DestinationType | SourceType> |
| 47 | + : Source | Array<DestinationType> |
| 48 | + : Source | Destination; |
| 49 | + |
| 50 | + export type Merge<Destination extends Input, Defaults extends Input> = |
| 51 | + // Remove explicitly null types |
| 52 | + Destination extends nullish |
| 53 | + ? Defaults extends nullish |
| 54 | + ? nullish |
| 55 | + : Defaults |
| 56 | + : Defaults extends nullish |
| 57 | + ? Destination |
| 58 | + : // Handle arrays |
| 59 | + Destination extends Array<any> |
| 60 | + ? Defaults extends Array<any> |
| 61 | + ? MergeArrays<Destination, Defaults> |
| 62 | + : Destination | Defaults |
| 63 | + : // Don't attempt to merge Functions, RegExps, Promises |
| 64 | + // eslint-disable-next-line @typescript-eslint/ban-types |
| 65 | + Destination extends (...args: any) => any |
| 66 | + ? Destination | Defaults |
| 67 | + : Destination extends RegExp |
| 68 | + ? Destination | Defaults |
| 69 | + : Destination extends Promise<any> |
| 70 | + ? Destination | Defaults |
| 71 | + : // Don't attempt to merge Functions, RegExps, Promises |
| 72 | + // eslint-disable-next-line @typescript-eslint/ban-types |
| 73 | + Defaults extends (...args: any) => any |
| 74 | + ? Destination | Defaults |
| 75 | + : Defaults extends RegExp |
| 76 | + ? Destination | Defaults |
| 77 | + : Defaults extends Promise<any> |
| 78 | + ? Destination | Defaults |
| 79 | + : // Ensure we only merge Records |
| 80 | + Destination extends Input |
| 81 | + ? Defaults extends Input |
| 82 | + ? MergeObjects<Destination, Defaults> |
| 83 | + : Destination | Defaults |
| 84 | + : Destination | Defaults; |
| 85 | +} |
0 commit comments