ohmyform-api/src/service/user/user.create.service.ts

97 lines
2.9 KiB
TypeScript

import { Injectable } from '@nestjs/common'
import { ConfigService } from '@nestjs/config'
import { InjectRepository } from '@nestjs/typeorm'
import crypto from 'crypto'
import { PinoLogger } from 'nestjs-pino'
import { Repository } from 'typeorm'
import { rolesType } from '../../config/roles'
import { UserCreateInput } from '../../dto/user/user.create.input'
import { UserEntity } from '../../entity/user.entity'
import { PasswordService } from '../auth/password.service'
import { MailService } from '../mail.service'
import { SettingService } from '../setting.service'
import { UserTokenService } from './user.token.service'
@Injectable()
export class UserCreateService {
constructor(
@InjectRepository(UserEntity)
private readonly userRepository: Repository<UserEntity>,
private readonly mailerService: MailService,
private readonly logger: PinoLogger,
private readonly passwordService: PasswordService,
private readonly settingService: SettingService,
private readonly configService: ConfigService,
private readonly userTokenService: UserTokenService,
) {
logger.setContext(this.constructor.name)
}
private getDefaultRoles(): rolesType {
const roleSetting = this.settingService.getByKey('DEFAULT_ROLE')
switch (roleSetting.value) {
case 'superuser':
return [
'superuser',
'admin',
'user',
]
case 'admin':
return ['admin', 'user']
}
return ['user']
}
async create(input: UserCreateInput, roles?: rolesType): Promise<UserEntity> {
if (undefined !== await this.userRepository.findOne({ username: input.username })) {
throw new Error('username already in use')
}
if (undefined !== await this.userRepository.findOne({ email: input.email })) {
throw new Error('email already in use')
}
const confirmToken = crypto.randomBytes(30).toString('base64')
let user = new UserEntity()
user.provider = 'local'
user.username = input.username
user.email = input.email
user.firstName = input.firstName
user.lastName = input.lastName
user.language = input.language ?? 'en'
user.roles = roles ? roles : this.getDefaultRoles()
user.passwordHash = await this.passwordService.hash(input.password)
user.token = await this.userTokenService.hash(confirmToken)
user = await this.userRepository.save(user)
const confirmUrl = [
this.configService.get('BASE_URL', 'http://localhost'),
this.configService.get('USER_CONFIRM_PATH', '/confirm?token={{token}}'),
]
.join('')
.replace('{{token}}', confirmToken)
const sent = await this.mailerService.send(
user.email,
'user/created',
{
username: user.username,
confirm: confirmUrl,
}
)
// so send email
if (!sent) {
this.logger.warn('failed to send email for user creation')
}
return user
}
}