jws-backend/src/controllers/00-stats-controller.ts

384 lines
10 KiB
TypeScript
Raw Normal View History

2025-03-04 15:19:44 +07:00
import config from "../config.json";
2025-03-04 13:42:10 +07:00
import {
Customer,
CustomerBranch,
ProductGroup,
QuotationStatus,
RequestWorkStatus,
User,
} from "@prisma/client";
2025-03-04 14:37:13 +07:00
import { Controller, Get, Query, Request, Route, Security, Tags } from "tsoa";
2025-03-04 11:27:15 +07:00
import prisma from "../db";
import { createPermCondition } from "../services/permission";
import { RequestWithUser } from "../interfaces/user";
import { PaymentStatus } from "../generated/kysely/types";
2025-03-04 15:19:44 +07:00
import { precisionRound } from "../utils/arithmetic";
2025-03-04 11:27:15 +07:00
const permissionCondCompany = createPermCondition((_) => true);
2025-03-04 15:19:44 +07:00
const VAT_DEFAULT = config.vat;
2025-03-04 11:27:15 +07:00
@Route("/api/v1/report")
@Security("keycloak")
2025-03-04 14:37:13 +07:00
@Tags("Report")
2025-03-04 11:27:15 +07:00
export class StatsController extends Controller {
@Get("quotation")
async quotationReport(
@Request() req: RequestWithUser,
@Query() limit?: number,
@Query() startDate?: Date,
@Query() endDate?: Date,
) {
const record = await prisma.quotation.findMany({
select: {
code: true,
quotationStatus: true,
createdAt: true,
updatedAt: true,
},
where: {
registeredBranch: { OR: permissionCondCompany(req.user) },
createdAt: { gte: startDate, lte: endDate },
},
2025-03-04 13:50:23 +07:00
orderBy: { createdAt: "desc" },
2025-03-04 11:27:15 +07:00
take: limit,
});
return record.map((v) => ({
document: "quotation",
code: v.code,
status: v.quotationStatus,
createdAt: v.createdAt,
updatedAt: v.updatedAt,
}));
}
@Get("invoice")
async invoiceReport(
@Request() req: RequestWithUser,
@Query() limit?: number,
@Query() startDate?: Date,
@Query() endDate?: Date,
) {
const record = await prisma.invoice.findMany({
select: {
code: true,
payment: {
select: {
paymentStatus: true,
},
},
createdAt: true,
},
where: {
quotation: {
isDebitNote: false,
registeredBranch: { OR: permissionCondCompany(req.user) },
},
createdAt: { gte: startDate, lte: endDate },
},
2025-03-04 13:50:23 +07:00
orderBy: { createdAt: "desc" },
2025-03-04 11:27:15 +07:00
take: limit,
});
return record.map((v) => ({
document: "invoice",
code: v.code,
status: v.payment?.paymentStatus,
createdAt: v.createdAt,
}));
}
@Get("receipt")
async receiptReport(
@Request() req: RequestWithUser,
@Query() limit?: number,
@Query() startDate?: Date,
@Query() endDate?: Date,
) {
const record = await prisma.payment.findMany({
select: {
code: true,
paymentStatus: true,
createdAt: true,
},
where: {
paymentStatus: PaymentStatus.PaymentSuccess,
invoice: {
quotation: {
isDebitNote: false,
registeredBranch: { OR: permissionCondCompany(req.user) },
},
},
createdAt: { gte: startDate, lte: endDate },
},
2025-03-04 13:50:23 +07:00
orderBy: { createdAt: "desc" },
2025-03-04 11:27:15 +07:00
take: limit,
});
return record.map((v) => ({
document: "receipt",
code: v.code,
status: v.paymentStatus,
createdAt: v.createdAt,
}));
}
@Get("product")
async productReport(
@Request() req: RequestWithUser,
@Query() limit?: number,
@Query() startDate?: Date,
@Query() endDate?: Date,
) {
2025-03-04 15:19:44 +07:00
await prisma.$transaction(async (tx) => {
const record = await tx.product.findMany({
include: {
quotationProductServiceList: {
include: {
quotation: true,
},
2025-03-04 13:42:01 +07:00
},
},
2025-03-04 15:19:44 +07:00
select: {
id: true,
code: true,
name: true,
createdAt: true,
updatedAt: true,
_count: {
select: {
quotationProductServiceList: {
where: {
quotation: {
quotationStatus: {
in: [
QuotationStatus.PaymentInProcess,
QuotationStatus.PaymentSuccess,
QuotationStatus.ProcessComplete,
],
},
2025-03-04 11:27:15 +07:00
},
},
},
},
},
},
2025-03-04 15:19:44 +07:00
where: {
quotationProductServiceList: {
some: {
quotation: { createdAt: { gte: startDate, lte: endDate } },
},
2025-03-04 13:42:01 +07:00
},
2025-03-04 15:19:44 +07:00
productGroup: { registeredBranch: { OR: permissionCondCompany(req.user) } },
2025-03-04 13:42:01 +07:00
},
2025-03-04 15:19:44 +07:00
orderBy: {
quotationProductServiceList: { _count: "desc" },
},
2025-03-04 15:19:44 +07:00
take: limit,
});
const doing = await tx.quotationProductServiceList.groupBy({
_count: true,
by: "productId",
where: {
quotation: {
createdAt: { gte: startDate, lte: endDate },
registeredBranch: { OR: permissionCondCompany(req.user) },
},
productId: { in: record.map((v) => v.id) },
requestWork: {
some: {
stepStatus: {
some: {
workStatus: {
in: [
RequestWorkStatus.Pending,
RequestWorkStatus.InProgress,
RequestWorkStatus.Validate,
RequestWorkStatus.Completed,
RequestWorkStatus.Ended,
],
},
2025-03-04 11:27:15 +07:00
},
},
},
},
},
2025-03-04 15:19:44 +07:00
});
2025-03-04 11:27:15 +07:00
2025-03-04 15:19:44 +07:00
const order = await tx.quotationProductServiceList.groupBy({
_count: true,
by: "productId",
where: {
quotation: {
createdAt: { gte: startDate, lte: endDate },
registeredBranch: { OR: permissionCondCompany(req.user) },
},
productId: { in: record.map((v) => v.id) },
},
2025-03-04 15:19:44 +07:00
});
2025-03-04 15:19:44 +07:00
return record.map((v) => ({
document: "product",
code: v.code,
name: v.name,
sale: v._count.quotationProductServiceList,
did: doing.find((item) => item.productId === v.id)?._count || 0,
order: order.find((item) => item.productId === v.id)?._count || 0,
createdAt: v.createdAt,
updatedAt: v.updatedAt,
}));
});
2025-03-04 11:27:15 +07:00
}
2025-03-04 11:46:11 +07:00
@Get("sale")
async saleReport(
@Request() req: RequestWithUser,
@Query() limit?: number,
@Query() startDate?: Date,
@Query() endDate?: Date,
2025-03-04 13:42:10 +07:00
) {
const list = await prisma.quotationProductServiceList.findMany({
include: {
quotation: {
include: {
createdBy: true,
customerBranch: {
include: { customer: true },
},
},
},
product: {
include: {
productGroup: true,
},
},
},
where: {
quotation: {
isDebitNote: false,
registeredBranch: { OR: permissionCondCompany(req.user) },
createdAt: { gte: startDate, lte: endDate },
quotationStatus: {
in: [
QuotationStatus.PaymentInProcess,
QuotationStatus.PaymentSuccess,
QuotationStatus.ProcessComplete,
],
},
},
},
take: limit,
});
return list.reduce<{
byProductGroup: (ProductGroup & { _count: number })[];
bySale: (User & { _count: number })[];
byCustomer: ((CustomerBranch & { customer: Customer }) & { _count: number })[];
}>(
(a, c) => {
{
const found = a.byProductGroup.find((v) => v.id === c.product.productGroupId);
if (found) {
found._count++;
} else {
a.byProductGroup.push({ ...c.product.productGroup, _count: 1 });
}
}
{
const found = a.bySale.find((v) => v.id === c.quotation.createdByUserId);
if (found) {
found._count++;
} else {
if (c.quotation.createdBy) {
a.bySale.push({ ...c.quotation.createdBy, _count: 1 });
}
}
}
{
const found = a.byCustomer.find((v) => v.id === c.quotation.customerBranchId);
if (found) {
found._count++;
} else {
a.byCustomer.push({ ...c.quotation.customerBranch, _count: 1 });
}
}
return a;
},
{ byProductGroup: [], bySale: [], byCustomer: [] },
);
}
2025-03-04 15:19:44 +07:00
@Get("profit")
async profit(
@Request() req: RequestWithUser,
@Query() limit?: number,
@Query() startDate?: Date,
@Query() endDate?: Date,
) {
const record = await prisma.quotationProductServiceList.findMany({
include: {
product: {
select: {
agentPrice: true,
agentPriceCalcVat: true,
agentPriceVatIncluded: true,
serviceCharge: true,
serviceChargeCalcVat: true,
serviceChargeVatIncluded: true,
price: true,
calcVat: true,
vatIncluded: true,
},
},
quotation: {
select: {
agentPrice: true,
},
},
},
where: {
quotation: {
quotationStatus: {
in: [
QuotationStatus.PaymentInProcess,
QuotationStatus.PaymentSuccess,
QuotationStatus.ProcessComplete,
],
},
registeredBranch: {
OR: permissionCondCompany(req.user),
},
},
},
take: limit,
});
let income = 0;
let expenses = 0;
let netProfit = 0;
record.forEach((v) => {
const originalPrice = v.product.serviceCharge;
const productExpenses = precisionRound(
originalPrice + (v.product.serviceChargeVatIncluded ? 0 : originalPrice * VAT_DEFAULT),
);
const finalPrice = v.pricePerUnit * v.amount * (1 + config.vat);
income += finalPrice;
expenses += productExpenses;
netProfit += finalPrice - productExpenses;
});
return {
income: precisionRound(income),
expenses: precisionRound(expenses),
netProfit: precisionRound(netProfit),
};
}
2025-03-04 11:27:15 +07:00
}