From 07d31bfe7dfe271b0139252410038e9707dbdab1 Mon Sep 17 00:00:00 2001 From: Julian Krauser Date: Tue, 21 Jan 2025 14:49:14 +0100 Subject: [PATCH] member and membership statistics --- src/controller/admin/club/memberController.ts | 26 +++++++ src/factory/admin/club/member/member.ts | 22 +++++- src/factory/admin/club/member/membership.ts | 26 ++++++- src/routes/admin/club/member.ts | 74 +++++++++++-------- src/service/club/member/memberService.ts | 20 +++++ src/service/club/member/membershipService.ts | 20 +++++ .../admin/club/member/member.models.ts | 12 +++ .../admin/club/member/membership.models.ts | 15 ++++ src/views/memberView.ts | 2 +- src/views/membershipsView.ts | 2 +- 10 files changed, 183 insertions(+), 36 deletions(-) diff --git a/src/controller/admin/club/memberController.ts b/src/controller/admin/club/memberController.ts index d4a9f7e..1111346 100644 --- a/src/controller/admin/club/memberController.ts +++ b/src/controller/admin/club/memberController.ts @@ -90,6 +90,19 @@ export async function getMemberById(req: Request, res: Response): Promise { res.json(MemberFactory.mapToSingle(member)); } +/** + * @description get member statistics by id + * @param req {Request} Express req object + * @param res {Response} Express res object + * @returns {Promise<*>} + */ +export async function getMemberStatisticsById(req: Request, res: Response): Promise { + const memberId = parseInt(req.params.id); + let member = await MemberService.getStatisticsById(memberId); + + res.json(MemberFactory.mapToMemberStatistic(member)); +} + /** * @description get memberships by member * @param req {Request} Express req object @@ -103,6 +116,19 @@ export async function getMembershipsByMember(req: Request, res: Response): Promi res.json(MembershipFactory.mapToBase(memberships)); } +/** + * @description get member statistics by id + * @param req {Request} Express req object + * @param res {Response} Express res object + * @returns {Promise<*>} + */ +export async function getMembershipStatisticsById(req: Request, res: Response): Promise { + const memberId = parseInt(req.params.id); + let member = await MembershipService.getStatisticsById(memberId); + + res.json(MembershipFactory.mapToMembershipStatistic(member)); +} + /** * @description get membership by member and record * @param req {Request} Express req object diff --git a/src/factory/admin/club/member/member.ts b/src/factory/admin/club/member/member.ts index 42faf7c..116a0c8 100644 --- a/src/factory/admin/club/member/member.ts +++ b/src/factory/admin/club/member/member.ts @@ -1,5 +1,6 @@ import { member } from "../../../../entity/club/member/member"; -import { MemberViewModel } from "../../../../viewmodel/admin/club/member/member.models"; +import { MemberStatisticsViewModel, MemberViewModel } from "../../../../viewmodel/admin/club/member/member.models"; +import { memberView } from "../../../../views/memberView"; import CommunicationFactory from "./communication"; import MembershipFactory from "./membership"; @@ -40,4 +41,23 @@ export default abstract class MemberFactory { public static mapToBase(records: Array): Array { return records.map((r) => this.mapToSingle(r)); } + + /** + * @description map view record to MemberMembershipStatisticsViewModel + * @param {memberView} record + * @returns {MemberStatisticsViewModel} + */ + public static mapToMemberStatistic(record: memberView): MemberStatisticsViewModel { + return { + id: record.id, + salutation: record.salutation, + firstname: record.firstname, + lastname: record.lastname, + nameaffix: record.nameaffix, + birthdate: record.birthdate, + todayAge: record.todayAge, + ageThisYear: record.ageThisYear, + exactAge: record.exactAge, + }; + } } diff --git a/src/factory/admin/club/member/membership.ts b/src/factory/admin/club/member/membership.ts index 8736785..56e0dbc 100644 --- a/src/factory/admin/club/member/membership.ts +++ b/src/factory/admin/club/member/membership.ts @@ -1,5 +1,9 @@ import { membership } from "../../../../entity/club/member/membership"; -import { MembershipViewModel } from "../../../../viewmodel/admin/club/member/membership.models"; +import { + MembershipStatisticsViewModel, + MembershipViewModel, +} from "../../../../viewmodel/admin/club/member/membership.models"; +import { membershipView } from "../../../../views/membershipsView"; export default abstract class MembershipFactory { /** @@ -26,4 +30,24 @@ export default abstract class MembershipFactory { public static mapToBase(records: Array): Array { return records.map((r) => this.mapToSingle(r)); } + + /** + * @description map view record to MembershipStatisticsViewModel + * @param {membershipView} record + * @returns {MembershipStatisticsViewModel} + */ + public static mapToMembershipStatistic(record: membershipView): MembershipStatisticsViewModel { + return { + durationInDays: record.durationInDays, + durationInYears: record.durationInYears, + status: record.status, + statusId: record.statusId, + memberId: record.memberId, + memberSalutation: record.memberSalutation, + memberFirstname: record.memberFirstname, + memberLastname: record.memberLastname, + memberNameaffix: record.memberNameaffix, + memberBirthdate: record.memberBirthdate, + }; + } } diff --git a/src/routes/admin/club/member.ts b/src/routes/admin/club/member.ts index f014d37..f621e0e 100644 --- a/src/routes/admin/club/member.ts +++ b/src/routes/admin/club/member.ts @@ -1,36 +1,38 @@ import express, { Request, Response } from "express"; import { - addAwardToMember, - addCommunicationToMember, - addExecutivePositionToMember, - addMembershipToMember, - addQualificationToMember, - createMember, - createMemberPrintoutList, - deleteAwardOfMember, - deleteCommunicationOfMember, - deleteExecutivePositionOfMember, - deleteMemberById, - deleteMembershipOfMember, - deleteQualificationOfMember, - getAllMembers, - getAwardByMemberAndRecord, - getAwardsByMember, - getCommunicationByMemberAndRecord, - getCommunicationsByMember, - getExecutivePositionByMemberAndRecord, - getExecutivePositionsByMember, - getMemberById, - getMembershipByMemberAndRecord, - getMembershipsByMember, - getQualificationByMemberAndRecord, - getQualificationsByMember, - updateAwardOfMember, - updateCommunicationOfMember, - updateExecutivePositionOfMember, - updateMemberById, - updateMembershipOfMember, - updateQualificationOfMember, + addAwardToMember, + addCommunicationToMember, + addExecutivePositionToMember, + addMembershipToMember, + addQualificationToMember, + createMember, + createMemberPrintoutList, + deleteAwardOfMember, + deleteCommunicationOfMember, + deleteExecutivePositionOfMember, + deleteMemberById, + deleteMembershipOfMember, + deleteQualificationOfMember, + getAllMembers, + getAwardByMemberAndRecord, + getAwardsByMember, + getCommunicationByMemberAndRecord, + getCommunicationsByMember, + getExecutivePositionByMemberAndRecord, + getExecutivePositionsByMember, + getMemberById, + getMembershipByMemberAndRecord, + getMembershipsByMember, + getMembershipStatisticsById, + getMemberStatisticsById, + getQualificationByMemberAndRecord, + getQualificationsByMember, + updateAwardOfMember, + updateCommunicationOfMember, + updateExecutivePositionOfMember, + updateMemberById, + updateMembershipOfMember, + updateQualificationOfMember, } from "../../../controller/admin/club/memberController"; import PermissionHelper from "../../../helpers/permissionHelper"; @@ -44,14 +46,22 @@ router.get("/:id", async (req: Request, res: Response) => { await getMemberById(req, res); }); +router.get("/:id/statistics", async (req: Request, res: Response) => { + await getMemberStatisticsById(req, res); +}); + router.get("/print/namelist", async (req: Request, res: Response) => { - await createMemberPrintoutList(req, res); + await createMemberPrintoutList(req, res); }); router.get("/:memberId/memberships", async (req: Request, res: Response) => { await getMembershipsByMember(req, res); }); +router.get("/:memberId/statistics/memberships", async (req: Request, res: Response) => { + await getMembershipStatisticsById(req, res); +}); + router.get("/:memberId/membership/:id", async (req: Request, res: Response) => { await getMembershipByMemberAndRecord(req, res); }); diff --git a/src/service/club/member/memberService.ts b/src/service/club/member/memberService.ts index c7da55e..3b08f9f 100644 --- a/src/service/club/member/memberService.ts +++ b/src/service/club/member/memberService.ts @@ -2,6 +2,7 @@ import { dataSource } from "../../../data-source"; import { member } from "../../../entity/club/member/member"; import { membership } from "../../../entity/club/member/membership"; import InternalException from "../../../exceptions/internalException"; +import { memberView } from "../../../views/memberView"; export default abstract class MemberService { /** @@ -132,6 +133,25 @@ export default abstract class MemberService { }); } + /** + * @description get member statistics by id + * @param {number} id + * @returns {Promise} + */ + static async getStatisticsById(id: number): Promise { + return await dataSource + .getRepository(memberView) + .createQueryBuilder("memberView") + .where("memberView.id = :id", { id: id }) + .getOneOrFail() + .then((res) => { + return res; + }) + .catch((err) => { + throw new InternalException("memberView not found by id", err); + }); + } + /** * @description get members where membership is setz * @returns {Promise} diff --git a/src/service/club/member/membershipService.ts b/src/service/club/member/membershipService.ts index a00daf2..604151f 100644 --- a/src/service/club/member/membershipService.ts +++ b/src/service/club/member/membershipService.ts @@ -1,6 +1,7 @@ import { dataSource } from "../../../data-source"; import { membership } from "../../../entity/club/member/membership"; import InternalException from "../../../exceptions/internalException"; +import { membershipView } from "../../../views/membershipsView"; export default abstract class MembershipService { /** @@ -45,4 +46,23 @@ export default abstract class MembershipService { throw new InternalException("member membership not found by id", err); }); } + + /** + * @description get membership statistics by memberId + * @param {number} memberId + * @returns {Promise} + */ + static async getStatisticsById(memberId: number): Promise { + return await dataSource + .getRepository(membershipView) + .createQueryBuilder("membershipView") + .where("membershipView.memberId = :memberId", { memberId: memberId }) + .getOneOrFail() + .then((res) => { + return res; + }) + .catch((err) => { + throw new InternalException("membershipView not found by id", err); + }); + } } diff --git a/src/viewmodel/admin/club/member/member.models.ts b/src/viewmodel/admin/club/member/member.models.ts index 60e1a27..c28ed73 100644 --- a/src/viewmodel/admin/club/member/member.models.ts +++ b/src/viewmodel/admin/club/member/member.models.ts @@ -16,3 +16,15 @@ export interface MemberViewModel { smsAlarming?: Array; preferredCommunication?: Array; } + +export interface MemberStatisticsViewModel { + id: number; + salutation: Salutation; + firstname: string; + lastname: string; + nameaffix: string; + birthdate: Date; + todayAge: number; + ageThisYear: number; + exactAge: string; +} diff --git a/src/viewmodel/admin/club/member/membership.models.ts b/src/viewmodel/admin/club/member/membership.models.ts index 59e2aba..da8bffa 100644 --- a/src/viewmodel/admin/club/member/membership.models.ts +++ b/src/viewmodel/admin/club/member/membership.models.ts @@ -1,3 +1,5 @@ +import { Salutation } from "../../../../enums/salutation"; + export interface MembershipViewModel { id: number; start: Date; @@ -6,3 +8,16 @@ export interface MembershipViewModel { status: string; statusId: number; } + +export interface MembershipStatisticsViewModel { + durationInDays: number; + durationInYears: string; + status: string; + statusId: number; + memberId: number; + memberSalutation: Salutation; + memberFirstname: string; + memberLastname: string; + memberNameaffix: string; + memberBirthdate: Date; +} diff --git a/src/views/memberView.ts b/src/views/memberView.ts index 1743ed4..6a980f0 100644 --- a/src/views/memberView.ts +++ b/src/views/memberView.ts @@ -43,5 +43,5 @@ export class memberView { ageThisYear: number; @ViewColumn() - exactAge: Date; + exactAge: string; } diff --git a/src/views/membershipsView.ts b/src/views/membershipsView.ts index 0b5df92..1e40209 100644 --- a/src/views/membershipsView.ts +++ b/src/views/membershipsView.ts @@ -30,7 +30,7 @@ export class membershipView { durationInDays: number; @ViewColumn() - durationInYears: Date; + durationInYears: string; @ViewColumn() status: string;