NestJS - Database, TypeORM


이 포스트는 NestJS 에 MySQL 과 TypeORM 을 적용해보는 법에 대해 알아본다.

소스는 user-service 에 있습니다.


1. MySQL 설정

NestJS 는 TypeORM, MikroORM, Sequelize, Knex.js, Prisma 와 같은 ORM 을 지원한다.
이 포스트에선 MySQL 와 TypeORM 을 이용할 예정이다.

docker 를 이용하여 mysql 을 실행하는 방법은 Rancher Desktop (Docker Desktop 유료화 대응) 을 참고하세요.

TypeORM 와 Sequelize 비교글은 Typescript - TypeORM vs Sequelize 을 참고하세요.

MySQL 8.0 부터는 설정에서 Public Key 등록을 허용해주어야 한다. Stars

MySQL 설정이 끝났으면 test DB 를 하나 만든다.
한글 정렬이 잘 되도록 하기 위해 Charset 은 utf8mb4, Collation 은 utf8mb4_unicode_ci 로 설정한다.

create database test default charset utf8mb4 collate utf8mb4_unicode_ci;

2. 유저 서비스

2.1. TypeORM 으로 DB 연결

NestJS 에 MySQL 연결을 위한 라이브러리를 설치한다.

$ npm i typeorm@0.3.12 @nestjs/typeorm@9.0.1 mysql2@3.2.0

app.module.ts

import { TypeOrmModule } from '@nestjs/typeorm';

@Module({
  imports: [
    UsersModule,
    ConfigModule.forRoot({
      envFilePath: [`${__dirname}/config/env/.${process.env.NODE_ENV}.env`],
      load: [emailConfig], // ConfigFactory 지정
      isGlobal: true, // 전역으로 등록해서 어느 모듈에서나 사용 가능
      validationSchema, // 환경 변수 값에 대해 유효성 검사 수행
    }),
    //TypeORMModule 을 동적으로 가져옴
    TypeOrmModule.forRoot({
      type: 'mysql',
      host: 'localhost',
      port: 3306,
      username: 'root',
      password: 'test',
      database: 'test',
      entities: [__dirname + '/**/*.entity{.ts,.js}'], // TypeORM 이 구동될 때 인식하도록 할 entity 클래스의 경로 지정
      synchronize: true, // 서비스 구동 시 소스 코드 기반으로 DB 스키마 동기화할지 여부, PROD 에서는 false 로 할 것
    }),
  ],
  controllers: [],
  providers: [],
})
export class AppModule {}

syncronize 옵션을 true 로 지정 시 서비스가 실행되서 DB 연결 시 DB 가 초기화되므로 PROD 에서는 절대 true 로 지정하면 안된다.

TypeOrmModule.forRoot 의 시그니처

static forRoot(options?: TypeOrmModuleOptions): DynamicModule;

TypeOrmModuleOptions 의 시그니처

export declare type TypeOrmModuleOptions = {
    /**
     * Number of times to retry connecting
     * Default: 10
     */
    retryAttempts?: number;
    /**
     * Delay between connection retry attempts (ms)
     * Default: 3000
     */
    retryDelay?: number;
    /**
     * Function that determines whether the module should
     * attempt to connect upon failure.
     *
     * @param err error that was thrown
     * @returns whether to retry connection or not
     */
    toRetry?: (err: any) => boolean;
    /**
     * If `true`, entities will be loaded automatically.
     */
    autoLoadEntities?: boolean;
    /**
     * If `true`, connection will not be closed on application shutdown.
     * @deprecated
     */
    keepConnectionAlive?: boolean;
    /**
     * If `true`, will show verbose error messages on each connection retry.
     */
    verboseRetryLog?: boolean;
} & Partial<DataSourceOptions>;
  • retryAttempts
    • 연결 시 재시도 횟수, default 10
  • retryDelay
    • 재시도 간 지연시간, ms 단위이며 default 3000
  • toRetry
    • 에러 발생 시 연결을 시도할 지 판단하는 함수
    • 콜백으로 받은 인수 err 을 이용하여 연결 여부를 판단하는 함수를 구현하면 됨
  • autoLoadEntities
    • entity 를 자동으로 로드할 지 여부
  • keepConnectionAlive
    • 애플리케이션 종료 후 연결을 유지할 지 여부
  • verboseRetryLog
    • 연결 재시도 시 verbose 레벨로 에러 메시지를 보여줄 지 여부

TypeOrmModuleOptionsDataSourceOptions 타입의 Partial 타입을 교차(&) 한 타입이다.
Partial 제네릭 타입은 선언한 타입의 일부 속성만을 가질 수 있도록 하는 타입이고, 교차 타입은 교차시킨 타입의 속성들을 모두 갖는 타입이다.

위에서 설정한 옵션 외 다른 옵션을 조정하려면 DataSourceOptionsMysqlConnectionOptions 을 보고 설정하면 된다.

이제 위에서 host, username, password 등과 같은 예민한 정보를 환경 변수에서 읽어오도록 수정한다.

TypeOrmModule.forRoot({
      type: 'mysql',
      host: process.env.DATABASE_HOST,
      port: 3306,
      username: process.env.DATABASE_USERNAME,
      password: process.env.DATABASE_PASSWORD,
      database: 'test',
      entities: [__dirname + '/**/*.entity{.ts,.js}'], // TypeORM 이 구동될 때 인식하도록 할 entity 클래스의 경로 지정
      synchronize: true, // 서비스 구동 시 소스 코드 기반으로 DB 스키마 동기화할지 여부, PROD 에서는 false 로 할 것
    }),

ormconfig.json 방식도 있는데 이 방식은 typeorm 0.3 버전에서는 지원하지 않는다. (0.2.x 버전까지만 지원)


2.2. 회원 가입 시 유저 정보 저장: @InjectRepository

NestJS 는 리포지터리 패턴 을 지원한다.

유저 Entity 를 정의한다.

/src/users/entity/user.entity.ts

import { Column, Entity, PrimaryColumn } from 'typeorm';

@Entity('User')
export class UserEntity {
  @PrimaryColumn()
  id: string;

  @Column({ length: 30 })
  name: string;

  @Column({ length: 60 })
  email: string;

  @Column({ length: 30 })
  password: string;

  @Column({ length: 60 })
  signupVerifyToken: string;
}

이제 이 유저 Entity 를 DB 에서 사용할 수 있도록 TypeOrmModuleOptions 의 entities 속성의 값으로 넣어준다.

TypeOrmModule.forRoot({
    ...
    entities: [UserEntity], // TypeORM 이 구동될 때 인식하도록 할 entity 클래스의 경로 지정
    ...
  }),

하지만 처음에 이미 dist 디렉터리 내의 .entity.ts 또는 .entity.js 로 끝나는 파일을 참조하도록 해두었으므로 수정하지 않아도 된다.

entities: [__dirname + '/**/*.entity{.ts,.js}'], // TypeORM 이 구동될 때 인식하도록 할 entity 클래스의 경로 지정

이제 서비스를 구동하면 User 테이블이 생성된 것을 확인할 수 있다. (synchronize 옵션이 true 이므로)

CREATE TABLE `User` (
  `id` varchar(255) COLLATE utf8mb4_unicode_ci NOT NULL,
  `name` varchar(30) COLLATE utf8mb4_unicode_ci NOT NULL,
  `email` varchar(60) COLLATE utf8mb4_unicode_ci NOT NULL,
  `password` varchar(30) COLLATE utf8mb4_unicode_ci NOT NULL,
  `signupVerifyToken` varchar(60) COLLATE utf8mb4_unicode_ci NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci

이제 NestJS - Provider2.2. 회원 가입 에서 TODO 로 남겨둔 작업들을 구현해본다.

UsersModule 에 forFeature() 로 유저 모듈 내에서 사용할 저장소 등록
/src/users/users.module.ts

import { TypeOrmModule } from '@nestjs/typeorm';
import { UserEntity } from './entity/user.entity';

@Module({
  imports: [EmailModule, 
    TypeOrmModule.forFeature([UserEntity])],    // UsersModule 에 forFeature() 로 유저 모듈 내에서 사용할 저장소 등록
  controllers: [UsersController],
  providers: [UsersService],
})
export class UsersModule {}

UsersService 에 @InjectRepository 데커레이터로 유저 저장소 주입
/src/users/users.service.ts

import { InjectRepository } from '@nestjs/typeorm';
import { UserEntity } from './entity/user.entity';
import { Repository } from 'typeorm';

@Injectable()
export class UsersService {
  constructor(
          private emailService: EmailService,
          //UsersService 에 `@InjectRepository` 데커레이터로 유저 저장소 주입
          @InjectRepository(UserEntity) 
          private userRepository: Repository<UserEntity>,
  ) { }
...
}

유저 정보 저장 (saveUser)

import { ulid } from 'ulid';
...

@Injectable()
export class UsersService {
    ...
  // 유저 정보 저장
  private async saveUser(
          name: string,
          email: string,
          password: string,
          signupVerifyToken: string,
  ) {
    const user = new UserEntity(); // 유저 엔티티 객체 생성
    user.id = ulid();
    user.name = name;
    user.email = email;
    user.password = password;
    user.signupVerifyToken = signupVerifyToken;

    await this.userRepository.save(user);
  }
}

ulid 는 ULID vs UUID: Sortable Random ID Generators for JavaScript 을 참고하세요.

이제 유저 생성 API 를 호출하면 DB 에 유저 정보가 저장되는 것을 확인할 수 있다.

$ curl --location 'http://localhost:3000/users' \
--header 'Content-Type: application/json' \
--data-raw '{
"name": "assu",
"email": "assu@test.com",
"password": "12341234"
}' | jq

가입 유무 확인 (checkUserExists)

  // 가입 유무 확인
  private async checkUserExists(email: string) {
    const user = await this.userRepository.findOne({
      where: { email: email },
    });

    return user !== null;
  }

유저 생성 시 이메일이 존재하면 422 리턴

// 회원 가입
async createUser(name: string, email: string, password: string)
{
  // 가입 유무 확인
  const userExist = await this.checkUserExists(email);
  if (userExist) {
    throw new UnprocessableEntityException('Email already exists');
  }
...
}
$ curl --location 'http://localhost:3000/users' \
--header 'Content-Type: application/json' \
--data-raw '{
"name": "assu",
"email": "assu@test.com",
"password": "12341234"
}' | jq

{
  "statusCode": 422,
  "message": "Email already exists",
  "error": "Unprocessable Entity"
}

2.3. Transaction 적용

TypeORM 에서 Transaction 을 처리하는 방법은 2가지가 있다.

  • QueryRunner 로 적용
  • transaction 함수 직접 사용하여 적용

2.3.1. QueryRunner 로 적용

TypeORM 에서 제공하는 DataSource 객체 주입
/src/users/users.service.ts

import { DataSource, Repository } from 'typeorm';

@Injectable()
export class UsersService {
...
  constructor(
          private dataSource: DataSource, // TypeORM 에서 제공하는 DataSource 객체 주입
  ) { }
...
}

위처럼 DataSource 객체를 주입하고 나면 DataSource 객체에서 트랜잭션을 생성할 수 있다.

// 유저 정보 저장 - QueryRunner 로 트랜잭션 제어
  private async saveUserUsingQueryRunner(
    name: string,
    email: string,
    password: string,
    signupVerifyToken: string,
  ) {
    // 주입받은 DataSource 객체에서 QueryRunner 생성
    const queryRunner = this.dataSource.createQueryRunner();

    // QueryRunner 에 DB 연결 후 트랜잭션 시작
    await queryRunner.connect();
    await queryRunner.startTransaction();
    try {
      const user = new UserEntity(); // 유저 엔티티 객체 생성
      user.id = ulid();
      user.name = name;
      user.email = email;
      user.password = password;
      user.signupVerifyToken = signupVerifyToken;

      // 트랜잭션을 커밋하여 영속화(persistence) 함
      await queryRunner.manager.save(user);

      // 일부러 에러 발생 시 데이터 저장 안됨
      // throw new InternalServerErrorException();

      // DB 작업 수행 수 커밋하여 영속화 완료
      await queryRunner.commitTransaction();
    } catch (e) {
      // 에러 발생 시 롤백
      await queryRunner.rollbackTransaction();
    } finally {
      // 직접 생성한 QueryRunner 해제
      await queryRunner.release();
    }
  }

2.3.2. transaction 함수 직접 사용하여 적용

dataSource 객체 내의 transaction 함수를 바료 이용할 수도 있다.

// 유저 정보 저장 - transaction 함수 직접 이용하여 트랜잭션 제어
  private async saveUserUsingTransaction(
    name: string,
    email: string,
    password: string,
    signupVerifyToken: string,
  ) {
    await this.dataSource.transaction(async (manager) => {
      const user = new UserEntity(); // 유저 엔티티 객체 생성
      user.id = ulid();
      user.name = name;
      user.email = email;
      user.password = password;
      user.signupVerifyToken = signupVerifyToken;

      await manager.save(user);

      // 일부러 에러 발생 시 데이터 저장 안됨
      //throw new InternalServerErrorException();
    });
  }

2.4. 마이그레이션

TypeORM 을 이용하면 아래와 같은 장점이 있다.

  • 마이그레이션을 위한 SQL 문을 직접 작성하지 않아도 됨
    • 마이그레이션 롤백 작업도 간단한 명령어로 수행 가능
  • 마이그레이션 코드를 일정한 형식으로 소스 저장소에서 관리 가능 (= DB 변경점을 소스 코드로 관리 가능)
  • 마이그레이션 이력 관리 가능
    • 특정 테이블에 마이그레이션 기록
    • 필요 시 처음부터 다시 수행 가능

2.4.1. 마이그레이션을 CLI 로 생성하고 실행할 수 있는 환경 구성

migration cli 로 명령어를 수행하는데 필요한 패키지를 설치한다. typeorm cli 는 이미 설치한 typeorm 패키지에 포함되어 있다. typeorm cli 는 타입스크립트로 된 엔티티 파일을 읽어들이므로 typeorm cli 를 실행하기 위해 ts-node 패키지를 글로벌 환경으로 설치한다.

$ npm i -g ts-node

이제 ts-node 를 이용하여 npm run typeorm 명령어로 typeorm cli 를 실행할 수 있도록 package.json 을 수정한다.

"scripts": {
...
"typeorm-create": "ts-node -r ts-node/register ./node_modules/typeorm/cli.js",
"typeorm-generate": "ts-node -r tsconfig-paths/register ./node_modules/typeorm/cli.js -d ormconfig.ts"
...
}

ormconfig.ts 파일을 루트 디렉터리에 생성한다.

import { DataSource } from 'typeorm';

export const AppDataSource = new DataSource({
  type: 'mysql',
  host: '127.0.0.1',
  //host: process.env.DATABASE_HOST,
  port: 13306,
  username: 'root',
  //username: process.env.DATABASE_USERNAME,
  password: ' ㅅㄷㄴㅅ',
  //password: process.env.DATABASE_PASSWORD,
  database: 'test',
  entities: [__dirname + '/**/*.entity{.ts,.js}'],
  synchronize: false,
  migrations: [__dirname + '/dist/migrations/**/*{.ts,.js}'],  // /dist/migrations 에 있는 파일 실행
  migrationsTableName: 'migrations',
});

각 항목을 환경 변수 값으로 읽어오도록 하면 ormconfig.ts 파일이 ConfigModule.forRoot 로 환경 변수를 읽어오기 전에 컴파일이 되기 때문에 서버 구동에서 에러가 발생한다. 따라서 tsconfig.json 에서 컴파일 대상 소스를 아래와 같이 지정해준다.

  "include": [
    "src/**/*"
  ]

이제 마이그레이션 이력을 관리할 테이블을 설정한다.

/src/app.module.ts

    //TypeORMModule 을 동적으로 가져옴
    TypeOrmModule.forRoot({
      ...
      synchronize: process.env.DATABASE_SYNCRONIZE === 'true', // 서비스 구동 시 소스 코드 기반으로 DB 스키마 동기화할지 여부, PROD 에서는 false 로 할 것
      migrationsRun: false, // 서버가 구동될 때 작성된 마이그레이션 파일을 기반으로 마이그레이션 수행할 지 여부 설정, false 로 하여 cli 명령어로 직접 입력하도록 함
      migrations: [__dirname + '/migrations/**/*{.ts,.js}'], // 마이그레이션을 수행할 파일이 관리되는 경로, 디폴트 migrations
      migrationsTableName: 'migrations',
    }),

이제 User 테이블 삭제 후 서버를 재기동한다.


2.4.2. 마이그레이션 실행

마이그레이션 파일 생성 방법은 2가지가 있다.

  • migration:create
    • 비어있는 파일 생성
  • migration:generate
    • 현재 소스 코드와 migrations 테이블에 기록된 이력을 기반으로 마이그레이션 파일 자동 생성

migration:create 로 마이그레이션 파일을 생성해본다.

$ npm run typeorm-create migration:create src/migrations/CreateUserTable

/src/migrations/1680768213491-CreateUserTable.ts

import { MigrationInterface, QueryRunner } from 'typeorm';

export class CreateUserTable1680768213491 implements MigrationInterface {
  // migration:run 명령으로 마이그레이션이 수행될 때 수행될 코드
  public async up(queryRunner: QueryRunner): Promise<void> {}

  // migration:revert 명령으로 마이그레이션을 되돌릴 때 수행될 코드
  public async down(queryRunner: QueryRunner): Promise<void> {}
}

위 파일을 삭제하고 migration:generate 로 다시 생성해본다.

$ npm run typeorm-generate migration:generate src/migrations/CreateUserTable

/src/migrations/1680768830565-CreateUserTable.ts

import { MigrationInterface, QueryRunner } from 'typeorm';

export class CreateUserTable1680768830565 implements MigrationInterface {
  name = 'CreateUserTable1680768830565';

  public async up(queryRunner: QueryRunner): Promise<void> {
    await queryRunner.query(
      `CREATE TABLE \`User\` (\`id\` varchar(255) NOT NULL, \`name\` varchar(30) NOT NULL, \`email\` varchar(60) NOT NULL, \`password\` varchar(30) NOT NULL, \`signupVerifyToken\` varchar(60) NOT NULL, PRIMARY KEY (\`id\`)) ENGINE=InnoDB`,
    );
  }

  public async down(queryRunner: QueryRunner): Promise<void> {
    await queryRunner.query(`DROP TABLE \`User\``);
  }
}

이제 migration:run 으로 마이그레이션을 수행해본다.

$ npm run typeorm-generate migration:run

> user-service@0.0.1 typeorm-generate
> ts-node -r tsconfig-paths/register ./node_modules/typeorm/cli.js -d ormconfig.ts migration:run

query: SELECT VERSION() AS `version`
query: SELECT * FROM `INFORMATION_SCHEMA`.`COLUMNS` WHERE `TABLE_SCHEMA` = 'test' AND `TABLE_NAME` = 'migrations'
query: SELECT * FROM `test`.`migrations` `migrations` ORDER BY `id` DESC
0 migrations are already loaded in the database.
1 migrations were found in the source code.
1 migrations are new migrations must be executed.
query: START TRANSACTION
query: CREATE TABLE `User` (`id` varchar(255) NOT NULL, `name` varchar(30) NOT NULL, `email` varchar(60) NOT NULL, `password` varchar(30) NOT NULL, `signupVerifyToken` varchar(60) NOT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB
query: INSERT INTO `test`.`migrations`(`timestamp`, `name`) VALUES (?, ?) -- PARAMETERS: [1680769589673,"CreateUserTable1680769589673"]
Migration CreateUserTable1680769589673 has been  executed successfully.
query: COMMIT

이제 migrate:revert 로 마이그레이션을 되돌려본다.

$ npm run typeorm-generate migration:revert

> user-service@0.0.1 typeorm-generate
> ts-node -r tsconfig-paths/register ./node_modules/typeorm/cli.js -d ormconfig.ts migration:revert

query: SELECT VERSION() AS `version`
query: SELECT * FROM `INFORMATION_SCHEMA`.`COLUMNS` WHERE `TABLE_SCHEMA` = 'test' AND `TABLE_NAME` = 'migrations'
query: SELECT * FROM `test`.`migrations` `migrations` ORDER BY `id` DESC
1 migrations are already loaded in the database.
CreateUserTable1680769589673 is the last executed migration. It was executed on Thu Apr 06 2023 17:26:29 GMT+0900 (Korean Standard Time).
Now reverting it...
query: START TRANSACTION
query: DROP TABLE `User`
query: DELETE FROM `test`.`migrations` WHERE `timestamp` = ? AND `name` = ? -- PARAMETERS: [1680769589673,"CreateUserTable1680769589673"]
Migration CreateUserTable1680769589673 has been  reverted successfully.
query: COMMIT

참고 사이트 & 함께 보면 좋은 사이트

본 포스트는 한용재 저자의 NestJS로 배우는 백엔드 프로그래밍을 기반으로 스터디하며 정리한 내용들입니다.






© 2020.08. by assu10

Powered by assu10