Использование хранилища параметров AWS SSM для конфигурации TypeORM в NestJS

У меня есть простое приложение NestJS, работающее на AWS Lambda. Я использую хранилище параметров AWS SSM для хранения информации о соединении с базой данных и учетных данных. Когда я импортирую TypeORM, я использую параметры, уже извлеченные из магазина.

Прямо сейчас я просто получаю параметры прямо в моем AppModule, куда импортируется TypeORM. Я уверен, что есть лучший способ сделать это, но я не уверен, что это будет. Пользовательский провайдер? Какие-то настройки Сервиса? Я не думаю, что мое текущее решение очень надежно и не очень хорошо обрабатывает ошибки.

Единственное требование - я получаю параметры SSM во время выполнения, а не во время сборки или развертывания. Любой совет

Вот что я сейчас делаю:

import { Module } from '@nestjs/common';
import { AppController } from './app.controller';
import { AppService } from './app.service';
import { TypeOrmModule } from '@nestjs/typeorm';
import { ThingyModule } from 'thingy/thingy.module';
import * as awsParamStore from 'aws-param-store';

const ssmParams = awsParamStore.getParametersByPathSync('/myapp/prod', {region: (process.env['AWS_DEFAULT_REGION'] ? process.env['AWS_DEFAULT_REGION'] : 'us-east-2')}); 
const ssmMap = ssmParams.reduce(function(map, obj) {
  map[obj.Name] = obj.Value;
  return map;
}, {});

@Module({
  imports: [
    TypeOrmModule.forRoot({
      type: 'postgres',
      host: ssmMap['/myapp/prod/db/host'],
      port: 5432,
      username: ssmMap['/myapp/prod/db/username'],
      password: ssmMap['/myapp/prod/db/password'],
      database: 'postgres',
      entities: [__dirname + '/**/*.entity{.ts,.js}'],
      synchronize: true,
    }),
    ThingyModule
  ],
  controllers: [AppController],
  providers: [AppService],
})
export class AppModule {}

Параметры были созданы ранее с помощью интерфейса командной строки AWS:

// aws ssm put-parameter --type String --name /myapp/prod/db/username --value postgres --region us-east-2
// aws ssm put-parameter --type String --name /myapp/prod/db/password --value supRCkrit --region us-east-2
// aws ssm put-parameter --type String --name /myapp/prod/db/host --value localhost --region us-east-2

1 ответ

Мы можем сделать это с помощью провайдеров async.

Образец из рабочего проекта:

Создайте асинхронный провайдер:

      import * as AWS from 'aws-sdk';
import { Parameter } from 'aws-sdk/clients/ssm';

export const ssmProvider = {
  provide: 'AWS_SSM',
  useFactory: async (): Promise<Parameter[]> => {
    const ssmClient = new AWS.SSM({
      endpoint: 'endpoint',
      region: 'us-west-2',
    });
    const result = await ssmClient
      .getParametersByPath({
        Path: '/ssm/path',
        Recursive: true,
      })
      .promise();
    return result?.Parameters;
  },
};

Затем мы можем использовать этого поставщика для ввода значений.

Например, добавление параметров AWS SSM в службу конфигурации:

      export class ConfigModule {
  static register(options: ConfigModuleOptions): DynamicModule {
    return {
      global: true,
      module: ConfigModule,
      providers: [
        ssmProvider,
        {
          provide: CONFIG_OPTIONS,
          useValue: options,
        },
        ConfigService,
      ],
      exports: [ConfigService],
    };
  }
}

И в конструкторе службы конфигурации:

      constructor(
    @Inject(CONFIG_OPTIONS) options: ConfigOptions,
    @Inject('AWS_SSM') awsParameters: Parameter[],
  ) {}
Другие вопросы по тегам