Modern Typescript oriented libraries start to use classes and decorators in their APIs.
Everything is awesome until libraries start to use reflect-metadata API which enforce you to define business logic in the static types which are magically reflected into your runtime code.
TLDR:
Don't vendor lock yourself with unsupported experimental syntax and
don't use reflect-metadata
which forces you to pre-process your runtime code.
Use raw vanilla Javascript and infer Typescript data types directly from the Javascript definitions.
Good code
const userModel = model({
id: nonNullable(stringType())
name: nonNullable(stringType())
})
Bad code
@Model()
class userModel
/* decorators are nice syntax sugar ^^ */
@Field()
id: string /* problem is that business logic is coded in typescript type here */
/* decorators are nice syntax sugar ^^ */
@Field()
name: string /* problem is that business logic is coded in typescript type here */
}
Check full working example of the good code in the Typescript playground
And... what is reflect-metadata
?
Before we dig deeper to reflect-metadata we need to understand what are decorators Typescript decorators API.
Decorators
Decorators are syntax sugar which gives us the option to write quasi high-order-functions
to enhance classes
, methods
, and properties
.
class ExampleClass {
@first() // decorators
@second() // decorators
method() {}
}
You may know a similar pattern from languages like C#
, Java
or Python
.
If you compare Typescript decorators to the Python implementation,
you can find the difference that Typescript implementation does not work for basic functions
or arrow functions
.
At the top of it, the decorators are only a Typescript specific feature.
But we have to pay attention because similar functionality is already in the tc39 Javascript proposal at stage 2.
reflect-metadata
That was decorators, now we have to look for the reflect-metadata library.
Let's check the documentation.
Background
- Decorators add the ability to augment a class and its members as the class is defined, through a declarative syntax.
- Traceur attaches annotations to a static property on the class.
- Languages like C# (.NET), and Java support attributes or annotations that add metadata to types, along with a reflective API for reading metadata.
If you don't fully understand who will use it in the real world you can check some libraries which use reflect-metadata
to define the applications data models.
- type-orm (~24K Github stars)
- type-graphql (~6K Github stars)
- nest.js (~37K Github Stars)
- and so on...
If you know these libraries you know what I'm talking about.
Thanks to the reflect-metadata
library you can "hack" into the Typescript compiler and get the static type metadata from compile-time into your Javascript runtime.
For example, you may have code like:
@ObjectType()
class Recipe {
@Field()
title: string;
}
The reflect-metadata
library enables us to write decorators that will read metadata from the static type and this metadata may affect your Javascript runtime code.
You may imagine this metadata as an information that field title is string
.
So that's pretty handy syntax sugar!
Yes...
But actually...
No... There is another side of the same coin.
Let's check on how to define an SQL table via the type-orm
library using decorators and reflect-metadata
.
@Entity()
export class User {
@PrimaryGeneratedColumn()
id: number;
@Column()
firstName: string;
}
As you may see, there is no Javascript runtime information about the data types of columns.
So that's magic because the basic Typescript compiler should transpile code into this:
@Entity()
export class User {
@PrimaryGeneratedColumn()
id;
@Column()
firstName;
}
The default Typescript compiler removes information about data types. Thanks to reflect-metadata
and "emitDecoratorMetadata": true
this code is still working
since it transfers information about static types into the runtime metadata descriptor which can be read in the Javascript runtime.
And where is the problem?
In my humble opinion the whole philosophy of influencing Javascript runtime via static types is bad and we should not use it in the Javascript ecosystem!
The reflect-metadata
library has to influence the Typescript compiler and forces us to vendor lock our code into Typescript specific syntax so we're no longer able to use raw vanilla Javascript. The beauty of standard Typescript is that it just enhances the real Javascript codebase and enables us to have better integration, stability and documentation.
If some typings do not work correctly we can just use as any
, @ts-expect-error
or @ts-ignore
, and everything is okay. We don't need to bend our application in the name of strict-type-safe-only faith. The strongest type-system advantage of Typescript over the others is that Typescript is just a tool for developers and it does not optimize the runtime.
If you define a variable in the C language, you know how many bits will be allocated in the memory thanks to the definition of a data-type.
At first glance, it could look like Typescript is missing this kind of optimization but on the other hand we should also realise that THIS is the game changer!
It enables us to just use a type system to help us document code and avoid runtime errors with the best developer experience.
If you combine this philosophy with Typescript type inferring you get the greatest dev-tool for avoiding runtime errors which is not affecting Javascript code.
If you're more interested in some fancy usage of Typescript type inference which solves real-world problems, you can check my other articles.
- World-first Static time RegEx engine with O(0) time complexity
- React typed state management under 10 lines of code
- Type inferred react-redux under 20 lines
- and so on...
Reflect-metadata vs single source of truth (SSOT)?
If you use libraries like typed-graphql
or type-orm
you can find that reflect-metadata
is only working for basic data types like: number
, string
, and boolean
.
If you want to refer to another data type, you have to create a real Javascript pointer reference.
There are some real-world examples where you can see that the code is "duplicated" and you have to define real Javascript reference and static type reference.
It mean that you do not follow SSOT (Single source of truth) and DRY (Don't repeat yourself) at all.
type-orm example
(you should read comments in the code snippet)
@Entity()
export class PhotoMetadata {
// here you have to define a reference into the real runtime Javascript pointer
@OneToOne(type => Photo)
@JoinColumn()
// here you duplicate the pointer into Photo just to have proper static types
photo: Photo;
}
type-graphql example
(you should read comments in the code snippet)
@InputType()
class NewRecipeInput {
// here you have to define a reference into the real runtime Javascript pointer
@Field(type => [String])
@ArrayMaxSize(30)
// here you duplicate the pointer into Photo just to have proper static types
// so that means you can have an inconsistency between the static type and @Field(...) definition
ingredients: string[];
}
Our target is to have SSOT which describes our data types and give us
- Static type inferring
- Infer cyclic pointer references
- Option to have runtime Javascript validations
- Type-safety
- Good documentation
- Enable us to use standard Javascript tooling
- Enable us to generate the schema in the runtime
The solution
So we have explained why using reflect-metadata
suc*s...so what should we use instead?
Thanks to Typescript generics we're able to write data types as a Javascript function composition or just simple hash-map
/object
.
Then we can Infer the data types. Thanks to that our code is pure Javascript, we're able to be more flexible and generate data types on the fly and not be fixed.
JSON Schema vs Class-based schema
In the previous examples we used class to define the schema, now we'll use a simple Javascript hashmap.
So let's define some basic ones.
const mySchema = {
type: 'object' as const,
properties: {
key1: {
type: 'number' as const,
required: true as const,
},
key2: {
type: 'string' as const,
required: false as const,
},
},
required: false as const,
}
The only Typescript-specific code there is the as const
notation which defines that the data type should have been the same as the value.
We're able to write a data type for a schema like this:
export type SchemaArr = {
type: 'array'
required?: boolean
items: Schema
}
export type SchemaObject = {
type: 'object'
required?: boolean
properties: Record<string, Schema>
}
type SchemaBoolean = {
type: 'boolean'
required?: boolean
}
type SchemaString = {
type: 'string'
required?: boolean
}
type SchemaNumber = {
type: 'number'
required?: boolean
}
export type Schema = SchemaArr | SchemaObject | SchemaString | SchemaNumber | SchemaBoolean
Let's go deeper, Infer type from the Javascript schema!
Now we can create a generic which extracts the data type from the schema definition.
type NiceMerge<T, U, T0 = T & U, T1 = { [K in keyof T0]: T0[K] }> = T1
type MakeOptional<T, Required extends boolean> = Required extends true ? T : T | undefined
export type InferSchemaType<T extends Schema> = T extends {
type: 'object'
properties: infer U
}
? // @ts-expect-error
{ [K in keyof U]: InferSchemaType<U[K]> }
: T extends { type: 'array'; items: any }
? // @ts-expect-error
MakeOptional<InferSchemaType<T['items']>[], T['required']>
: T extends { type: 'boolean' }
? // @ts-expect-error
MakeOptional<boolean, T['required']>
: T extends { type: 'string' }
? // @ts-expect-error
MakeOptional<string, T['required']>
: T extends { type: 'number' }
? // @ts-expect-error
MakeOptional<number, T['required']>
: never
For simplicity I will not be describing how the InferSchemaType<T>
generic was crafted. If you want to know more, just mention me below in the comment section.
This generic is kinda more complicated but if we look at the result we can see that the generics work perfectly.
type MySchemaType = InferSchemaType<typeof mySchema>
Or we can create builder util functions which build JSON with the nicer API.
You can check full source code here
This is phenomenal code to define a schema and infer a type from it.
It's very strong because it enables us to just write simple raw Javascript and 100% of static types are inferred via a few generics and functions.
At the end...
Thanks to omitting experimental Typescript API and returning into good old Javascript we don't vendor-lock our code into the Typescript compiler.
Validators
Even if we want to have runtime-validations, it's super easy to write a runtime validator on top of this schema definition.
If you're more interested in how to write validation from schema you can check the source code on my Github
https://github.com/Svehla/reflect-metadata-SCKS/blob/master/index.ts
Use High-order-functions vs Decorators API
But what if you just like decorators and you want to enhance some functions?
Decorators are just syntax-sugar. We can program the same in raw Javascript with a few TS generics.
Decoration API
class ExampleClass {
@first() // decorator
@second() // decorator
method() {
console.log('hi')
}
}
vs
HOF (high-order-function) API
Raw Javascript
// these two examples are not the same because
// 1. the second one will instance the method function code every time that class is instanced
// 2. there is different `this` binding
// but we'll ignore that small difference and we'll focus on different stuff...
const fn1 = first()(second()((self) => {
console.log('hi')
})))
with usage of Ramda.js library
import * as R from 'ramda'
const fn1 = R.pipe(
second(),
first()
)(self => {
console.log('hi')
})
If you want to see more about how to add proper types for HOC or the Pipe function just tell me down in the comment section.
Who should care about this article the most?
The problem is not with the regular programmers who just install npm libraries.
The problem is the authors of libraries who think that this new reflect-metadata
API with experimental decorators will save the world, but at the opposite side it just vendor locks your codebase into 1 edge-case technology.
Is there some good library too?
Haha! good question, of course there is.
I picked two libraries which uses the same philosophy as we described in this article.
1. Typed-env-parser
Typed env parser - npm.
Typed env parser - github.
If you look for the API:
You can find that the definition of users does not include Typescript and the API of the function is pure Javascript.
Thanks to the type inference we get all the features of a strongly typed system in vanilla js implementation.
2. Yup
Yup enable us to define JS schema and infer its data type from raw Javascript schema.
Well That's all...
I hope that you find time & energy to read whole article with a clear and open mind.
Try to think about the syntax which you may use in your codebase on the daily basis and be sceptical about new fancy stuff, which enforces you to do extra compilation to make the code work...
If you enjoyed reading the article don’t forget to like it to tell me if it makes sense to continue.
Top comments (11)
I landed here cause vitest doesn't support 'reflect-metadata' #$@$!@$!#%!@%.
Thank you for the details. I guess I will need another testing framework to run my type-graphql & mikrorm app
PS:I realize every day that java had reflection right.
This is great. I think Google suggested this article to me because I've been trying to figure out why I need a class in order to generate Swagger docs for my APIs which use TS interfaces. I wish I could tap into the TS compiler with decorators on my controller FUNCTIONs as I don't see the benefit of using a CLASS other than 'thats what you need to do'.
exactly! I love the minimalism of
express
andnodejs
/javascript
ecosystem... So I don't see the purpose to have classes and methods with tons of boilerplate to programme simple typed REST-API endpoints...Another great example that followed your approach is mobx, the had decorator syntax added and made the default years ago when they were certain it'd come into ecma and then... It never did, last November they removed it (.. Kinda) and went for a purely js approach - kinda glad Google managed to suggest this post to me as I've recently been looking into decorators for TS as the DX is great and I'd like to get some use out of them 😅
thank you for a feedback :) I'm happy that you enjoy the article
I think you are missing the point here. The whole idea of TypeScript and Reflect is to bring JS world closer to high level programming languages, like Java and C#.
Yeah, sure, you can do it as close to native JS as possible, but then again, why bother with TS? Your basic argument is not against reflect-metadata, but whole concept of using typescript. Yeah, you might be vendor-locking yourself, but you do so too, whenever you use any framework for development. You are locked with that framework for the rest of the project, unless you expect some major refactoring.
So tell me - whoever read this comment after this article - do you want to trade code clarity to stay "pure"? Just because you can achieve same thing harder in native way, does it mean, you have to?
For me, it's just waste of time and energy.
What if you don't understand decorators, reflections and aspect programming? Don't worry, in 90% cases you don't have to.
You're right in one thing... C# & Java sucks as well
I think that you absolutely miss understand the whole point of this article... i can't imagine to write out more code without typescript and I think that TS type system is the best type system ever because you can infer almonost all types from your pure javascript code and keep 100% type safeness. Imho there is only one similar good type system and its from F# compiler. i do not blame TS, i love it... but my code looks like JS and works like TS <3 its 2 in 1, win win situation
BTW reflect-metadata is not as powerfull as the type inferring system for more complicated data types like structures etc...
its reason why you have to define pointer into
Photo
two times in this example:I was searching for why I can't use tsx (the esbuild based typescript execute) in replace of ts-node, found their doc and this article. I guess I have to stick with ts-node for now as my project relies on typedi and routing-controllers which are basically built around reflect-metada.
Hi Jakub,
This is a great article. Thank you for all the insights. I have two questions for you. Is it possible to use it without decorators and reflect-metadata. If not, do you know any alternative library that does the same job without decorators?
Thanks.
Hey Jakub,
I just wanted to thank you for this article. It was a true eye opener. I was just digging myself deep into decorators and reflect-metadata. I had a weird feeling throughout that this is all wrong somehow.
I was searching for documentation on Reflect when Google fed me your article and you expressed precisely what to me was still just a hunch. I was shouting YES all the way through and ended my brief affair with Reflect instantly.
I also rediscovered yup, which I used in a project a few years back and found it matured into a polished library. I'm currently hosting a pool party with yup, Typescript and Wallaby and it's epic!
Godspeed and keep up the great writing!
Haha!
I hope that there will be more people with the same mindset as you have!
Thanks for the feedback! I appreciate it a lot! 🥰 😄