Skip to content
Snippets Groups Projects
Commit ae549c30 authored by ALGLAVE Ivan's avatar ALGLAVE Ivan
Browse files

Switched fully to promises, added whitelist to DTO properties

parent fb29154d
No related branches found
No related tags found
1 merge request!2Crud groups
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { InjectModel } from '@nestjs/mongoose'; import { InjectModel } from '@nestjs/mongoose';
import { Model } from 'mongoose'; import { Model } from 'mongoose';
import { from, map, Observable } from 'rxjs';
import { CreateGroupDto } from '../dto/create-group.dto'; import { CreateGroupDto } from '../dto/create-group.dto';
import { UpdateGroupDto } from '../dto/update-group.dto'; import { UpdateGroupDto } from '../dto/update-group.dto';
import { Group } from '../schemas/group.schema'; import { Group } from '../schemas/group.schema';
...@@ -21,23 +20,46 @@ export class GroupsDao { ...@@ -21,23 +20,46 @@ export class GroupsDao {
}); });
}); });
findById = (id: string): Observable<Group | void> => findById = (id: string): Promise<Group | void> =>
from(this._groupModel.findById(id)); new Promise((resolve, reject) => {
this._groupModel.findOne({ id: id }, {}, {}, (err, value) => {
if (err) reject(err.message);
resolve(value);
});
});
save = (group: CreateGroupDto): Observable<Group> => save = (group: CreateGroupDto): Promise<Group> =>
from(new this._groupModel(group).save()); new Promise((resolve, reject) => {
new this._groupModel(group).save((err, value) => {
if (err) reject(err.message);
resolve(value);
});
});
findByIdAndUpdate = ( findByIdAndUpdate = (
id: string, id: string,
group: UpdateGroupDto, group: UpdateGroupDto,
): Observable<Group | void> => ): Promise<Group | void> =>
from( new Promise((resolve, reject) => {
this._groupModel.findByIdAndUpdate(id, group, { this._groupModel.updateOne(
new: true, { id: id },
runValidators: true, group,
}), {
); new: true,
runValidators: true,
},
(err, value) => {
if (err) reject(err.message);
resolve(value);
},
);
});
findByIdAndRemove = (id: string): Observable<Group | void> => findByIdAndRemove = (id: string): Promise<Group | void> =>
from(this._groupModel.findByIdAndRemove(id)); new Promise((resolve, reject) => {
this._groupModel.deleteOne({ id: id }, {}, (err) => {
if (err) reject(err.message);
resolve();
});
});
} }
import { IsBoolean, IsString, IsNotEmpty, IsMongoId } from 'class-validator'; import { IsBoolean, IsString, IsNotEmpty } from 'class-validator';
export class CreateGroupDto { export class CreateGroupDto {
@IsString() @IsString()
...@@ -8,7 +8,7 @@ export class CreateGroupDto { ...@@ -8,7 +8,7 @@ export class CreateGroupDto {
@IsBoolean() @IsBoolean()
final: boolean; final: boolean;
@IsMongoId() @IsString()
@IsNotEmpty() @IsNotEmpty()
parent: any; parent: any;
} }
import { IsBoolean, IsString, IsNotEmpty, IsMongoId } from 'class-validator'; import { IsArray, IsOptional } from 'class-validator';
export class UpdateGroupDto { export class UpdateGroupDto {
@IsString() @IsArray()
@IsNotEmpty() @IsOptional()
id: string; responsibles: string[];
@IsBoolean() @IsArray()
final: boolean; @IsOptional()
secretaries: string[];
@IsMongoId() @IsArray()
@IsNotEmpty() @IsOptional()
parent: string; students: string[];
} }
...@@ -26,12 +26,12 @@ export class GroupsController { ...@@ -26,12 +26,12 @@ export class GroupsController {
} }
@Get(':id') @Get(':id')
findOne(@Param() params: { id: string }): Observable<GroupEntity> { findOne(@Param() params: { id: string }): Promise<GroupEntity | void> {
return this._groupsService.findOne(params.id); return this._groupsService.findOne(params.id);
} }
@Post() @Post()
create(@Body() createGroupDto: CreateGroupDto): Observable<GroupEntity> { create(@Body() createGroupDto: CreateGroupDto): Promise<GroupEntity> {
return this._groupsService.create(createGroupDto); return this._groupsService.create(createGroupDto);
} }
...@@ -39,12 +39,12 @@ export class GroupsController { ...@@ -39,12 +39,12 @@ export class GroupsController {
update( update(
@Param() params: { id: string }, @Param() params: { id: string },
@Body() updateGroupDto: UpdateGroupDto, @Body() updateGroupDto: UpdateGroupDto,
): Observable<GroupEntity> { ): Promise<GroupEntity | void> {
return this._groupsService.update(params.id, updateGroupDto); return this._groupsService.update(params.id, updateGroupDto);
} }
@Delete(':id') @Delete(':id')
delete(@Param() params: { id: string }): Observable<void> { delete(@Param() params: { id: string }): Promise<GroupEntity | void> {
return this._groupsService.delete(params.id); return this._groupsService.delete(params.id);
} }
} }
import { import { Injectable } from '@nestjs/common';
Injectable,
UnprocessableEntityException,
NotFoundException,
ConflictException,
} from '@nestjs/common';
import {
Observable,
of,
filter,
map,
mergeMap,
defaultIfEmpty,
catchError,
throwError,
} from 'rxjs';
import { GroupsDao } from './dao/groups.dao'; import { GroupsDao } from './dao/groups.dao';
import { CreateGroupDto } from './dto/create-group.dto'; import { CreateGroupDto } from './dto/create-group.dto';
import { UpdateGroupDto } from './dto/update-group.dto'; import { UpdateGroupDto } from './dto/update-group.dto';
...@@ -25,80 +10,15 @@ export class GroupsService { ...@@ -25,80 +10,15 @@ export class GroupsService {
findAll = (): Promise<GroupEntity[] | void> => this._groupsDao.find(); findAll = (): Promise<GroupEntity[] | void> => this._groupsDao.find();
findOne = (id: string): Observable<GroupEntity> => findOne = (id: string): Promise<GroupEntity | void> =>
this._groupsDao.findById(id).pipe( this._groupsDao.findById(id);
catchError((e) =>
throwError(() => new UnprocessableEntityException(e.message)),
),
mergeMap((group) =>
!!group
? of(new GroupEntity(group))
: throwError(
() => new NotFoundException(`Group with id ${id} not found`),
),
),
);
create = (group: CreateGroupDto): Observable<GroupEntity> => create = (group: CreateGroupDto): Promise<GroupEntity> =>
this._prepareNewGroup(group).pipe( this._groupsDao.save(group);
mergeMap((newPreparedGroup: CreateGroupDto) =>
this._groupsDao.save(newPreparedGroup),
),
catchError((e) =>
e.code === 11000
? throwError(
() =>
new ConflictException(
`Group with id ${group.id} already exists`,
),
)
: throwError(() => new UnprocessableEntityException(e.message)),
),
map((groupCreated) => new GroupEntity(groupCreated)),
);
update = (id: string, group: UpdateGroupDto): Observable<GroupEntity> => update = (id: string, group: UpdateGroupDto): Promise<GroupEntity | void> =>
this._groupsDao.findByIdAndUpdate(id, group).pipe( this._groupsDao.findByIdAndUpdate(id, group);
catchError((e) =>
e.code === 11000
? throwError(
() =>
new ConflictException(
`Group with id ${group.id} already exists`,
),
)
: throwError(() => new UnprocessableEntityException(e.message)),
),
mergeMap((groupUpdated) =>
!!groupUpdated
? of(new GroupEntity(groupUpdated))
: throwError(
() => new NotFoundException(`Group with id '${id}' not found`),
),
),
);
delete = (id: string): Observable<void> => delete = (id: string): Promise<GroupEntity | void> =>
this._groupsDao.findByIdAndRemove(id).pipe( this._groupsDao.findByIdAndRemove(id);
catchError((e) =>
throwError(() => new UnprocessableEntityException(e.message)),
),
mergeMap((groupDeleted) =>
!!groupDeleted
? of(undefined)
: throwError(
() => new NotFoundException(`Group with id '${id}' not found`),
),
),
);
private _prepareNewGroup = (
group: CreateGroupDto,
): Observable<CreateGroupDto> =>
of({
...group,
responsibles: [],
secretaries: [],
students: [],
});
} }
import * as mongoose from 'mongoose'; import * as mongoose from 'mongoose';
import { Document } from 'mongoose'; import { Document } from 'mongoose';
import { Prop, raw, Schema, SchemaFactory } from '@nestjs/mongoose'; import { Prop, Schema, SchemaFactory } from '@nestjs/mongoose';
export type GroupDocument = Group & Document; export type GroupDocument = Group & Document;
...@@ -34,28 +34,28 @@ export class Group { ...@@ -34,28 +34,28 @@ export class Group {
final: boolean; final: boolean;
@Prop({ @Prop({
type: [mongoose.Schema.Types.ObjectId], type: [String],
required: true, required: true,
trim: true, trim: true,
}) })
responsibles: string[]; responsibles: string[];
@Prop({ @Prop({
type: [mongoose.Schema.Types.ObjectId], type: [String],
required: true, required: true,
trim: true, trim: true,
}) })
secretaries: string[]; secretaries: string[];
@Prop({ @Prop({
type: [mongoose.Schema.Types.ObjectId], type: [String],
required: true, required: true,
trim: true, trim: true,
}) })
students: string[]; students: string[];
@Prop({ @Prop({
type: mongoose.Schema.Types.ObjectId, type: String,
required: true, required: true,
trim: true, trim: true,
}) })
......
import { ValidationPipe } from '@nestjs/common';
import { NestFactory } from '@nestjs/core'; import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module'; import { AppModule } from './app.module';
import { server } from './config'; import { server } from './config';
async function bootstrap() { async function bootstrap() {
const app = await NestFactory.create(AppModule); const app = await NestFactory.create(AppModule);
await app.useGlobalPipes(
new ValidationPipe({
whitelist: true,
forbidNonWhitelisted: true,
}),
);
await app.listen(server.port); await app.listen(server.port);
} }
bootstrap(); bootstrap();
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment