Skip to content

Commit 84269a7

Browse files
zorjisindresorhus
andauthoredDec 13, 2022
Exact: Fix handling of Opaque types (#525)
Co-authored-by: Sindre Sorhus <sindresorhus@gmail.com>
1 parent 8eb7bf2 commit 84269a7

File tree

2 files changed

+43
-3
lines changed

2 files changed

+43
-3
lines changed
 

‎source/exact.d.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import type {KeysOfUnion, ArrayElement, ObjectValue} from './internal';
2+
import type {Opaque} from './opaque';
23

34
/**
45
Create a type from `ParameterType` and `InputType` and change keys exclusive to `InputType` to `never`.
@@ -53,5 +54,7 @@ export type Exact<ParameterType, InputType> =
5354
ParameterType extends unknown[] ? Array<Exact<ArrayElement<ParameterType>, ArrayElement<InputType>>>
5455
// In TypeScript, Array is a subtype of ReadonlyArray, so always test Array before ReadonlyArray.
5556
: ParameterType extends readonly unknown[] ? ReadonlyArray<Exact<ArrayElement<ParameterType>, ArrayElement<InputType>>>
56-
: ParameterType extends object ? ExactObject<ParameterType, InputType>
57-
: ParameterType;
57+
// For Opaque types, internal details are hidden from public, so let's leave it as is.
58+
: ParameterType extends Opaque<infer OpaqueType, infer OpaqueToken> ? ParameterType
59+
: ParameterType extends object ? ExactObject<ParameterType, InputType>
60+
: ParameterType;

‎test-d/exact.ts

+38-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import type {Exact} from '../index';
1+
import type {Exact, Opaque} from '../index';
22

33
{ // Spec - string type
44
type Type = string;
@@ -334,3 +334,40 @@ import type {Exact} from '../index';
334334
fn(input);
335335
}
336336
}
337+
338+
// Spec - special test case for Opaque type
339+
// @see https://github.com/sindresorhus/type-fest/issues/508
340+
{
341+
type SpecialName = Opaque<string, 'special name'>;
342+
343+
type OnlyAcceptName = {
344+
name: SpecialName;
345+
};
346+
347+
const onlyAcceptNameImproved = <T extends Exact<OnlyAcceptName, T>>(arguments_: T) => arguments_;
348+
349+
onlyAcceptNameImproved({
350+
// The error before the workaround:
351+
// Error: Type 'SpecialName' is not assignable to type 'never'
352+
name: 'name' as SpecialName,
353+
});
354+
}
355+
356+
// Spec - special test case for Opaque type
357+
// @see https://github.com/sindresorhus/type-fest/issues/508
358+
{
359+
// Test for number Opaque type
360+
type SpecialName = Opaque<number, 'special name'>;
361+
362+
type OnlyAcceptName = {
363+
name: SpecialName;
364+
};
365+
366+
const fn = <T extends Exact<OnlyAcceptName, T>>(arguments_: T) => arguments_;
367+
368+
fn({
369+
// The error before the workaround:
370+
// Error: Type 'SpecialName' is not assignable to type 'never'
371+
name: 1 as SpecialName,
372+
});
373+
}

0 commit comments

Comments
 (0)
Please sign in to comment.