Interview

10 TypeORM Interview Questions and Answers

Prepare for your next technical interview with this guide on TypeORM, featuring common questions and answers to enhance your database management skills.

TypeORM is a powerful and flexible Object-Relational Mapper (ORM) for TypeScript and JavaScript. It simplifies database interactions by allowing developers to work with database entities as if they were regular TypeScript or JavaScript objects. TypeORM supports multiple database systems, including MySQL, PostgreSQL, SQLite, and more, making it a versatile choice for various projects.

This article provides a curated selection of TypeORM interview questions designed to help you demonstrate your proficiency and understanding of this ORM. By reviewing these questions and their answers, you can better prepare for technical interviews and showcase your ability to effectively manage database operations using TypeORM.

TypeORM Interview Questions and Answers

1. Write a basic entity definition for a
User
User with fields
id
id,
name
name, and
email
email.

To define a basic entity in TypeORM for a

User
User with fields
id
id,
name
name, and
email
email, you can use the following code snippet. This example uses TypeScript, which is commonly used with TypeORM.

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';
@Entity()
export class User {
@PrimaryGeneratedColumn()
id: number;
@Column()
name: string;
@Column()
email: string;
}
import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm'; @Entity() export class User { @PrimaryGeneratedColumn() id: number; @Column() name: string; @Column() email: string; }
import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';

@Entity()
export class User {
    @PrimaryGeneratedColumn()
    id: number;

    @Column()
    name: string;

    @Column()
    email: string;
}

2. How do you establish a one-to-many relationship between two entities? Provide an example.

In TypeORM, a one-to-many relationship is established between two entities when one entity can be associated with multiple instances of another entity. This is typically done using the

@OneToMany
@OneToMany and
@ManyToOne
@ManyToOne decorators. The
@OneToMany
@OneToMany decorator is used on the entity that holds the collection, while the
@ManyToOne
@ManyToOne decorator is used on the entity that holds the reference to the single instance.

Example:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne } from 'typeorm';
@Entity()
export class User {
@PrimaryGeneratedColumn()
id: number;
@Column()
name: string;
@OneToMany(() => Post, post => post.user)
posts: Post[];
}
@Entity()
export class Post {
@PrimaryGeneratedColumn()
id: number;
@Column()
title: string;
@ManyToOne(() => User, user => user.posts)
user: User;
}
import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne } from 'typeorm'; @Entity() export class User { @PrimaryGeneratedColumn() id: number; @Column() name: string; @OneToMany(() => Post, post => post.user) posts: Post[]; } @Entity() export class Post { @PrimaryGeneratedColumn() id: number; @Column() title: string; @ManyToOne(() => User, user => user.posts) user: User; }
import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne } from 'typeorm';

@Entity()
export class User {
    @PrimaryGeneratedColumn()
    id: number;

    @Column()
    name: string;

    @OneToMany(() => Post, post => post.user)
    posts: Post[];
}

@Entity()
export class Post {
    @PrimaryGeneratedColumn()
    id: number;

    @Column()
    title: string;

    @ManyToOne(() => User, user => user.posts)
    user: User;
}

In this example, the

User
User entity has a one-to-many relationship with the
Post
Post entity. The
User
User entity uses the
@OneToMany
@OneToMany decorator to indicate that it can have multiple posts, while the
Post
Post entity uses the
@ManyToOne
@ManyToOne decorator to indicate that each post belongs to a single user.

3. Write a query to find all users whose name starts with ‘A’.

To find all users whose name starts with ‘A’ using TypeORM, you can utilize the

find
find method along with a
where
where clause.

Here is an example of how you can achieve this:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { getRepository } from 'typeorm';
import { User } from './entity/User';
async function findUsersStartingWithA() {
const userRepository = getRepository(User);
const users = await userRepository.find({
where: {
name: Like('A%')
}
});
return users;
}
import { getRepository } from 'typeorm'; import { User } from './entity/User'; async function findUsersStartingWithA() { const userRepository = getRepository(User); const users = await userRepository.find({ where: { name: Like('A%') } }); return users; }
import { getRepository } from 'typeorm';
import { User } from './entity/User';

async function findUsersStartingWithA() {
    const userRepository = getRepository(User);
    const users = await userRepository.find({
        where: {
            name: Like('A%')
        }
    });
    return users;
}

In this example, the

Like
Like operator is used to match any user whose name starts with ‘A’. The
getRepository
getRepository function is used to get the repository for the
User
User entity, and the
find
find method is used to perform the query.

4. Write a migration script to add a new column
age
age to the
User
User entity.

To add a new column

age
age to the
User
User entity, you would create a migration script. Below is an example of how to achieve this:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { MigrationInterface, QueryRunner, TableColumn } from "typeorm";
export class AddAgeColumnToUser1634567890123 implements MigrationInterface {
public async up(queryRunner: QueryRunner): Promise<void> {
await queryRunner.addColumn("user", new TableColumn({
name: "age",
type: "int",
isNullable: true
}));
}
public async down(queryRunner: QueryRunner): Promise<void> {
await queryRunner.dropColumn("user", "age");
}
}
import { MigrationInterface, QueryRunner, TableColumn } from "typeorm"; export class AddAgeColumnToUser1634567890123 implements MigrationInterface { public async up(queryRunner: QueryRunner): Promise<void> { await queryRunner.addColumn("user", new TableColumn({ name: "age", type: "int", isNullable: true })); } public async down(queryRunner: QueryRunner): Promise<void> { await queryRunner.dropColumn("user", "age"); } }
import { MigrationInterface, QueryRunner, TableColumn } from "typeorm";

export class AddAgeColumnToUser1634567890123 implements MigrationInterface {
    public async up(queryRunner: QueryRunner): Promise<void> {
        await queryRunner.addColumn("user", new TableColumn({
            name: "age",
            type: "int",
            isNullable: true
        }));
    }

    public async down(queryRunner: QueryRunner): Promise<void> {
        await queryRunner.dropColumn("user", "age");
    }
}

In this example, the

up
up method is used to define the changes to be applied to the database, which in this case is adding a new column
age
age to the
user
user table. The
down
down method is used to revert these changes, ensuring that the migration can be rolled back if necessary.

5. How do you handle transactions? Provide a code example.

In TypeORM, transactions are used to ensure that a series of database operations are executed in a way that maintains data integrity.

Here is an example of how to handle transactions in TypeORM:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { getManager } from "typeorm";
async function transactionalOperation() {
await getManager().transaction(async transactionalEntityManager => {
// Perform your database operations here
await transactionalEntityManager.save(entity1);
await transactionalEntityManager.save(entity2);
// Add more operations as needed
});
}
import { getManager } from "typeorm"; async function transactionalOperation() { await getManager().transaction(async transactionalEntityManager => { // Perform your database operations here await transactionalEntityManager.save(entity1); await transactionalEntityManager.save(entity2); // Add more operations as needed }); }
import { getManager } from "typeorm";

async function transactionalOperation() {
    await getManager().transaction(async transactionalEntityManager => {
        // Perform your database operations here
        await transactionalEntityManager.save(entity1);
        await transactionalEntityManager.save(entity2);
        // Add more operations as needed
    });
}

In this example, the

transaction
transaction method of
getManager
getManager is used to execute a series of operations within a transaction. If any of the operations fail, the entire transaction is rolled back.

6. Explain how to use custom repositories and why you might need them.

Custom repositories in TypeORM allow you to extend the default repository functionality by adding custom methods and encapsulating complex query logic. This is particularly useful when you have specific business logic that needs to be reused across different parts of your application.

To create a custom repository, you need to extend the

Repository
Repository class and define your custom methods. You can then use this custom repository in your services or controllers.

Example:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { EntityRepository, Repository } from 'typeorm';
import { User } from '../entity/User';
@EntityRepository(User)
export class UserRepository extends Repository<User> {
findByName(firstName: string, lastName: string) {
return this.findOne({ where: { firstName, lastName } });
}
}
// Usage in a service
import { getCustomRepository } from 'typeorm';
import { UserRepository } from '../repository/UserRepository';
const userRepository = getCustomRepository(UserRepository);
const user = await userRepository.findByName('John', 'Doe');
import { EntityRepository, Repository } from 'typeorm'; import { User } from '../entity/User'; @EntityRepository(User) export class UserRepository extends Repository<User> { findByName(firstName: string, lastName: string) { return this.findOne({ where: { firstName, lastName } }); } } // Usage in a service import { getCustomRepository } from 'typeorm'; import { UserRepository } from '../repository/UserRepository'; const userRepository = getCustomRepository(UserRepository); const user = await userRepository.findByName('John', 'Doe');
import { EntityRepository, Repository } from 'typeorm';
import { User } from '../entity/User';

@EntityRepository(User)
export class UserRepository extends Repository<User> {
    findByName(firstName: string, lastName: string) {
        return this.findOne({ where: { firstName, lastName } });
    }
}

// Usage in a service
import { getCustomRepository } from 'typeorm';
import { UserRepository } from '../repository/UserRepository';

const userRepository = getCustomRepository(UserRepository);
const user = await userRepository.findByName('John', 'Doe');

7. Write a custom repository method to find users by their email domain (e.g., all users with emails ending in ‘@example.com’).

To create a custom repository method to find users by their email domain, you can extend the base repository and add a method that uses a query to filter users based on the email domain.

Example:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { EntityRepository, Repository } from 'typeorm';
import { User } from './user.entity';
@EntityRepository(User)
export class UserRepository extends Repository<User> {
async findByEmailDomain(domain: string): Promise<User[]> {
return this.createQueryBuilder('user')
.where('user.email LIKE :domain', { domain: `%@${domain}` })
.getMany();
}
}
import { EntityRepository, Repository } from 'typeorm'; import { User } from './user.entity'; @EntityRepository(User) export class UserRepository extends Repository<User> { async findByEmailDomain(domain: string): Promise<User[]> { return this.createQueryBuilder('user') .where('user.email LIKE :domain', { domain: `%@${domain}` }) .getMany(); } }
import { EntityRepository, Repository } from 'typeorm';
import { User } from './user.entity';

@EntityRepository(User)
export class UserRepository extends Repository<User> {
    async findByEmailDomain(domain: string): Promise<User[]> {
        return this.createQueryBuilder('user')
            .where('user.email LIKE :domain', { domain: `%@${domain}` })
            .getMany();
    }
}

In this example, the

findByEmailDomain
findByEmailDomain method constructs a query to find users whose email addresses end with the specified domain. The
LIKE
LIKE operator is used to match the email pattern.

8. Provide an example of how to use TypeORM with a NestJS application.

To use TypeORM with a NestJS application, you need to follow these steps:

1. Install the necessary packages.
2. Configure the TypeORM module.
3. Create an entity.
4. Use a repository to perform database operations.

First, install the required packages:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
npm install @nestjs/typeorm typeorm mysql2
npm install @nestjs/typeorm typeorm mysql2
npm install @nestjs/typeorm typeorm mysql2

Next, configure the TypeORM module in your

app.module.ts
app.module.ts:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { Module } from '@nestjs/common';
import { TypeOrmModule } from '@nestjs/typeorm';
import { AppController } from './app.controller';
import { AppService } from './app.service';
import { User } from './user.entity';
@Module({
imports: [
TypeOrmModule.forRoot({
type: 'mysql',
host: 'localhost',
port: 3306,
username: 'root',
password: 'password',
database: 'test',
entities: [User],
synchronize: true,
}),
TypeOrmModule.forFeature([User]),
],
controllers: [AppController],
providers: [AppService],
})
export class AppModule {}
import { Module } from '@nestjs/common'; import { TypeOrmModule } from '@nestjs/typeorm'; import { AppController } from './app.controller'; import { AppService } from './app.service'; import { User } from './user.entity'; @Module({ imports: [ TypeOrmModule.forRoot({ type: 'mysql', host: 'localhost', port: 3306, username: 'root', password: 'password', database: 'test', entities: [User], synchronize: true, }), TypeOrmModule.forFeature([User]), ], controllers: [AppController], providers: [AppService], }) export class AppModule {}
import { Module } from '@nestjs/common';
import { TypeOrmModule } from '@nestjs/typeorm';
import { AppController } from './app.controller';
import { AppService } from './app.service';
import { User } from './user.entity';

@Module({
  imports: [
    TypeOrmModule.forRoot({
      type: 'mysql',
      host: 'localhost',
      port: 3306,
      username: 'root',
      password: 'password',
      database: 'test',
      entities: [User],
      synchronize: true,
    }),
    TypeOrmModule.forFeature([User]),
  ],
  controllers: [AppController],
  providers: [AppService],
})
export class AppModule {}

Create an entity

user.entity.ts
user.entity.ts:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { Entity, Column, PrimaryGeneratedColumn } from 'typeorm';
@Entity()
export class User {
@PrimaryGeneratedColumn()
id: number;
@Column()
name: string;
@Column()
age: number;
}
import { Entity, Column, PrimaryGeneratedColumn } from 'typeorm'; @Entity() export class User { @PrimaryGeneratedColumn() id: number; @Column() name: string; @Column() age: number; }
import { Entity, Column, PrimaryGeneratedColumn } from 'typeorm';

@Entity()
export class User {
  @PrimaryGeneratedColumn()
  id: number;

  @Column()
  name: string;

  @Column()
  age: number;
}

Finally, use a repository to perform database operations in a service:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm';
import { Repository } from 'typeorm';
import { User } from './user.entity';
@Injectable()
export class UserService {
constructor(
@InjectRepository(User)
private userRepository: Repository<User>,
) {}
async createUser(name: string, age: number): Promise<User> {
const user = this.userRepository.create({ name, age });
return this.userRepository.save(user);
}
async findAll(): Promise<User[]> {
return this.userRepository.find();
}
}
import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { User } from './user.entity'; @Injectable() export class UserService { constructor( @InjectRepository(User) private userRepository: Repository<User>, ) {} async createUser(name: string, age: number): Promise<User> { const user = this.userRepository.create({ name, age }); return this.userRepository.save(user); } async findAll(): Promise<User[]> { return this.userRepository.find(); } }
import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm';
import { Repository } from 'typeorm';
import { User } from './user.entity';

@Injectable()
export class UserService {
  constructor(
    @InjectRepository(User)
    private userRepository: Repository<User>,
  ) {}

  async createUser(name: string, age: number): Promise<User> {
    const user = this.userRepository.create({ name, age });
    return this.userRepository.save(user);
  }

  async findAll(): Promise<User[]> {
    return this.userRepository.find();
  }
}

9. Explain the role of decorators in TypeORM and provide examples.

Decorators in TypeORM are used to define the structure of the database schema directly within the entity classes. They provide a way to annotate classes and their properties to specify how they should be mapped to the database. This approach makes the code more readable and maintainable.

For example, the

@Entity
@Entity decorator is used to mark a class as a database table, while the
@Column
@Column decorator is used to mark a property as a table column. Other decorators like
@PrimaryGeneratedColumn
@PrimaryGeneratedColumn and
@ManyToOne
@ManyToOne are used to define primary keys and relationships between tables, respectively.

Example:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { Entity, PrimaryGeneratedColumn, Column, ManyToOne } from 'typeorm';
@Entity()
class User {
@PrimaryGeneratedColumn()
id: number;
@Column()
name: string;
@ManyToOne(() => Role, role => role.users)
role: Role;
}
@Entity()
class Role {
@PrimaryGeneratedColumn()
id: number;
@Column()
name: string;
@OneToMany(() => User, user => user.role)
users: User[];
}
import { Entity, PrimaryGeneratedColumn, Column, ManyToOne } from 'typeorm'; @Entity() class User { @PrimaryGeneratedColumn() id: number; @Column() name: string; @ManyToOne(() => Role, role => role.users) role: Role; } @Entity() class Role { @PrimaryGeneratedColumn() id: number; @Column() name: string; @OneToMany(() => User, user => user.role) users: User[]; }
import { Entity, PrimaryGeneratedColumn, Column, ManyToOne } from 'typeorm';

@Entity()
class User {
    @PrimaryGeneratedColumn()
    id: number;

    @Column()
    name: string;

    @ManyToOne(() => Role, role => role.users)
    role: Role;
}

@Entity()
class Role {
    @PrimaryGeneratedColumn()
    id: number;

    @Column()
    name: string;

    @OneToMany(() => User, user => user.role)
    users: User[];
}

In this example, the

User
User class is marked as an entity with the
@Entity
@Entity decorator. The
id
id property is marked as a primary key with the
@PrimaryGeneratedColumn
@PrimaryGeneratedColumn decorator, and the
name
name property is marked as a column with the
@Column
@Column decorator. The
role
role property is marked with the
@ManyToOne
@ManyToOne decorator to indicate a many-to-one relationship with the
Role
Role entity.

10. Describe how to test TypeORM entities and repositories.

Testing TypeORM entities and repositories involves several steps to ensure that your database interactions are functioning correctly. The process generally includes setting up a test database, configuring TypeORM to use this test database, and writing unit tests using a testing framework like Jest.

First, you need to configure a test database. This can be an in-memory database like SQLite for faster tests or a separate instance of your actual database. You will then configure TypeORM to connect to this test database during testing.

Next, you write unit tests for your entities and repositories. These tests should cover CRUD operations and any custom queries or methods you have implemented. Using Jest, you can mock dependencies and focus on testing the logic within your repositories.

Example:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
import { createConnection, getRepository } from 'typeorm';
import { User } from './entities/User';
import { UserRepository } from './repositories/UserRepository';
beforeAll(async () => {
await createConnection({
type: 'sqlite',
database: ':memory:',
dropSchema: true,
entities: [User],
synchronize: true,
logging: false,
});
});
afterAll(async () => {
const connection = getConnection();
await connection.close();
});
test('should create a new user', async () => {
const userRepository = getRepository(User);
const user = new User();
user.name = 'John Doe';
user.email = 'john.doe@example.com';
await userRepository.save(user);
const savedUser = await userRepository.findOne({ email: 'john.doe@example.com' });
expect(savedUser).toBeDefined();
expect(savedUser.name).toBe('John Doe');
});
import { createConnection, getRepository } from 'typeorm'; import { User } from './entities/User'; import { UserRepository } from './repositories/UserRepository'; beforeAll(async () => { await createConnection({ type: 'sqlite', database: ':memory:', dropSchema: true, entities: [User], synchronize: true, logging: false, }); }); afterAll(async () => { const connection = getConnection(); await connection.close(); }); test('should create a new user', async () => { const userRepository = getRepository(User); const user = new User(); user.name = 'John Doe'; user.email = 'john.doe@example.com'; await userRepository.save(user); const savedUser = await userRepository.findOne({ email: 'john.doe@example.com' }); expect(savedUser).toBeDefined(); expect(savedUser.name).toBe('John Doe'); });
import { createConnection, getRepository } from 'typeorm';
import { User } from './entities/User';
import { UserRepository } from './repositories/UserRepository';

beforeAll(async () => {
    await createConnection({
        type: 'sqlite',
        database: ':memory:',
        dropSchema: true,
        entities: [User],
        synchronize: true,
        logging: false,
    });
});

afterAll(async () => {
    const connection = getConnection();
    await connection.close();
});

test('should create a new user', async () => {
    const userRepository = getRepository(User);
    const user = new User();
    user.name = 'John Doe';
    user.email = 'john.doe@example.com';

    await userRepository.save(user);
    const savedUser = await userRepository.findOne({ email: 'john.doe@example.com' });

    expect(savedUser).toBeDefined();
    expect(savedUser.name).toBe('John Doe');
});
Previous

10 File Management Interview Questions and Answers

Back to Interview
Next

10 RestTemplate Interview Questions and Answers