Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 11 additions & 9 deletions src/purchases/dtos/purchase.dto.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,15 @@

export interface PurchaseHistoryItemDTO {
prompt_id: number;
title: string;
price: number;
seller_nickname: string;
prompt_id: number;
title: string;
price: number;
seller_nickname: string;
purchased_at: string;
status: 'Succeed' | 'Refunded';
refunded_at: string | null;
}

export interface PurchaseHistoryResponseDTO {
message: string;
purchases: PurchaseHistoryItemDTO[];
statusCode: number;
}
message: string;
purchases: PurchaseHistoryItemDTO[];
statusCode: number;
}
18 changes: 11 additions & 7 deletions src/purchases/repositories/purchase.repository.ts
Original file line number Diff line number Diff line change
@@ -1,22 +1,26 @@
import prisma from '../../config/prisma';

export const PurchaseRepository = {
findSucceededByUser(userId: number) {
// κ²°μ œλ‚΄μ—­: 결제 성곡(Succeed) + ν™˜λΆˆ(Refunded) 거래 포함.
// ν™˜λΆˆλœ κ±°λž˜λ„ μ‚¬μš©μžκ°€ κ²°μ œλ‚΄μ—­μ—μ„œ 식별할 수 μžˆμ–΄μ•Ό 함 (#497).
findPurchaseHistory(userId: number) {
return prisma.purchase.findMany({
where: {
user_id: userId,
payment: { is: { status: 'Succeed'}},
payment: { is: { status: { in: ['Succeed', 'Refunded'] } } },
},
include: {
prompt: {
select: {
prompt_id: true,
title: true,
user: { select: { nickname: true}},
user: { select: { nickname: true } },
},
},
payment: { select: { status: true } },
refund: { select: { refunded_at: true } },
},
orderBy: { created_at: 'desc'},
})
}
};
orderBy: { created_at: 'desc' },
});
},
};
9 changes: 9 additions & 0 deletions src/purchases/routes/purchase.route.ts
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,15 @@ router.post('/requests', authenticateJwt, PurchaseRequestController.requestPurch
* purchased_at:
* type: string
* format: date-time
* status:
* type: string
* enum: [Succeed, Refunded]
* description: 결제 μƒνƒœ. ν™˜λΆˆ μ‹œ Refunded
* refunded_at:
* type: string
* format: date-time
* nullable: true
* description: ν™˜λΆˆ μ‹œκ°. statusκ°€ Refunded일 λ•Œλ§Œ 쑴재
* statusCode:
* type: integer
*/
Expand Down
40 changes: 21 additions & 19 deletions src/purchases/services/purchase.service.ts
Original file line number Diff line number Diff line change
@@ -1,22 +1,24 @@
import { PurchaseRepository } from "../repositories/purchase.repository";
import { PurchaseHistoryItemDTO, PurchaseHistoryResponseDTO } from "../dtos/purchase.dto";
import { PurchaseRepository } from '../repositories/purchase.repository';
import { PurchaseHistoryItemDTO, PurchaseHistoryResponseDTO } from '../dtos/purchase.dto';

export const PurchaseHistoryService = {
async list(userId: number): Promise<PurchaseHistoryResponseDTO> {
const rows = await PurchaseRepository.findSucceededByUser(userId);

const purchases: PurchaseHistoryItemDTO[] = rows.map((r) => ({
prompt_id: r.prompt.prompt_id,
title: r.prompt.title,
price: r.amount,
purchased_at: r.created_at.toISOString(),
seller_nickname: r.prompt.user.nickname,
}));
async list(userId: number): Promise<PurchaseHistoryResponseDTO> {
const rows = await PurchaseRepository.findPurchaseHistory(userId);

return {
message: "결제 λ‚΄μ—­ 쑰회 성곡",
purchases,
statusCode: 200,
}
}
}
const purchases: PurchaseHistoryItemDTO[] = rows.map((r) => ({
prompt_id: r.prompt.prompt_id,
title: r.prompt.title,
price: r.amount,
purchased_at: r.created_at.toISOString(),
seller_nickname: r.prompt.user.nickname,
status: (r.payment?.status ?? 'Succeed') as 'Succeed' | 'Refunded',
refunded_at: r.refund?.refunded_at?.toISOString() ?? null,
}));

return {
message: '결제 λ‚΄μ—­ 쑰회 성곡',
purchases,
statusCode: 200,
};
},
};
4 changes: 3 additions & 1 deletion src/settlements/controllers/settlement.history.controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,10 @@ export const getMonthlySales = async (req: Request, res: Response) => {
const now = new Date();
const year = req.query.year ? Number(req.query.year) : now.getUTCFullYear();
const month = req.query.month ? Number(req.query.month) : now.getUTCMonth() + 1;
const page = req.query.page ? Number(req.query.page) : undefined;
const limit = req.query.limit ? Number(req.query.limit) : undefined;

const result = await SettlementHistoryService.getMonthlySales(userId, year, month);
const result = await SettlementHistoryService.getMonthlySales(userId, year, month, page, limit);
return res.status(200).json(result);
} catch (error: any) {
const status = error.status || 500;
Expand Down
14 changes: 12 additions & 2 deletions src/settlements/dtos/settlement.history.dto.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,20 +13,30 @@ export interface MonthlySalesItemDto {
status: 'Pending' | 'Succeed' | 'Failed' | 'Refunded';
}

// μ›” 전체 합계 (νŽ˜μ΄μ§€ 무관 β€” #497 νŽ˜μ΄μ§€λ„€μ΄μ…˜ λ„μž…κ³Ό ν•¨κ»˜ 의미 μ •ν•©ν™”)
export interface MonthlySalesSummaryDto {
count: number;
count: number; // μ›” 전체 거래 건수
total_sales: number;
total_settled: number; // net β€” Succeed만, ν™˜λΆˆ μ œμ™Έ
total_settled: number; // net β€” ν™˜λΆˆ μ œμ™Έ
total_fee: number;
refunded_count: number;
refunded_amount: number;
}

export interface PaginationDto {
page: number;
limit: number;
total: number;
total_pages: number;
has_next: boolean;
}

export interface MonthlySalesResponseDto {
message: string;
year: number;
month: number;
summary: MonthlySalesSummaryDto;
pagination: PaginationDto;
items: MonthlySalesItemDto[];
statusCode: number;
}
Expand Down
68 changes: 67 additions & 1 deletion src/settlements/repositories/settlement.history.repository.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,15 @@ export const SettlementHistoryRepository = {
};
},

async findSalesByMonth(userId: number, year: number, month: number) {
// 월별 판맀 λ‚΄μ—­ (νŽ˜μ΄μ§€λ„€μ΄μ…˜ 적용).
// νŽ˜μ΄μ§€ λ°μ΄ν„°λ§Œ λ°˜ν™˜. 합계/μΉ΄μš΄νŠΈλŠ” aggregateSalesByMonth/countSalesByMonth둜 별도 쑰회.
async findSalesByMonth(
userId: number,
year: number,
month: number,
skip: number,
take: number,
) {
const start = new Date(Date.UTC(year, month - 1, 1));
const end = new Date(Date.UTC(year, month, 1));

Expand All @@ -27,6 +35,8 @@ export const SettlementHistoryRepository = {
created_at: { gte: start, lt: end },
},
orderBy: { created_at: 'desc' },
skip,
take,
select: {
settlement_id: true,
amount: true,
Expand All @@ -51,6 +61,62 @@ export const SettlementHistoryRepository = {
});
},

// 월별 판맀 전체 건수 (νŽ˜μ΄μ§€λ„€μ΄μ…˜ 메타).
async countSalesByMonth(userId: number, year: number, month: number): Promise<number> {
const start = new Date(Date.UTC(year, month - 1, 1));
const end = new Date(Date.UTC(year, month, 1));
return prisma.settlement.count({
where: { user_id: userId, created_at: { gte: start, lt: end } },
});
},

// μ›” 전체 합계 (νŽ˜μ΄μ§€ 무관).
// summary 의미λ₯Ό "ν˜„μž¬ νŽ˜μ΄μ§€ 합계"κ°€ μ•„λ‹Œ "μ›” 전체 합계"둜 μ •ν•©ν™” (#497).
async aggregateSalesByMonth(
userId: number,
year: number,
month: number,
): Promise<{
total_sales: number;
total_settled: number;
total_fee: number;
refunded_amount: number;
refunded_count: number;
}> {
const start = new Date(Date.UTC(year, month - 1, 1));
const end = new Date(Date.UTC(year, month, 1));
const rows = await prisma.$queryRaw<
Array<{
total_sales: bigint | null;
total_settled: bigint | null;
total_fee: bigint | null;
refunded_amount: bigint | null;
refunded_count: bigint | null;
}>
>`
SELECT
SUM(p.amount) AS total_sales,
SUM(CASE WHEN s.status = ${Status.Refunded} THEN 0 ELSE s.amount END) AS total_settled,
SUM(s.fee) AS total_fee,
SUM(CASE WHEN s.status = ${Status.Refunded} THEN s.amount ELSE 0 END) AS refunded_amount,
SUM(CASE WHEN s.status = ${Status.Refunded} THEN 1 ELSE 0 END) AS refunded_count
FROM Settlement s
JOIN Payment pm ON pm.payment_id = s.payment_id
JOIN Purchase p ON p.purchase_id = pm.purchase_id
WHERE s.user_id = ${userId}
AND s.created_at >= ${start}
AND s.created_at < ${end}
`;
const r = rows[0] ?? {};
return {
total_sales: Number(r.total_sales ?? 0),
total_settled: Number(r.total_settled ?? 0),
total_fee: Number(r.total_fee ?? 0),
refunded_amount: Number(r.refunded_amount ?? 0),
refunded_count: Number(r.refunded_count ?? 0),
};
},

async aggregateYearlyTotals(userId: number): Promise<
Array<{
year: number;
Expand Down
36 changes: 30 additions & 6 deletions src/settlements/routes/settlement.route.ts
Original file line number Diff line number Diff line change
Expand Up @@ -455,6 +455,21 @@ router.post("/register/business", authenticateJwt, registerBusiness);
* - in: query
* name: month
* schema: { type: integer, minimum: 1, maximum: 12, example: 5 }
* - in: query
* name: page
* description: νŽ˜μ΄μ§€ 번호 1λΆ€ν„° μ‹œμž‘
* schema:
* type: integer
* minimum: 1
* default: 1
* - in: query
* name: limit
* description: νŽ˜μ΄μ§€λ‹Ή ν•­λͺ© 수 μ΅œλŒ€ 100
* schema:
* type: integer
* minimum: 1
* maximum: 100
* default: 10
* responses:
* 200:
* description: 쑰회 성곡
Expand All @@ -468,13 +483,22 @@ router.post("/register/business", authenticateJwt, registerBusiness);
* month: { type: integer, example: 5 }
* summary:
* type: object
* description: μ›” 전체 합계 (νŽ˜μ΄μ§€ 무관)
* properties:
* count: { type: integer, description: μ›” 전체 거래 건수, example: 25 }
* total_sales: { type: integer, description: μ›λž˜ νŒλ§€κ°€ 합계 (ν™˜λΆˆ 포함), example: 250000 }
* total_settled: { type: integer, description: net μ •μ‚° κΈˆμ•‘ (ν™˜λΆˆ μ œμ™Έ), example: 200000 }
* total_fee: { type: integer, example: 25000 }
* refunded_count: { type: integer, example: 2 }
* refunded_amount: { type: integer, example: 18000 }
* pagination:
* type: object
* properties:
* count: { type: integer, example: 3 }
* total_sales: { type: integer, description: μ›λž˜ νŒλ§€κ°€ 합계 (ν™˜λΆˆ 포함), example: 30000 }
* total_settled: { type: integer, description: net μ •μ‚° κΈˆμ•‘ (ν™˜λΆˆ μ œμ™Έ), example: 18000 }
* total_fee: { type: integer, example: 3000 }
* refunded_count: { type: integer, example: 1 }
* refunded_amount: { type: integer, example: 9000 }
* page: { type: integer, example: 1 }
* limit: { type: integer, example: 10 }
* total: { type: integer, example: 25 }
* total_pages: { type: integer, example: 3 }
* has_next: { type: boolean, example: true }
* items:
* type: array
* items:
Expand Down
Loading
Loading