You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Attach a "tag" to an arbitrary type. This allows you to create distinct types, that aren't assignable to one another, for runtime values that would otherwise have the same type. (See examples.)
@@ -113,19 +111,26 @@ type WillWork = UnwrapOpaque<Tagged<number, 'AccountNumber'>>; // number
Attach a "tag" to an arbitrary type. This allows you to create distinct types, that aren't assignable to one another, for runtime values that would otherwise have the same type. (See examples.)
121
+
Attach a "tag" to an arbitrary type. This allows you to create distinct types, that aren't assignable to one another, for distinct concepts in your program that should not be interchangeable, even if their runtime values have the same type. (See examples.)
124
122
125
123
A type returned by `Tagged` can be passed to `Tagged` again, to create a type with multiple tags.
126
124
127
125
[Read more about tagged types.](https://medium.com/@KevinBGreene/surviving-the-typescript-ecosystem-branding-and-type-tagging-6cf6e516523d)
128
126
127
+
A tag's name is usually a string (and must be a string, number, or symbol), but each application of a tag can also contain an arbitrary type as its "metadata". See {@link GetTagMetadata} for examples and explanation.
128
+
129
+
A type `A` returned by `Tagged` is assignable to another type `B` returned by `Tagged` if and only if:
130
+
- the underlying (untagged) type of `A` is assignable to the underlying type of `B`;
131
+
- `A` contains at least all the tags `B` has;
132
+
- and the metadata type for each of `A`'s tags is assignable to the metadata type of `B`'s corresponding tag.
133
+
129
134
There have been several discussions about adding similar features to TypeScript. Unfortunately, nothing has (yet) moved forward:
@@ -151,21 +156,62 @@ function getMoneyForAccount(accountNumber: AccountNumber): AccountBalance {
151
156
getMoneyForAccount(createAccountNumber());
152
157
153
158
// But this won't, because it has to be explicitly passed as an `AccountNumber` type!
159
+
// Critically, you could not accidentally use an `AccountBalance` as an `AccountNumber`.
154
160
getMoneyForAccount(2);
155
161
156
-
// You can use opaque values like they aren't opaque too.
157
-
const accountNumber = createAccountNumber();
162
+
// You can also use tagged values like their underlying, untagged type.
163
+
// I.e., this will compile successfully because an `AccountNumber` can be used as a regular `number`.
164
+
// In this sense, the underlying base type is not hidden, which differentiates tagged types from opaque types in other languages.
165
+
const accountNumber = createAccountNumber() + 2;
166
+
```
158
167
159
-
// This will compile successfully.
160
-
const newAccountNumber = accountNumber + 2;
168
+
@example
169
+
```
170
+
import type {Tagged} from 'type-fest';
171
+
172
+
// You can apply multiple tags to a type by using `Tagged` repeatedly.
173
+
type Url = Tagged<string, 'URL'>;
174
+
type SpecialCacheKey = Tagged<Url, 'SpecialCacheKey'>;
175
+
176
+
// You can also pass a union of tag names, so this is equivalent to the above, although it doesn't give you the ability to assign distinct metadata to each tag.
177
+
type SpecialCacheKey2 = Tagged<string, 'URL' | 'SpecialCacheKey'>;
Given a type and a tag name, returns the metadata associated with that tag on that type.
186
+
187
+
In the example below, one could use `Tagged<string, 'JSON'>` to represent "a string that is valid JSON". That type might be useful -- for instance, it communicates that the value can be safely passed to `JSON.parse` without it throwing an exception. However, it doesn't indicate what type of value will be produced on parse (which is sometimes known). `JsonOf<T>` solves this; it represents "a string that is valid JSON and that, if parsed, would produce a value of type T". The type T is held in the metadata associated with the `'JSON'` tag.
188
+
189
+
This article explains more about [how tag metadata works and when it can be useful](https://medium.com/@ethanresnick/advanced-typescript-tagged-types-improved-with-type-level-metadata-5072fc125fcf).
190
+
191
+
@example
192
+
```
193
+
import type {Tagged} from 'type-fest';
194
+
195
+
type JsonOf<T> = Tagged<string, 'JSON', T>;
196
+
197
+
function stringify<T>(it: T) {
198
+
return JSON.stringify(it) as JsonOf<T>;
199
+
}
200
+
201
+
function parse<T extends JsonOf<unknown>>(it: T) {
202
+
return JSON.parse(it) as GetTagMetadata<T, 'JSON'>;
203
+
}
204
+
205
+
const x = stringify({ hello: 'world' });
206
+
const parsed = parse(x); // The type of `parsed` is { hello: string }
0 commit comments