# FAQ
- How do I change a column name in the database?
- How can I set value of default some function, for example
NOW()
? - How to do validation?
- What does "owner side" in relations mean or why we need to put
@JoinColumn
and@JoinTable
decorators? - How do I add extra columns into many-to-many (junction) table?
- How to use TypeORM with dependency injection tool?
- How to handle outDir TypeScript compiler option?
- How to use TypeORM with ts-node?
- How to use Webpack for the backend
- How to use TypeORM in ESM projects?
# How do I update a database schema?
One of the main responsibilities of TypeORM is to keep your database tables in sync with your entities. There are two ways that help you achieve this:
Use
synchronize: true
in data source options:import { DataSource } from "typeorm" const myDataSource = new DataSource({ // ... synchronize: true, })
1
2
3
4
5
6This option automatically syncs your database tables with the given entities each time you run this code. This option is perfect during development, but in production you may not want this option to be enabled.
Use command line tools and run schema sync manually in the command line:
typeorm schema:sync
1This command will execute schema synchronization.
Schema sync is extremely fast. If you are considering to disable synchronize option during development because of performance issues, first check how fast it is.
# How do I change a column name in the database?
By default, column names are generated from property names.
You can simply change it by specifying a name
column option:
@Column({ name: "is_active" })
isActive: boolean;
2
# How can I set the default value to some function, for example NOW()
?
default
column option supports a function.
If you are passing a function which returns a string,
it will use that string as a default value without escaping it.
For example:
@Column({ default: () => "NOW()" })
date: Date;
2
# How to do validation?
Validation is not part of TypeORM because validation is a separate process not really related to what TypeORM does. If you want to use validation use class-validator (opens new window) - it works perfectly with TypeORM.
# What does "owner side" in a relations mean or why we need to use @JoinColumn
and @JoinTable
?
Let's start with one-to-one
relation.
Let's say we have two entities: User
and Photo
:
@Entity()
export class User {
@PrimaryGeneratedColumn()
id: number
@Column()
name: string
@OneToOne()
photo: Photo
}
2
3
4
5
6
7
8
9
10
11
@Entity()
export class Photo {
@PrimaryGeneratedColumn()
id: number
@Column()
url: string
@OneToOne()
user: User
}
2
3
4
5
6
7
8
9
10
11
This example does not have a @JoinColumn
which is incorrect.
Why? Because to make a real relation, we need to create a column in the database.
We need to create a column userId
in photo
or photoId
in user
.
But which column should be created - userId
or photoId
?
TypeORM cannot decide for you.
To make a decision, you must use @JoinColumn
on one of the sides.
If you put @JoinColumn
in Photo
then a column called userId
will be created in the photo
table.
If you put @JoinColumn
in User
then a column called photoId
will be created in the user
table.
The side with @JoinColumn
will be called the "owner side of the relationship".
The other side of the relation, without @JoinColumn
, is called the "inverse (non-owner) side of relationship".
It is the same in a @ManyToMany
relation. You use @JoinTable
to show the owner side of the relation.
In @ManyToOne
or @OneToMany
relations, @JoinColumn
is not necessary because
both decorators are different, and the table where you put the @ManyToOne
decorator will have the relational column.
@JoinColumn
and @JoinTable
decorators can also be used to specify additional
join column / junction table settings, like join column name or junction table name.
# How do I add extra columns into many-to-many (junction) table?
It's not possible to add extra columns into a table created by a many-to-many relation. You'll need to create a separate entity and bind it using two many-to-one relations with the target entities (the effect will be same as creating a many-to-many table), and add extra columns in there. You can read more about this in Many-to-Many relations.
# How to handle outDir TypeScript compiler option?
When you are using the outDir
compiler option, don't forget to copy assets and resources your app is using into the output directory.
Otherwise, make sure to setup correct paths to those assets.
One important thing to know is that when you remove or move entities, the old entities are left untouched inside the output directory.
For example, you create a Post
entity and rename it to Blog
,
you no longer have Post.ts
in your project. However, Post.js
is left inside the output directory.
Now, when TypeORM reads entities from your output directory, it sees two entities - Post
and Blog
.
This may be a source of bugs.
That's why when you remove and move entities with outDir
enabled, it's strongly recommended to remove your output directory and recompile the project again.
# How to use TypeORM with ts-node?
You can prevent compiling files each time using ts-node (opens new window).
If you are using ts-node, you can specify ts
entities inside data source options:
{
entities: ["src/entity/*.ts"],
subscribers: ["src/subscriber/*.ts"]
}
2
3
4
Also, if you are compiling js files into the same folder where your typescript files are,
make sure to use the outDir
compiler option to prevent
this issue (opens new window).
Also, if you want to use the ts-node CLI, you can execute TypeORM the following way:
npx typeorm-ts-node-commonjs schema:sync
For ESM projects use this instead:
npx typeorm-ts-node-esm schema:sync
# How to use Webpack for the backend?
Webpack produces warnings due to what it views as missing require statements -- require statements for all drivers supported by TypeORM. To suppress these warnings for unused drivers, you will need to edit your webpack config file.
const FilterWarningsPlugin = require('webpack-filter-warnings-plugin');
module.exports = {
...
plugins: [
//ignore the drivers you don't want. This is the complete list of all drivers -- remove the suppressions for drivers you want to use.
new FilterWarningsPlugin({
exclude: [/mongodb/, /mssql/, /mysql/, /mysql2/, /oracledb/, /pg/, /pg-native/, /pg-query-stream/, /react-native-sqlite-storage/, /redis/, /sqlite3/, /sql.js/, /typeorm-aurora-data-api-driver/]
})
]
};
2
3
4
5
6
7
8
9
10
11
# Bundling Migration Files
By default Webpack tries to bundle everything into one file. This can be problematic when your project has migration files which are meant to be executed after bundled code is deployed to production. To make sure all your migrations can be recognized and executed by TypeORM, you may need to use "Object Syntax" for the entry
configuration for the migration files only.
const glob = require("glob")
const path = require("path")
module.exports = {
// ... your webpack configurations here...
// Dynamically generate a `{ [name]: sourceFileName }` map for the `entry` option
// change `src/db/migrations` to the relative path to your migration folder
entry: glob
.sync(path.resolve("src/db/migrations/*.ts"))
.reduce((entries, filename) => {
const migrationName = path.basename(filename, ".ts")
return Object.assign({}, entries, {
[migrationName]: filename,
})
}, {}),
resolve: {
// assuming all your migration files are written in TypeScript
extensions: [".ts"],
},
output: {
// change `path` to where you want to put transpiled migration files.
path: __dirname + "/dist/db/migrations",
// this is important - we want UMD (Universal Module Definition) for migration files.
libraryTarget: "umd",
filename: "[name].js",
},
}
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
Also, since Webpack 4, when using mode: 'production'
, files are optimized by default which includes mangling your code in order to minimize file sizes. This breaks the migrations because TypeORM relies on their names to determine which has already been executed. You may disable minimization completely by adding:
module.exports = {
// ... other Webpack configurations here
optimization: {
minimize: false,
},
}
2
3
4
5
6
Alternatively, if you are using the UglifyJsPlugin
, you can tell it to not change class or function names like so:
const UglifyJsPlugin = require("uglifyjs-webpack-plugin")
module.exports = {
// ... other Webpack configurations here
optimization: {
minimizer: [
new UglifyJsPlugin({
uglifyOptions: {
keep_classnames: true,
keep_fnames: true,
},
}),
],
},
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Lastly, make sure in your data source options, the transpiled migration files are included:
// TypeORM Configurations
module.exports = {
// ...
migrations: [
// this is the relative path to the transpiled migration files in production
"db/migrations/**/*.js",
// your source migration files, used in development mode
"src/db/migrations/**/*.ts",
],
}
2
3
4
5
6
7
8
9
10
# How to use TypeORM in ESM projects?
Make sure to add "type": "module"
in the package.json
of your project so TypeORM will know to use import( ... )
on files.
To avoid circular dependency import issues use the Relation
wrapper type for relation type definitions in entities:
@Entity()
export class User {
@OneToOne(() => Profile, (profile) => profile.user)
profile: Relation<Profile>
}
2
3
4
5
Doing this prevents the type of the property from being saved in the transpiled code in the property metadata, preventing circular dependency issues.
Since the type of the column is already defined using the @OneToOne
decorator, there's no use of the additional type metadata saved by TypeScript.
Important: Do not use
Relation
on non-relation column types