diff --git a/packages/backend/src/chat/chat.controller.ts b/packages/backend/src/chat/chat.controller.ts new file mode 100644 index 00000000..f2a5452d --- /dev/null +++ b/packages/backend/src/chat/chat.controller.ts @@ -0,0 +1,39 @@ +import { Controller, Get, Query } from '@nestjs/common'; +import { + ApiBadRequestResponse, + ApiOkResponse, + ApiOperation, +} from '@nestjs/swagger'; +import { ChatService } from '@/chat/chat.service'; +import { ChatScrollRequest } from '@/chat/dto/chat.request'; +import { ChatScrollResponse } from '@/chat/dto/chat.response'; + +@Controller('chat') +export class ChatController { + constructor(private readonly chatService: ChatService) {} + + @ApiOperation({ + summary: '채팅 스크롤 조회 API', + description: '채팅을 스크롤하여 조회한다.', + }) + @ApiOkResponse({ + description: '스크롤 조회 성공', + type: ChatScrollResponse, + }) + @ApiBadRequestResponse({ + description: '스크롤 크기 100 초과', + example: { + message: 'pageSize should be less than 100', + error: 'Bad Request', + statusCode: 400, + }, + }) + @Get() + async findChatList(@Query() request: ChatScrollRequest) { + return await this.chatService.scrollNextChat( + request.stockId, + request.latestChatId, + request.pageSize, + ); + } +} diff --git a/packages/backend/src/chat/chat.gateway.ts b/packages/backend/src/chat/chat.gateway.ts index 322f35e9..f5fc099e 100644 --- a/packages/backend/src/chat/chat.gateway.ts +++ b/packages/backend/src/chat/chat.gateway.ts @@ -67,18 +67,23 @@ export class ChatGateway implements OnGatewayConnection { async handleConnection(client: Socket) { const room = client.handshake.query.stockId; - if (!room || !(await this.stockService.checkStockExist(room as string))) { + if ( + !this.isString(room) || + !(await this.stockService.checkStockExist(room)) + ) { client.emit('error', 'Invalid stockId'); this.logger.warn(`client connected with invalid stockId: ${room}`); client.disconnect(); return; } - if (room) { - client.join(room); - const messages = await this.chatService.getChatList(room as string); - this.logger.info(`client joined room ${room}`); - client.emit('chat', messages); - } + client.join(room); + const messages = await this.chatService.scrollFirstChat(room); + this.logger.info(`client joined room ${room}`); + client.emit('chat', messages); + } + + private isString(value: string | string[] | undefined): value is string { + return typeof value === 'string'; } private toResponse(chat: Chat): chatResponse { diff --git a/packages/backend/src/chat/chat.module.ts b/packages/backend/src/chat/chat.module.ts index 345a0f76..089b37e4 100644 --- a/packages/backend/src/chat/chat.module.ts +++ b/packages/backend/src/chat/chat.module.ts @@ -1,13 +1,15 @@ import { Module } from '@nestjs/common'; import { TypeOrmModule } from '@nestjs/typeorm'; import { SessionModule } from '@/auth/session.module'; +import { ChatController } from '@/chat/chat.controller'; import { ChatGateway } from '@/chat/chat.gateway'; +import { ChatService } from '@/chat/chat.service'; import { Chat } from '@/chat/domain/chat.entity'; import { StockModule } from '@/stock/stock.module'; -import { ChatService } from '@/chat/chat.service'; @Module({ imports: [TypeOrmModule.forFeature([Chat]), StockModule, SessionModule], + controllers: [ChatController], providers: [ChatGateway, ChatService], }) export class ChatModule {} diff --git a/packages/backend/src/chat/chat.service.spec.ts b/packages/backend/src/chat/chat.service.spec.ts new file mode 100644 index 00000000..7d181061 --- /dev/null +++ b/packages/backend/src/chat/chat.service.spec.ts @@ -0,0 +1,23 @@ +import { DataSource } from 'typeorm'; +import { ChatService } from '@/chat/chat.service'; +import { createDataSourceMock } from '@/user/user.service.spec'; + +describe('ChatService 테스트', () => { + test('첫 스크롤을 조회시 100개 이상 조회하면 예외가 발생한다.', async () => { + const dataSource = createDataSourceMock({}); + const chatService = new ChatService(dataSource as DataSource); + + await expect(() => + chatService.scrollNextChat('A005930', 1, 101), + ).rejects.toThrow('pageSize should be less than 100'); + }); + + test('100개 이상의 채팅을 조회하려 하면 예외가 발생한다.', async () => { + const dataSource = createDataSourceMock({}); + const chatService = new ChatService(dataSource as DataSource); + + await expect(() => + chatService.scrollFirstChat('A005930', 101), + ).rejects.toThrow('pageSize should be less than 100'); + }); +}); diff --git a/packages/backend/src/chat/chat.service.ts b/packages/backend/src/chat/chat.service.ts index 624618bd..ef262e7b 100644 --- a/packages/backend/src/chat/chat.service.ts +++ b/packages/backend/src/chat/chat.service.ts @@ -1,12 +1,15 @@ -import { Injectable } from '@nestjs/common'; +import { BadRequestException, Injectable } from '@nestjs/common'; import { DataSource } from 'typeorm'; import { Chat } from '@/chat/domain/chat.entity'; +import { ChatScrollResponse } from '@/chat/dto/chat.response'; -interface ChatMessage { +export interface ChatMessage { message: string; stockId: string; } +const DEFAULT_PAGE_SIZE = 20; + @Injectable() export class ChatService { constructor(private readonly dataSource: DataSource) {} @@ -19,13 +22,77 @@ export class ChatService { }); } - async getChatList(stockId: string) { + async scrollFirstChat(stockId: string, scrollSize?: number) { + this.validatePageSize(scrollSize); + const result = await this.findFirstChatScroll(stockId, scrollSize); + return await this.toScrollResponse(result, scrollSize); + } + + async scrollNextChat( + stockId: string, + latestChatId?: number, + pageSize?: number, + ) { + this.validatePageSize(pageSize); + const result = await this.findChatScroll(stockId, latestChatId, pageSize); + return await this.toScrollResponse(result, pageSize); + } + + private validatePageSize(scrollSize?: number) { + if (scrollSize && scrollSize > 100) { + throw new BadRequestException('pageSize should be less than 100'); + } + } + + private async toScrollResponse(result: Chat[], pageSize: number | undefined) { + const hasMore = + !!result && result.length > (pageSize ? pageSize : DEFAULT_PAGE_SIZE); + if (hasMore) { + result.pop(); + } + return new ChatScrollResponse(result, hasMore); + } + + private async findChatScroll( + stockId: string, + latestChatId?: number, + pageSize?: number, + ) { + if (!latestChatId) { + return await this.findFirstChatScroll(stockId, pageSize); + } else { + return await this.findNextChatScroll(stockId, latestChatId, pageSize); + } + } + + private async findFirstChatScroll(stockId: string, pageSize?: number) { const queryBuilder = this.dataSource.createQueryBuilder(Chat, 'chat'); - console.log(stockId); + if (!pageSize) { + pageSize = DEFAULT_PAGE_SIZE; + } return queryBuilder .where('chat.stock_id = :stockId', { stockId }) - .orderBy('chat.created_at', 'DESC') - .limit(100) + .orderBy('chat.id', 'DESC') + .limit(pageSize + 1) + .getMany(); + } + + private async findNextChatScroll( + stockId: string, + latestChatId: number, + pageSize?: number, + ) { + const queryBuilder = this.dataSource.createQueryBuilder(Chat, 'chat'); + if (!pageSize) { + pageSize = DEFAULT_PAGE_SIZE; + } + return queryBuilder + .where('chat.stock_id = :stockId and chat.id < :latestChatId', { + stockId, + latestChatId, + }) + .orderBy('chat.id', 'DESC') + .limit(pageSize + 1) .getMany(); } } diff --git a/packages/backend/src/chat/domain/chat.entity.ts b/packages/backend/src/chat/domain/chat.entity.ts index 9406a2e4..13800bff 100644 --- a/packages/backend/src/chat/domain/chat.entity.ts +++ b/packages/backend/src/chat/domain/chat.entity.ts @@ -33,5 +33,5 @@ export class Chat { likeCount: number = 0; @Column(() => DateEmbedded, { prefix: '' }) - date?: DateEmbedded; + date: DateEmbedded; } diff --git a/packages/backend/src/chat/dto/chat.request.ts b/packages/backend/src/chat/dto/chat.request.ts new file mode 100644 index 00000000..f3260509 --- /dev/null +++ b/packages/backend/src/chat/dto/chat.request.ts @@ -0,0 +1,30 @@ +import { ApiProperty } from '@nestjs/swagger'; +import { IsNumber, IsOptional, IsString } from 'class-validator'; + +export class ChatScrollRequest { + @ApiProperty({ + description: '종목 주식 id(종목방 id)', + example: 'A005930', + }) + @IsString() + readonly stockId: string; + + @ApiProperty({ + description: '최신 채팅 id', + example: 99999, + required: false, + }) + @IsOptional() + @IsNumber() + readonly latestChatId?: number; + + @ApiProperty({ + description: '페이지 크기', + example: 20, + default: 20, + required: false, + }) + @IsOptional() + @IsNumber() + readonly pageSize?: number; +} diff --git a/packages/backend/src/chat/dto/chat.response.ts b/packages/backend/src/chat/dto/chat.response.ts new file mode 100644 index 00000000..48aacb0d --- /dev/null +++ b/packages/backend/src/chat/dto/chat.response.ts @@ -0,0 +1,44 @@ +import { ApiProperty } from '@nestjs/swagger'; +import { Chat } from '@/chat/domain/chat.entity'; +import { ChatType } from '@/chat/domain/chatType.enum'; + +interface ChatResponse { + id: number; + likeCount: number; + message: string; + type: string; + createdAt: Date; +} + +export class ChatScrollResponse { + @ApiProperty({ + description: '다음 페이지가 있는지 여부', + example: true, + }) + readonly hasMore: boolean; + + @ApiProperty({ + description: '채팅 목록', + example: [ + { + id: 1, + likeCount: 0, + message: '안녕하세요', + type: ChatType.NORMAL, + createdAt: new Date(), + }, + ], + }) + readonly chats: ChatResponse[]; + + constructor(chats: Chat[], hasMore: boolean) { + this.chats = chats.map((chat) => ({ + id: chat.id, + likeCount: chat.likeCount, + message: chat.message, + type: chat.type, + createdAt: chat.date!.createdAt, + })); + this.hasMore = hasMore; + } +} diff --git a/packages/backend/src/common/dateEmbedded.entity.ts b/packages/backend/src/common/dateEmbedded.entity.ts index 2a0c9dd8..a16c1cf9 100644 --- a/packages/backend/src/common/dateEmbedded.entity.ts +++ b/packages/backend/src/common/dateEmbedded.entity.ts @@ -2,8 +2,8 @@ import { CreateDateColumn, UpdateDateColumn } from 'typeorm'; export class DateEmbedded { @CreateDateColumn({ type: 'timestamp', name: 'created_at' }) - createdAt?: Date; + createdAt: Date; @UpdateDateColumn({ type: 'timestamp', name: 'updated_at' }) - updatedAt?: Date; + updatedAt: Date; } diff --git a/packages/backend/src/main.ts b/packages/backend/src/main.ts index 14613072..0be16e68 100644 --- a/packages/backend/src/main.ts +++ b/packages/backend/src/main.ts @@ -13,7 +13,12 @@ async function bootstrap() { app.setGlobalPrefix('api'); app.use(session({ ...sessionConfig, store })); - app.useGlobalPipes(new ValidationPipe({ transform: true })); + app.useGlobalPipes( + new ValidationPipe({ + transform: true, + transformOptions: { enableImplicitConversion: true }, + }), + ); useSwagger(app); app.use(passport.initialize()); app.use(passport.session()); diff --git a/packages/backend/src/stock/decorator/stock.decorator.ts b/packages/backend/src/stock/decorator/stock.decorator.ts index 1412860e..4e2f3469 100644 --- a/packages/backend/src/stock/decorator/stock.decorator.ts +++ b/packages/backend/src/stock/decorator/stock.decorator.ts @@ -1,12 +1,7 @@ /* eslint-disable @typescript-eslint/naming-convention */ -import { - Query, - ParseIntPipe, - DefaultValuePipe, - applyDecorators, -} from '@nestjs/common'; -import { ApiOperation, ApiQuery, ApiResponse } from '@nestjs/swagger'; -import { StocksResponse } from '../dto/stock.Response'; +import { applyDecorators, DefaultValuePipe, ParseIntPipe, Query } from "@nestjs/common"; +import { ApiOperation, ApiQuery, ApiResponse } from "@nestjs/swagger"; +import { StocksResponse } from "../dto/stock.response"; export function LimitQuery(defaultValue = 5): ParameterDecorator { return Query('limit', new DefaultValuePipe(defaultValue), ParseIntPipe); diff --git a/packages/backend/src/stock/dto/stock.request.ts b/packages/backend/src/stock/dto/stock.request.ts new file mode 100644 index 00000000..255e4481 --- /dev/null +++ b/packages/backend/src/stock/dto/stock.request.ts @@ -0,0 +1,12 @@ +import { ApiProperty } from '@nestjs/swagger'; +import { IsNotEmpty, IsString } from 'class-validator'; + +export class StockSearchRequest { + @ApiProperty({ + description: '검색할 단어', + example: '삼성', + }) + @IsNotEmpty() + @IsString() + name: string; +} diff --git a/packages/backend/src/stock/dto/stock.Response.ts b/packages/backend/src/stock/dto/stock.response.ts similarity index 66% rename from packages/backend/src/stock/dto/stock.Response.ts rename to packages/backend/src/stock/dto/stock.response.ts index c5139450..6ac76f31 100644 --- a/packages/backend/src/stock/dto/stock.Response.ts +++ b/packages/backend/src/stock/dto/stock.response.ts @@ -1,5 +1,6 @@ import { ApiProperty } from '@nestjs/swagger'; import { Transform } from 'class-transformer'; +import { Stock } from '@/stock/domain/stock.entity'; export class StockViewsResponse { @ApiProperty({ @@ -33,32 +34,70 @@ export class StocksResponse { example: 'A005930', }) id: string; + @ApiProperty({ description: '주식 종목 이름', example: '삼성전자', }) name: string; + @ApiProperty({ description: '주식 현재가', example: 100000.0, }) @Transform(({ value }) => parseFloat(value)) currentPrice: number; + @ApiProperty({ description: '주식 변동률', example: 2.5, }) @Transform(({ value }) => parseFloat(value)) changeRate: number; + @ApiProperty({ description: '주식 거래량', example: 500000, }) @Transform(({ value }) => parseInt(value)) volume: number; + @ApiProperty({ description: '주식 시가 총액', example: '500000000000.00', }) marketCap: string; } + +class StockSearchResult { + @ApiProperty({ + description: '주식 종목 코드', + example: 'A005930', + }) + id: string; + + @ApiProperty({ + description: '주식 종목 이름', + example: '삼성전자', + }) + name: string; +} + +export class StockSearchResponse { + @ApiProperty({ + description: '주식 검색 결과', + type: [StockSearchResult], + }) + searchResults: StockSearchResult[]; + + constructor(stocks?: Stock[]) { + if (!stocks) { + this.searchResults = []; + return; + } + this.searchResults = stocks.map((stock) => ({ + id: stock.id as string, + name: stock.name as string, + })); + } +} diff --git a/packages/backend/src/stock/stock.controller.ts b/packages/backend/src/stock/stock.controller.ts index 042353a3..ed85f3c1 100644 --- a/packages/backend/src/stock/stock.controller.ts +++ b/packages/backend/src/stock/stock.controller.ts @@ -32,7 +32,10 @@ import { StockDetailService } from './stockDetail.service'; import SessionGuard from '@/auth/session/session.guard'; import { GetUser } from '@/common/decorator/user.decorator'; import { sessionConfig } from '@/configs/session.config'; -import { StockViewsResponse } from '@/stock/dto/stock.Response'; +import { + StockSearchResponse, + StockViewsResponse, +} from '@/stock/dto/stock.response'; import { StockViewRequest } from '@/stock/dto/stockView.request'; import { UserStockDeleteRequest, @@ -43,6 +46,7 @@ import { UserStockResponse, } from '@/stock/dto/userStock.response'; import { User } from '@/user/domain/user.entity'; +import { StockSearchRequest } from '@/stock/dto/stock.request'; @Controller('stock') export class StockController { @@ -150,6 +154,20 @@ export class StockController { return new UserStockOwnerResponse(result); } + @ApiOperation({ + summary: '주식 검색 API', + description: '주식 이름에 매칭되는 주식을 검색', + }) + @ApiOkResponse({ + description: '검색 완료', + type: StockSearchResponse, + }) + @Get() + async searchStock(@Query() request: StockSearchRequest) { + console.log(request.name); + return await this.stockService.searchStock(request.name); + } + @Get(':stockId/minutely') @ApiGetStockData('주식 분 단위 데이터 조회 API', '분') async getStockDataMinutely( diff --git a/packages/backend/src/stock/stock.service.ts b/packages/backend/src/stock/stock.service.ts index 1f23e48f..154eb9d6 100644 --- a/packages/backend/src/stock/stock.service.ts +++ b/packages/backend/src/stock/stock.service.ts @@ -3,7 +3,7 @@ import { plainToInstance } from 'class-transformer'; import { DataSource, EntityManager } from 'typeorm'; import { Logger } from 'winston'; import { Stock } from './domain/stock.entity'; -import { StocksResponse } from './dto/stock.Response'; +import { StockSearchResponse, StocksResponse } from './dto/stock.response'; import { UserStock } from '@/stock/domain/userStock.entity'; @Injectable() @@ -68,6 +68,20 @@ export class StockService { }); } + async searchStock(stockName: string) { + const queryBuilder = this.datasource + .getRepository(Stock) + .createQueryBuilder(); + const result = await queryBuilder + .where('stock.stock_name LIKE :name', { + isTrading: true, + name: `%${stockName}%`, + }) + .limit(10) + .getMany(); + return new StockSearchResponse(result); + } + validateUserStock(userId: number, userStock: UserStock | null) { if (!userStock) { throw new BadRequestException('user stock not found');