-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[#12] 4.04 오늘의 상/하위 종목 API 구현 #30
Changes from 7 commits
a86bf21
414b03c
2af0a7d
f8c7046
ba8e2d6
7736420
3c1ffcb
4375e5f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
import { ApiProperty } from '@nestjs/swagger'; | ||
|
||
/** | ||
* 등락률 API 요청 후 받은 응답값 정제용 DTO | ||
*/ | ||
export class StockRankingDataDto { | ||
@ApiProperty({ description: 'HTS 한글 종목명' }) | ||
hts_kor_isnm: string; | ||
|
||
@ApiProperty({ description: '주식 현재가' }) | ||
stck_prpr: string; | ||
|
||
@ApiProperty({ description: '전일 대비' }) | ||
prdy_vrss: string; | ||
|
||
@ApiProperty({ description: '전일 대비 부호' }) | ||
prdy_vrss_sign: string; | ||
|
||
@ApiProperty({ description: '전일 대비율' }) | ||
prdy_ctrt: string; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
/** | ||
* 등락률 API를 사용할 때 쿼리 파라미터로 사용할 요청값 DTO | ||
*/ | ||
export class StockRankigRequestDto { | ||
/** | ||
* 조건 시장 분류 코드 | ||
* 'J' 주식 | ||
*/ | ||
fid_cond_mrkt_div_code: string; | ||
|
||
/** | ||
* 입력 종목 코드 | ||
* '0000' 전체 / '0001' 코스피 | ||
* '1001' 코스닥 / '2001' 코스피200 | ||
*/ | ||
fid_input_iscd: string; | ||
|
||
/** | ||
* 순위 정렬 구분 코드 | ||
* '0' 상승률 / '1' 하락률 | ||
*/ | ||
fid_rank_sort_cls_code: string; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
import { ApiProperty } from '@nestjs/swagger'; | ||
import { StockRankingDataDto } from './stock-ranking-data.dto'; | ||
|
||
/** | ||
* 순위 정렬 후 FE에 보낼 DTO | ||
*/ | ||
export class StockRankingResponseDto { | ||
@ApiProperty({ type: [StockRankingDataDto], description: '상승률 순위' }) | ||
high: StockRankingDataDto[]; | ||
|
||
@ApiProperty({ type: [StockRankingDataDto], description: '하락률 순위' }) | ||
low: StockRankingDataDto[]; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
import { ApiOperation, ApiQuery, ApiResponse } from '@nestjs/swagger'; | ||
import { Controller, Get, Query } from '@nestjs/common'; | ||
import { TopFiveService, MarketType } from './topfive.service'; | ||
import { StockRankingResponseDto } from './dto/stock-ranking-response.dto'; | ||
|
||
@Controller('/api/stocks') | ||
export class TopfiveController { | ||
constructor(private readonly topFiveService: TopFiveService) {} | ||
|
||
@Get('topfive') | ||
@ApiOperation({ summary: '오늘의 상/하위 종목 조회 API' }) | ||
@ApiQuery({ | ||
name: 'market', | ||
enum: MarketType, | ||
required: true, | ||
description: | ||
'주식 시장 구분\n' + | ||
'ALL: 전체, KOSPI: 코스피, KOSDAQ: 코스닥, KOSPI200: 코스피200', | ||
}) | ||
@ApiResponse({ | ||
status: 200, | ||
description: '주식 시장별 순위 조회 성공', | ||
type: StockRankingResponseDto, | ||
}) | ||
async getTopFive(@Query('market') market: MarketType) { | ||
return this.topFiveService.getMarketRanking(market); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
import { Module } from '@nestjs/common'; | ||
import { ConfigModule } from '@nestjs/config'; | ||
import { TopfiveController } from './topfive.controller'; | ||
import { TopFiveService } from './topfive.service'; | ||
|
||
@Module({ | ||
imports: [ConfigModule], | ||
controllers: [TopfiveController], | ||
providers: [TopFiveService], | ||
}) | ||
export class TopfiveModule {} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,197 @@ | ||
import axios from 'axios'; | ||
import { Injectable } from '@nestjs/common'; | ||
import { ConfigService } from '@nestjs/config'; | ||
import { StockRankigRequestDto } from './dto/stock-ranking-request.dto'; | ||
import { StockRankingResponseDto } from './dto/stock-ranking-response.dto'; | ||
import { StockRankingDataDto } from './dto/stock-ranking-data.dto'; | ||
|
||
export enum MarketType { | ||
ALL = 'ALL', | ||
KOSPI = 'KOSPI', | ||
KOSDAQ = 'KOSDAQ', | ||
KOSPI200 = 'KOSPI200', | ||
} | ||
|
||
interface StockApiOutputData { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🟢 인터페이스 다른 파일에 선언하니까 린트 에러 나서 여기 두신건가요?! There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 아니 저 방금 시은님 코드 리뷰에 이 내용 달고 왔는데 같은 생각을 ....ㅎㅎㅎ 나중에 한번 이야기해봐요!! |
||
stck_shrn_iscd: string; | ||
data_rank: string; | ||
hts_kor_isnm: string; | ||
stck_prpr: string; | ||
prdy_vrss: string; | ||
prdy_vrss_sign: string; | ||
prdy_ctrt: string; | ||
acml_vol: string; | ||
stck_hgpr: string; | ||
hgpr_hour: string; | ||
acml_hgpr_data: string; | ||
stck_lwpr: string; | ||
lwpr_hour: string; | ||
acml_lwpr_date: string; | ||
lwpr_vrss_prpr_rate: string; | ||
dsgt_date_clpr_vrss_prpr_rate: string; | ||
cnnt_ascn_dynu: string; | ||
hgpr_vrss_prpr_rate: string; | ||
cnnt_down_dynu: string; | ||
oprc_vrss_prpr_sign: string; | ||
oprc_vrss_prpr: string; | ||
oprc_vrss_prpr_rate: string; | ||
prd_rsfl: string; | ||
prd_rsfl_rate: string; | ||
} | ||
|
||
interface StockApiResponse { | ||
output: StockApiOutputData[]; | ||
rt_cd: string; | ||
msg_cd: string; | ||
msg1: string; | ||
} | ||
|
||
@Injectable() | ||
export class TopFiveService { | ||
private accessToken: string; | ||
private tokenExpireTime: Date; | ||
private readonly koreaInvestmentConfig: { | ||
appKey: string; | ||
appSecret: string; | ||
baseUrl: string; | ||
}; | ||
|
||
constructor(private readonly config: ConfigService) { | ||
this.koreaInvestmentConfig = { | ||
appKey: this.config.get<string>('KOREA_INVESTMENT_APP_KEY'), | ||
appSecret: this.config.get<string>('KOREA_INVESTMENT_APP_SECRET'), | ||
baseUrl: this.config.get<string>('KOREA_INVESTMENT_BASE_URL'), | ||
}; | ||
} | ||
|
||
private async getAccessToken() { | ||
// accessToken이 유효한 경우 | ||
if (this.accessToken && this.tokenExpireTime > new Date()) { | ||
return this.accessToken; | ||
} | ||
|
||
const response = await axios.post( | ||
`${this.koreaInvestmentConfig.baseUrl}/oauth2/tokenP`, | ||
{ | ||
grant_type: 'client_credentials', | ||
appkey: this.koreaInvestmentConfig.appKey, | ||
appsecret: this.koreaInvestmentConfig.appSecret, | ||
}, | ||
); | ||
|
||
this.accessToken = response.data.access_token; | ||
this.tokenExpireTime = new Date(Date.now() + +response.data.expires_in); | ||
|
||
return this.accessToken; | ||
} | ||
|
||
private async requestApi(params: StockRankigRequestDto) { | ||
// eslint-disable-next-line no-useless-catch | ||
try { | ||
const token = await this.getAccessToken(); | ||
|
||
const response = await axios.get<StockApiResponse>( | ||
`${this.koreaInvestmentConfig.baseUrl}/uapi/domestic-stock/v1/ranking/fluctuation`, | ||
{ | ||
headers: { | ||
'content-type': 'application/json; charset=utf-8', | ||
authorization: `Bearer ${token}`, | ||
appkey: this.koreaInvestmentConfig.appKey, | ||
appsecret: this.koreaInvestmentConfig.appSecret, | ||
tr_id: 'FHPST01700000', | ||
custtype: 'P', | ||
}, | ||
params: { | ||
fid_rsfl_rate2: '', | ||
fid_cond_mrkt_div_code: params.fid_cond_mrkt_div_code, | ||
fid_cond_scr_div_code: '20170', | ||
fid_input_iscd: params.fid_input_iscd, | ||
fid_rank_sort_cls_code: params.fid_rank_sort_cls_code, | ||
fid_input_cnt_1: '0', | ||
fid_prc_cls_code: '1', | ||
fid_input_price_1: '', | ||
fid_input_price_2: '', | ||
fid_vol_cnt: '', | ||
fid_trgt_cls_code: '0', | ||
fid_trgt_exls_cls_code: '0', | ||
fid_div_cls_code: '0', | ||
fid_rsfl_rate1: '', | ||
}, | ||
}, | ||
); | ||
return response.data; | ||
} catch (error) { | ||
// console.error('API Error Details:', { | ||
// status: error.response?.status, | ||
// statusText: error.response?.statusText, | ||
// data: error.response?.data, | ||
// headers: error.response?.config?.headers, // 실제 요청 헤더 | ||
// message: error.message, | ||
// }); | ||
throw error; | ||
} | ||
} | ||
|
||
async getMarketRanking(marketType: MarketType) { | ||
// eslint-disable-next-line no-useless-catch | ||
try { | ||
const params = new StockRankigRequestDto(); | ||
params.fid_cond_mrkt_div_code = 'J'; | ||
|
||
switch (marketType) { | ||
case MarketType.ALL: | ||
params.fid_input_iscd = '0000'; | ||
break; | ||
case MarketType.KOSPI: | ||
params.fid_input_iscd = '0001'; | ||
break; | ||
case MarketType.KOSDAQ: | ||
params.fid_input_iscd = '1001'; | ||
break; | ||
case MarketType.KOSPI200: | ||
params.fid_input_iscd = '2001'; | ||
break; | ||
default: | ||
break; | ||
} | ||
|
||
const highResponse = await this.requestApi({ | ||
...params, | ||
fid_rank_sort_cls_code: '0', | ||
}); | ||
|
||
const lowResponse = await this.requestApi({ | ||
...params, | ||
fid_rank_sort_cls_code: '1', | ||
}); | ||
|
||
const response = new StockRankingResponseDto(); | ||
response.high = this.formatStockData(highResponse.output); | ||
response.low = this.formatStockData(lowResponse.output); | ||
|
||
return response; | ||
} catch (error) { | ||
// console.error('API Error Details:', { | ||
// status: error.response?.status, | ||
// statusText: error.response?.statusText, | ||
// data: error.response?.data, | ||
// headers: error.response?.config?.headers, // 실제 요청 헤더 | ||
// message: error.message, | ||
// }); | ||
throw error; | ||
} | ||
} | ||
|
||
private formatStockData(stocks: StockApiOutputData[]) { | ||
return stocks.slice(0, 5).map((stock) => { | ||
const stockData = new StockRankingDataDto(); | ||
stockData.hts_kor_isnm = stock.hts_kor_isnm; | ||
stockData.stck_prpr = stock.stck_prpr; | ||
stockData.prdy_vrss = stock.prdy_vrss; | ||
stockData.prdy_vrss_sign = stock.prdy_vrss_sign; | ||
stockData.prdy_ctrt = stock.prdy_ctrt; | ||
|
||
return stockData; | ||
}); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
swagger 관련 코드를 생각 못했네요..! 빨리 구현하고 저도 해보도록 하겠습니다!