think/packages/server/src/services/user.service.ts

168 lines
5.2 KiB
TypeScript
Raw Normal View History

2022-05-16 09:23:59 +00:00
import { CreateUserDto } from '@dtos/create-user.dto';
import { LoginUserDto } from '@dtos/login-user.dto';
import { UpdateUserDto } from '@dtos/update-user.dto';
import { UserEntity } from '@entities/user.entity';
import { forwardRef, HttpException, HttpStatus, Inject, Injectable } from '@nestjs/common';
2022-02-20 11:51:55 +00:00
import { ConfigService } from '@nestjs/config';
import { JwtService } from '@nestjs/jwt';
2022-05-16 09:23:59 +00:00
import { InjectRepository } from '@nestjs/typeorm';
2022-02-20 11:51:55 +00:00
import { CollectorService } from '@services/collector.service';
2022-05-16 09:23:59 +00:00
import { MessageService } from '@services/message.service';
2022-02-20 11:51:55 +00:00
import { WikiService } from '@services/wiki.service';
2022-05-16 09:23:59 +00:00
import { CollectType, UserStatus } from '@think/domains';
import { instanceToPlain } from 'class-transformer';
import { Repository } from 'typeorm';
2022-02-20 11:51:55 +00:00
2022-03-22 05:37:53 +00:00
export type OutUser = Omit<UserEntity, 'comparePassword' | 'encryptPassword' | 'encrypt' | 'password'>;
2022-02-20 11:51:55 +00:00
@Injectable()
export class UserService {
constructor(
@InjectRepository(UserEntity)
private readonly userRepo: Repository<UserEntity>,
2022-04-04 13:42:52 +00:00
2022-02-20 11:51:55 +00:00
private readonly confifgService: ConfigService,
2022-04-04 13:42:52 +00:00
2022-02-20 11:51:55 +00:00
@Inject(forwardRef(() => JwtService))
private readonly jwtService: JwtService,
2022-04-04 13:42:52 +00:00
2022-02-20 11:51:55 +00:00
@Inject(forwardRef(() => MessageService))
private readonly messageService: MessageService,
2022-04-04 13:42:52 +00:00
2022-02-20 11:51:55 +00:00
@Inject(forwardRef(() => CollectorService))
private readonly collectorService: CollectorService,
2022-04-04 13:42:52 +00:00
2022-02-20 11:51:55 +00:00
@Inject(forwardRef(() => WikiService))
2022-03-12 03:27:56 +00:00
private readonly wikiService: WikiService
2022-04-04 13:42:52 +00:00
) {}
2022-02-20 11:51:55 +00:00
/**
* id
* @param id
* @returns
*/
async findById(id): Promise<OutUser> {
const user = await this.userRepo.findOne(id);
return instanceToPlain(user) as OutUser;
}
/**
*
* @param opts
* @returns
*/
async findOne(opts: Partial<OutUser>): Promise<UserEntity> {
const user = await this.userRepo.findOne(opts);
return user;
}
/**
* ids
* @param id
* @returns
*/
async findByIds(ids): Promise<OutUser[]> {
const users = await this.userRepo.findByIds(ids);
return users.map((user) => instanceToPlain(user)) as OutUser[];
}
/**
*
* @param user CreateUserDto
* @returns
*/
async createUser(user: CreateUserDto): Promise<OutUser> {
if (await this.userRepo.findOne({ name: user.name })) {
throw new HttpException('该账户已被注册', HttpStatus.BAD_REQUEST);
}
if (user.password !== user.confirmPassword) {
throw new HttpException('两次密码不一致,请重试', HttpStatus.BAD_REQUEST);
}
if (await this.userRepo.findOne({ name: user.name })) {
throw new HttpException('该账户已被注册', HttpStatus.BAD_REQUEST);
}
if (user.email && (await this.userRepo.findOne({ email: user.email }))) {
throw new HttpException('该邮箱已被注册', HttpStatus.BAD_REQUEST);
}
const res = await this.userRepo.create(user);
const createdUser = await this.userRepo.save(res);
const wiki = await this.wikiService.createWiki(createdUser, {
name: createdUser.name,
description: `${createdUser.name}的个人空间`,
});
await this.collectorService.toggleStar(createdUser, {
targetId: wiki.id,
type: CollectType.wiki,
});
await this.messageService.notify(createdUser, {
title: `欢迎「${createdUser.name}`,
message: `系统已自动为您创建知识库,快去看看吧!`,
url: `/wiki/${wiki.id}`,
});
return instanceToPlain(createdUser) as OutUser;
}
/**
*
* @param user
* @returns
*/
2022-05-24 09:33:30 +00:00
async login(user: LoginUserDto): Promise<{ user: OutUser; token: string; domain: string; expiresIn: number }> {
2022-02-20 11:51:55 +00:00
const { name, password } = user;
const existUser = await this.userRepo.findOne({ where: { name } });
2022-03-12 03:27:56 +00:00
if (!existUser || !(await UserEntity.comparePassword(password, existUser.password))) {
2022-02-20 11:51:55 +00:00
throw new HttpException('用户名或密码错误', HttpStatus.BAD_REQUEST);
}
if (existUser.status === UserStatus.locked) {
throw new HttpException('用户已锁定,无法登录', HttpStatus.BAD_REQUEST);
}
const res = instanceToPlain(existUser) as OutUser;
const token = this.jwtService.sign(res);
2022-05-24 09:33:30 +00:00
const domain = this.confifgService.get('client.siteDomain');
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
const expiresIn = this.jwtService.decode(token, { complete: true }).payload.exp;
return { user: res, token, domain, expiresIn };
2022-02-20 11:51:55 +00:00
}
async validateUser(user: UserEntity) {
return await this.findById(user.id);
}
/**
*
* @param user
* @param dto
* @returns
*/
async updateUser(user: UserEntity, dto: UpdateUserDto): Promise<OutUser> {
const oldData = await this.userRepo.findOne(user.id);
const res = await this.userRepo.merge(oldData, dto);
const ret = await this.userRepo.save(res);
return instanceToPlain(ret) as OutUser;
}
async decodeToken(token) {
const user = this.jwtService.decode(token) as UserEntity;
return user;
}
2022-04-05 05:33:45 +00:00
/**
*
* @param pagination
* @returns
*/
async getUsers() {
const query = this.userRepo.createQueryBuilder('user');
const [data] = await query.getManyAndCount();
return data;
}
2022-02-20 11:51:55 +00:00
}