@things-factory/kpi 10.0.0-beta.89 → 10.0.0-beta.90
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist-server/service/kpi/kpi-query.d.ts +1 -1
- package/dist-server/service/kpi/kpi-query.js +38 -16
- package/dist-server/service/kpi/kpi-query.js.map +1 -1
- package/dist-server/service/kpi-metric/kpi-metric-query.js +18 -5
- package/dist-server/service/kpi-metric/kpi-metric-query.js.map +1 -1
- package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js +10 -3
- package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js.map +1 -1
- package/dist-server/service/kpi-value/kpi-value-query.js +10 -3
- package/dist-server/service/kpi-value/kpi-value-query.js.map +1 -1
- package/dist-server/service/utils/domain-inheritance.d.ts +17 -0
- package/dist-server/service/utils/domain-inheritance.js +69 -0
- package/dist-server/service/utils/domain-inheritance.js.map +1 -0
- package/dist-server/tsconfig.tsbuildinfo +1 -1
- package/package.json +3 -3
|
@@ -9,7 +9,7 @@ export declare class KpiQuery {
|
|
|
9
9
|
kpisLevel1(rootId: string, context: ResolverContext): Promise<Kpi[]>;
|
|
10
10
|
kpi(id: string, context: ResolverContext): Promise<Kpi>;
|
|
11
11
|
kpis(params: ListParam, context: ResolverContext): Promise<KpiList>;
|
|
12
|
-
children(kpi: Kpi): Promise<Kpi[]>;
|
|
12
|
+
children(kpi: Kpi, context: ResolverContext): Promise<Kpi[]>;
|
|
13
13
|
thumbnail(kpi: Kpi): Promise<string | undefined>;
|
|
14
14
|
domain(kpi: Kpi): Promise<Domain>;
|
|
15
15
|
updater(kpi: Kpi): Promise<User>;
|
|
@@ -12,17 +12,20 @@ const kpi_type_1 = require("./kpi-type");
|
|
|
12
12
|
const kpi_value_1 = require("../kpi-value/kpi-value");
|
|
13
13
|
const kpi_history_1 = require("./kpi-history");
|
|
14
14
|
const type_graphql_2 = require("type-graphql");
|
|
15
|
+
const domain_inheritance_1 = require("../utils/domain-inheritance");
|
|
15
16
|
let KpiQuery = class KpiQuery {
|
|
16
17
|
async kpisLevel0(context) {
|
|
17
18
|
const { domain } = context.state;
|
|
19
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
18
20
|
return await (0, shell_1.getRepository)(kpi_1.Kpi).find({
|
|
19
|
-
where: { domain: { id:
|
|
21
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, parent: (0, typeorm_1.IsNull)() },
|
|
20
22
|
order: { name: 'ASC' }
|
|
21
23
|
});
|
|
22
24
|
}
|
|
23
25
|
async kpisLevel1(rootId, context) {
|
|
24
26
|
const { domain } = context.state;
|
|
25
|
-
|
|
27
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
28
|
+
let whereCondition = { domain: { id: (0, typeorm_1.In)(domainIds) } };
|
|
26
29
|
if (rootId) {
|
|
27
30
|
// 특정 부모의 자식들
|
|
28
31
|
whereCondition.parent = { id: rootId };
|
|
@@ -35,7 +38,7 @@ let KpiQuery = class KpiQuery {
|
|
|
35
38
|
.leftJoinAndSelect('kpi.parent', 'parent')
|
|
36
39
|
.leftJoinAndSelect('kpi.children', 'children')
|
|
37
40
|
.leftJoinAndSelect('children.children', 'grandchildren')
|
|
38
|
-
.where('kpi.domain
|
|
41
|
+
.where('kpi.domain IN (:...domainIds)', { domainIds })
|
|
39
42
|
.andWhere('parent.id IS NOT NULL')
|
|
40
43
|
.andWhere('parent.parent IS NULL')
|
|
41
44
|
.orderBy('kpi.name', 'ASC')
|
|
@@ -52,27 +55,41 @@ let KpiQuery = class KpiQuery {
|
|
|
52
55
|
}
|
|
53
56
|
async kpi(id, context) {
|
|
54
57
|
const { domain } = context.state;
|
|
58
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
55
59
|
return await (0, shell_1.getRepository)(kpi_1.Kpi).findOne({
|
|
56
|
-
where: { domain: { id:
|
|
60
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, id }
|
|
57
61
|
});
|
|
58
62
|
}
|
|
59
63
|
async kpis(params, context) {
|
|
60
64
|
const { domain } = context.state;
|
|
65
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
66
|
+
// closest-wins dedup 을 위해 pagination 은 dedup 이후 직접 적용한다.
|
|
67
|
+
const { pagination, ...rest } = params || {};
|
|
61
68
|
const queryBuilder = (0, shell_1.getQueryBuilderFromListParams)({
|
|
62
|
-
domain
|
|
63
|
-
|
|
64
|
-
|
|
69
|
+
// domain 필터링은 ancestor 기반으로 직접 적용 (shell helper 는 single-parent 만 지원)
|
|
70
|
+
domain: undefined,
|
|
71
|
+
params: rest,
|
|
72
|
+
repository: (0, shell_1.getRepository)(kpi_1.Kpi),
|
|
65
73
|
searchables: ['name', 'description'],
|
|
66
74
|
filtersMap: {
|
|
67
75
|
parent: { columnName: 'id', relationColumn: 'parent' }
|
|
68
76
|
}
|
|
69
77
|
});
|
|
70
|
-
|
|
78
|
+
queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {
|
|
79
|
+
ancestorDomainIds: domainIds
|
|
80
|
+
});
|
|
81
|
+
const allItems = await queryBuilder.getMany();
|
|
82
|
+
const deduped = (0, domain_inheritance_1.dedupByClosestDomain)(allItems, domainIds, k => k.name);
|
|
83
|
+
const { page = 1, limit } = pagination || {};
|
|
84
|
+
const total = deduped.length;
|
|
85
|
+
const items = limit && limit > 0 ? deduped.slice(Math.max(0, (page - 1) * limit), Math.max(0, (page - 1) * limit) + limit) : deduped;
|
|
71
86
|
return { items, total };
|
|
72
87
|
}
|
|
73
|
-
children(kpi) {
|
|
74
|
-
|
|
75
|
-
|
|
88
|
+
async children(kpi, context) {
|
|
89
|
+
const { domain } = context.state;
|
|
90
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
91
|
+
return await (0, shell_1.getRepository)(kpi_1.Kpi).find({
|
|
92
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, parent: { id: kpi.id } }
|
|
76
93
|
});
|
|
77
94
|
}
|
|
78
95
|
async thumbnail(kpi) {
|
|
@@ -96,8 +113,9 @@ let KpiQuery = class KpiQuery {
|
|
|
96
113
|
}
|
|
97
114
|
async value(kpi, context, orgScope) {
|
|
98
115
|
const { domain } = context.state;
|
|
116
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
99
117
|
let whereCondition = {
|
|
100
|
-
domain: { id:
|
|
118
|
+
domain: { id: (0, typeorm_1.In)(domainIds) },
|
|
101
119
|
kpi: { id: kpi.id }
|
|
102
120
|
};
|
|
103
121
|
// orgScope 필터가 있으면 KpiOrgScope로 필터링
|
|
@@ -123,9 +141,10 @@ let KpiQuery = class KpiQuery {
|
|
|
123
141
|
}
|
|
124
142
|
async histories(kpi, context, limit) {
|
|
125
143
|
const { domain } = context.state;
|
|
144
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
126
145
|
const qb = (0, shell_1.getRepository)(kpi_history_1.KpiHistory)
|
|
127
146
|
.createQueryBuilder('history')
|
|
128
|
-
.where('history.domain
|
|
147
|
+
.where('history.domain IN (:...domainIds)', { domainIds })
|
|
129
148
|
.andWhere('history.originalId = :kpiId', { kpiId: kpi.id })
|
|
130
149
|
.orderBy('history.updatedAt', 'DESC');
|
|
131
150
|
if (limit)
|
|
@@ -136,16 +155,18 @@ let KpiQuery = class KpiQuery {
|
|
|
136
155
|
}
|
|
137
156
|
async kpiTree(context) {
|
|
138
157
|
const { domain } = context.state;
|
|
158
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
139
159
|
return await (0, shell_1.getRepository)(kpi_1.Kpi).find({
|
|
140
|
-
where: { domain: { id:
|
|
160
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, parent: (0, typeorm_1.IsNull)() },
|
|
141
161
|
relations: ['children', 'children.children', 'children.children.children'],
|
|
142
162
|
order: { name: 'ASC' }
|
|
143
163
|
});
|
|
144
164
|
}
|
|
145
165
|
async kpiChildren(parentId, context) {
|
|
146
166
|
const { domain } = context.state;
|
|
167
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
147
168
|
return await (0, shell_1.getRepository)(kpi_1.Kpi).find({
|
|
148
|
-
where: { domain: { id:
|
|
169
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, parent: { id: parentId } },
|
|
149
170
|
order: { name: 'ASC' }
|
|
150
171
|
});
|
|
151
172
|
}
|
|
@@ -194,8 +215,9 @@ tslib_1.__decorate([
|
|
|
194
215
|
tslib_1.__decorate([
|
|
195
216
|
(0, type_graphql_1.FieldResolver)(type => [kpi_1.Kpi]),
|
|
196
217
|
tslib_1.__param(0, (0, type_graphql_1.Root)()),
|
|
218
|
+
tslib_1.__param(1, (0, type_graphql_1.Ctx)()),
|
|
197
219
|
tslib_1.__metadata("design:type", Function),
|
|
198
|
-
tslib_1.__metadata("design:paramtypes", [kpi_1.Kpi]),
|
|
220
|
+
tslib_1.__metadata("design:paramtypes", [kpi_1.Kpi, Object]),
|
|
199
221
|
tslib_1.__metadata("design:returntype", Promise)
|
|
200
222
|
], KpiQuery.prototype, "children", null);
|
|
201
223
|
tslib_1.__decorate([
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"kpi-query.js","sourceRoot":"","sources":["../../../server/service/kpi/kpi-query.ts"],"names":[],"mappings":"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","sourcesContent":["import { IsNull } from 'typeorm'\nimport { Resolver, Query, FieldResolver, Root, Args, Arg, Ctx, Directive } from 'type-graphql'\nimport { Attachment } from '@things-factory/attachment-base'\nimport { Domain, getQueryBuilderFromListParams, getRepository, ListParam } from '@things-factory/shell'\nimport { User } from '@things-factory/auth-base'\nimport type ResolverContext from '@things-factory/auth-base'\nimport { Kpi } from './kpi'\nimport { KpiList } from './kpi-type'\nimport { KpiValue } from '../kpi-value/kpi-value'\nimport { KpiHistory } from './kpi-history'\nimport { Int } from 'type-graphql'\n\n@Resolver(Kpi)\nexport class KpiQuery {\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], { description: 'Fetch root KPIs (KPIs without parent)' })\n async kpisLevel0(@Ctx() context: ResolverContext): Promise<Kpi[]> {\n const { domain } = context.state\n\n return await getRepository(Kpi).find({\n where: { domain: { id: domain.id }, parent: null },\n order: { name: 'ASC' }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], {\n description:\n 'Fetch first level child KPIs. If rootId is provided, get children of that parent. If not, get all level 1 KPIs (children of root KPIs)'\n })\n async kpisLevel1(\n @Arg('rootId', {\n nullable: true,\n description: 'ID of the parent KPI to get level 1 children. If not provided, returns all level 1 KPIs'\n })\n rootId: string,\n @Ctx() context: ResolverContext\n ): Promise<Kpi[]> {\n const { domain } = context.state\n\n let whereCondition: any = { domain: { id: domain.id } }\n\n if (rootId) {\n // 특정 부모의 자식들\n whereCondition.parent = { id: rootId }\n } else {\n // 모든 level 1 KPI들 (root KPI들의 직계 자식들)\n // 부모가 있으면서 부모의 부모가 null인 KPI들\n const kpis = await getRepository(Kpi)\n .createQueryBuilder('kpi')\n .leftJoinAndSelect('kpi.parent', 'parent')\n .leftJoinAndSelect('kpi.children', 'children')\n .leftJoinAndSelect('children.children', 'grandchildren')\n .where('kpi.domain = :domainId', { domainId: domain.id })\n .andWhere('parent.id IS NOT NULL')\n .andWhere('parent.parent IS NULL')\n .orderBy('kpi.name', 'ASC')\n .getMany()\n\n return kpis\n }\n\n // rootId 제공된 경우\n const kpis = await getRepository(Kpi).find({\n where: whereCondition,\n relations: ['children', 'children.children'],\n order: { name: 'ASC' }\n })\n\n return kpis\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => Kpi!, { nullable: true, description: 'Fetch a single KPI by its unique identifier.' })\n async kpi(\n @Arg('id', { description: 'Unique identifier of the KPI to fetch.' }) id: string,\n @Ctx() context: ResolverContext\n ): Promise<Kpi> {\n const { domain } = context.state\n\n return await getRepository(Kpi).findOne({\n where: { domain: { id: domain.id }, id }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiList, { description: 'To fetch multiple Kpis' })\n async kpis(@Args(type => ListParam) params: ListParam, @Ctx() context: ResolverContext): Promise<KpiList> {\n const { domain } = context.state\n\n const queryBuilder = getQueryBuilderFromListParams({\n domain,\n params,\n repository: await getRepository(Kpi),\n searchables: ['name', 'description'],\n filtersMap: {\n parent: { columnName: 'id', relationColumn: 'parent' }\n }\n })\n\n const [items, total] = await queryBuilder.getManyAndCount()\n\n return { items, total }\n }\n\n @FieldResolver(type => [Kpi])\n children(@Root() kpi: Kpi): Promise<Kpi[]> {\n return getRepository(Kpi).find({\n where: { parent: { id: kpi.id } }\n })\n }\n\n @FieldResolver(type => String)\n async thumbnail(@Root() kpi: Kpi): Promise<string | undefined> {\n const attachment: Attachment = await getRepository(Attachment).findOne({\n where: {\n domain: { id: kpi.domainId },\n refType: Kpi.name,\n refBy: kpi.id\n }\n })\n\n return attachment?.fullpath\n }\n\n @FieldResolver(type => Domain)\n async domain(@Root() kpi: Kpi): Promise<Domain> {\n return kpi.domainId && (await getRepository(Domain).findOneBy({ id: kpi.domainId }))\n }\n\n @FieldResolver(type => User)\n async updater(@Root() kpi: Kpi): Promise<User> {\n return kpi.updaterId && (await getRepository(User).findOneBy({ id: kpi.updaterId }))\n }\n\n @FieldResolver(type => User)\n async creator(@Root() kpi: Kpi): Promise<User> {\n return kpi.creatorId && (await getRepository(User).findOneBy({ id: kpi.creatorId }))\n }\n\n @FieldResolver(type => KpiValue, { nullable: true })\n async value(\n @Root() kpi: Kpi,\n @Ctx() context,\n @Arg('orgScope', { nullable: true, description: 'Organization scope filter for value' }) orgScope?: string\n ): Promise<KpiValue | null> {\n const { domain } = context.state\n\n let whereCondition: any = {\n domain: { id: domain.id },\n kpi: { id: kpi.id }\n }\n\n // orgScope 필터가 있으면 KpiOrgScope로 필터링\n if (orgScope) {\n whereCondition.kpiOrgScope = { org: orgScope }\n }\n\n return await getRepository(KpiValue).findOne({\n where: whereCondition,\n relations: ['kpiOrgScope'],\n order: { valueDate: 'DESC' }\n })\n }\n\n @FieldResolver(type => Number, { nullable: true })\n targetValue(@Root() kpi: Kpi): number | undefined {\n return kpi.vizMeta?.targetValue\n }\n\n @FieldResolver(type => String, { nullable: true })\n unit(@Root() kpi: Kpi): string | undefined {\n return kpi.vizMeta?.unit\n }\n\n @FieldResolver(type => Kpi, { nullable: true })\n async parent(@Root() kpi: Kpi): Promise<Kpi | null> {\n if (!kpi.parentId) return null\n return await getRepository(Kpi).findOneBy({ id: kpi.parentId })\n }\n\n @FieldResolver(type => [KpiHistory], { nullable: true })\n async histories(\n @Root() kpi: Kpi,\n @Ctx() context,\n @Arg('limit', type => Int, { nullable: true }) limit?: number\n ): Promise<KpiHistory[]> {\n const { domain } = context.state\n const qb = getRepository(KpiHistory)\n .createQueryBuilder('history')\n .where('history.domain = :domainId', { domainId: domain.id })\n .andWhere('history.originalId = :kpiId', { kpiId: kpi.id })\n .orderBy('history.updatedAt', 'DESC')\n if (limit) qb.limit(limit)\n else qb.limit(1)\n return await qb.getMany()\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], { description: 'Fetch KPIs in hierarchical tree structure' })\n async kpiTree(@Ctx() context: ResolverContext): Promise<Kpi[]> {\n const { domain } = context.state\n\n return await getRepository(Kpi).find({\n where: { domain: { id: domain.id }, parent: IsNull() },\n relations: ['children', 'children.children', 'children.children.children'],\n order: { name: 'ASC' }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], { description: 'Fetch child KPIs of a given parent KPI' })\n async kpiChildren(\n @Arg('parentId', { description: 'ID of the parent KPI' }) parentId: string,\n @Ctx() context: ResolverContext\n ): Promise<Kpi[]> {\n const { domain } = context.state\n\n return await getRepository(Kpi).find({\n where: { domain: { id: domain.id }, parent: { id: parentId } },\n order: { name: 'ASC' }\n })\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"kpi-query.js","sourceRoot":"","sources":["../../../server/service/kpi/kpi-query.ts"],"names":[],"mappings":"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","sourcesContent":["import { In, IsNull } from 'typeorm'\nimport { Resolver, Query, FieldResolver, Root, Args, Arg, Ctx, Directive } from 'type-graphql'\nimport { Attachment } from '@things-factory/attachment-base'\nimport { Domain, getQueryBuilderFromListParams, getRepository, ListParam } from '@things-factory/shell'\nimport { User } from '@things-factory/auth-base'\nimport type ResolverContext from '@things-factory/auth-base'\nimport { Kpi } from './kpi'\nimport { KpiList } from './kpi-type'\nimport { KpiValue } from '../kpi-value/kpi-value'\nimport { KpiHistory } from './kpi-history'\nimport { Int } from 'type-graphql'\nimport { dedupByClosestDomain, getDomainIdsWithAncestors } from '../utils/domain-inheritance'\n\n@Resolver(Kpi)\nexport class KpiQuery {\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], { description: 'Fetch root KPIs (KPIs without parent)' })\n async kpisLevel0(@Ctx() context: ResolverContext): Promise<Kpi[]> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(Kpi).find({\n where: { domain: { id: In(domainIds) }, parent: IsNull() },\n order: { name: 'ASC' }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], {\n description:\n 'Fetch first level child KPIs. If rootId is provided, get children of that parent. If not, get all level 1 KPIs (children of root KPIs)'\n })\n async kpisLevel1(\n @Arg('rootId', {\n nullable: true,\n description: 'ID of the parent KPI to get level 1 children. If not provided, returns all level 1 KPIs'\n })\n rootId: string,\n @Ctx() context: ResolverContext\n ): Promise<Kpi[]> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n let whereCondition: any = { domain: { id: In(domainIds) } }\n\n if (rootId) {\n // 특정 부모의 자식들\n whereCondition.parent = { id: rootId }\n } else {\n // 모든 level 1 KPI들 (root KPI들의 직계 자식들)\n // 부모가 있으면서 부모의 부모가 null인 KPI들\n const kpis = await getRepository(Kpi)\n .createQueryBuilder('kpi')\n .leftJoinAndSelect('kpi.parent', 'parent')\n .leftJoinAndSelect('kpi.children', 'children')\n .leftJoinAndSelect('children.children', 'grandchildren')\n .where('kpi.domain IN (:...domainIds)', { domainIds })\n .andWhere('parent.id IS NOT NULL')\n .andWhere('parent.parent IS NULL')\n .orderBy('kpi.name', 'ASC')\n .getMany()\n\n return kpis\n }\n\n // rootId 제공된 경우\n const kpis = await getRepository(Kpi).find({\n where: whereCondition,\n relations: ['children', 'children.children'],\n order: { name: 'ASC' }\n })\n\n return kpis\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => Kpi!, { nullable: true, description: 'Fetch a single KPI by its unique identifier.' })\n async kpi(\n @Arg('id', { description: 'Unique identifier of the KPI to fetch.' }) id: string,\n @Ctx() context: ResolverContext\n ): Promise<Kpi> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(Kpi).findOne({\n where: { domain: { id: In(domainIds) }, id }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiList, { description: 'To fetch multiple Kpis' })\n async kpis(@Args(type => ListParam) params: ListParam, @Ctx() context: ResolverContext): Promise<KpiList> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n // closest-wins dedup 을 위해 pagination 은 dedup 이후 직접 적용한다.\n const { pagination, ...rest } = params || ({} as ListParam)\n\n const queryBuilder = getQueryBuilderFromListParams({\n // domain 필터링은 ancestor 기반으로 직접 적용 (shell helper 는 single-parent 만 지원)\n domain: undefined,\n params: rest as ListParam,\n repository: getRepository(Kpi),\n searchables: ['name', 'description'],\n filtersMap: {\n parent: { columnName: 'id', relationColumn: 'parent' }\n }\n })\n\n queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {\n ancestorDomainIds: domainIds\n })\n\n const allItems = await queryBuilder.getMany()\n const deduped = dedupByClosestDomain(allItems, domainIds, k => k.name)\n\n const { page = 1, limit } = pagination || {}\n const total = deduped.length\n const items =\n limit && limit > 0 ? deduped.slice(Math.max(0, (page - 1) * limit), Math.max(0, (page - 1) * limit) + limit) : deduped\n\n return { items, total }\n }\n\n @FieldResolver(type => [Kpi])\n async children(@Root() kpi: Kpi, @Ctx() context: ResolverContext): Promise<Kpi[]> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(Kpi).find({\n where: { domain: { id: In(domainIds) }, parent: { id: kpi.id } }\n })\n }\n\n @FieldResolver(type => String)\n async thumbnail(@Root() kpi: Kpi): Promise<string | undefined> {\n const attachment: Attachment = await getRepository(Attachment).findOne({\n where: {\n domain: { id: kpi.domainId },\n refType: Kpi.name,\n refBy: kpi.id\n }\n })\n\n return attachment?.fullpath\n }\n\n @FieldResolver(type => Domain)\n async domain(@Root() kpi: Kpi): Promise<Domain> {\n return kpi.domainId && (await getRepository(Domain).findOneBy({ id: kpi.domainId }))\n }\n\n @FieldResolver(type => User)\n async updater(@Root() kpi: Kpi): Promise<User> {\n return kpi.updaterId && (await getRepository(User).findOneBy({ id: kpi.updaterId }))\n }\n\n @FieldResolver(type => User)\n async creator(@Root() kpi: Kpi): Promise<User> {\n return kpi.creatorId && (await getRepository(User).findOneBy({ id: kpi.creatorId }))\n }\n\n @FieldResolver(type => KpiValue, { nullable: true })\n async value(\n @Root() kpi: Kpi,\n @Ctx() context,\n @Arg('orgScope', { nullable: true, description: 'Organization scope filter for value' }) orgScope?: string\n ): Promise<KpiValue | null> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n let whereCondition: any = {\n domain: { id: In(domainIds) },\n kpi: { id: kpi.id }\n }\n\n // orgScope 필터가 있으면 KpiOrgScope로 필터링\n if (orgScope) {\n whereCondition.kpiOrgScope = { org: orgScope }\n }\n\n return await getRepository(KpiValue).findOne({\n where: whereCondition,\n relations: ['kpiOrgScope'],\n order: { valueDate: 'DESC' }\n })\n }\n\n @FieldResolver(type => Number, { nullable: true })\n targetValue(@Root() kpi: Kpi): number | undefined {\n return kpi.vizMeta?.targetValue\n }\n\n @FieldResolver(type => String, { nullable: true })\n unit(@Root() kpi: Kpi): string | undefined {\n return kpi.vizMeta?.unit\n }\n\n @FieldResolver(type => Kpi, { nullable: true })\n async parent(@Root() kpi: Kpi): Promise<Kpi | null> {\n if (!kpi.parentId) return null\n return await getRepository(Kpi).findOneBy({ id: kpi.parentId })\n }\n\n @FieldResolver(type => [KpiHistory], { nullable: true })\n async histories(\n @Root() kpi: Kpi,\n @Ctx() context,\n @Arg('limit', type => Int, { nullable: true }) limit?: number\n ): Promise<KpiHistory[]> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n const qb = getRepository(KpiHistory)\n .createQueryBuilder('history')\n .where('history.domain IN (:...domainIds)', { domainIds })\n .andWhere('history.originalId = :kpiId', { kpiId: kpi.id })\n .orderBy('history.updatedAt', 'DESC')\n if (limit) qb.limit(limit)\n else qb.limit(1)\n return await qb.getMany()\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], { description: 'Fetch KPIs in hierarchical tree structure' })\n async kpiTree(@Ctx() context: ResolverContext): Promise<Kpi[]> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(Kpi).find({\n where: { domain: { id: In(domainIds) }, parent: IsNull() },\n relations: ['children', 'children.children', 'children.children.children'],\n order: { name: 'ASC' }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => [Kpi], { description: 'Fetch child KPIs of a given parent KPI' })\n async kpiChildren(\n @Arg('parentId', { description: 'ID of the parent KPI' }) parentId: string,\n @Ctx() context: ResolverContext\n ): Promise<Kpi[]> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(Kpi).find({\n where: { domain: { id: In(domainIds) }, parent: { id: parentId } },\n order: { name: 'ASC' }\n })\n }\n}\n"]}
|
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.KpiMetricQuery = void 0;
|
|
4
4
|
const tslib_1 = require("tslib");
|
|
5
|
+
const typeorm_1 = require("typeorm");
|
|
5
6
|
const type_graphql_1 = require("type-graphql");
|
|
6
7
|
const attachment_base_1 = require("@things-factory/attachment-base");
|
|
7
8
|
const shell_1 = require("@things-factory/shell");
|
|
@@ -9,23 +10,35 @@ const auth_base_1 = require("@things-factory/auth-base");
|
|
|
9
10
|
const kpi_metric_1 = require("./kpi-metric");
|
|
10
11
|
const kpi_metric_type_1 = require("./kpi-metric-type");
|
|
11
12
|
const dataset_1 = require("@things-factory/dataset");
|
|
13
|
+
const domain_inheritance_1 = require("../utils/domain-inheritance");
|
|
12
14
|
let KpiMetricQuery = class KpiMetricQuery {
|
|
13
15
|
async kpiMetric(id, context) {
|
|
14
16
|
const { domain } = context.state;
|
|
17
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
15
18
|
return await (0, shell_1.getRepository)(kpi_metric_1.KpiMetric).findOne({
|
|
16
|
-
where: { domain: { id:
|
|
19
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, id },
|
|
17
20
|
relations: ['dataSet']
|
|
18
21
|
});
|
|
19
22
|
}
|
|
20
23
|
async kpiMetrics(params, context) {
|
|
21
24
|
const { domain } = context.state;
|
|
25
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
26
|
+
// closest-wins dedup 을 위해 pagination 은 dedup 이후 직접 적용한다.
|
|
27
|
+
const { pagination, ...rest } = params || {};
|
|
22
28
|
const queryBuilder = (0, shell_1.getQueryBuilderFromListParams)({
|
|
23
|
-
domain,
|
|
24
|
-
params,
|
|
25
|
-
repository:
|
|
29
|
+
domain: undefined,
|
|
30
|
+
params: rest,
|
|
31
|
+
repository: (0, shell_1.getRepository)(kpi_metric_1.KpiMetric),
|
|
26
32
|
searchables: ['name', 'description']
|
|
27
33
|
});
|
|
28
|
-
|
|
34
|
+
queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {
|
|
35
|
+
ancestorDomainIds: domainIds
|
|
36
|
+
});
|
|
37
|
+
const allItems = await queryBuilder.getMany();
|
|
38
|
+
const deduped = (0, domain_inheritance_1.dedupByClosestDomain)(allItems, domainIds, m => m.name);
|
|
39
|
+
const { page = 1, limit } = pagination || {};
|
|
40
|
+
const total = deduped.length;
|
|
41
|
+
const items = limit && limit > 0 ? deduped.slice(Math.max(0, (page - 1) * limit), Math.max(0, (page - 1) * limit) + limit) : deduped;
|
|
29
42
|
return { items, total };
|
|
30
43
|
}
|
|
31
44
|
async thumbnail(kpiMetric) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"kpi-metric-query.js","sourceRoot":"","sources":["../../../server/service/kpi-metric/kpi-metric-query.ts"],"names":[],"mappings":";;;;AAAA,+CAA8F;AAC9F,qEAA4D;AAC5D,iDAAuG;AACvG,yDAAgD;AAChD,6CAAwC;AACxC,uDAAiD;AACjD,qDAAiD;
|
|
1
|
+
{"version":3,"file":"kpi-metric-query.js","sourceRoot":"","sources":["../../../server/service/kpi-metric/kpi-metric-query.ts"],"names":[],"mappings":"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","sourcesContent":["import { In } from 'typeorm'\nimport { Resolver, Query, FieldResolver, Root, Args, Arg, Ctx, Directive } from 'type-graphql'\nimport { Attachment } from '@things-factory/attachment-base'\nimport { Domain, getQueryBuilderFromListParams, getRepository, ListParam } from '@things-factory/shell'\nimport { User } from '@things-factory/auth-base'\nimport { KpiMetric } from './kpi-metric'\nimport { KpiMetricList } from './kpi-metric-type'\nimport { DataSet } from '@things-factory/dataset'\nimport { dedupByClosestDomain, getDomainIdsWithAncestors } from '../utils/domain-inheritance'\n\n@Resolver(KpiMetric)\nexport class KpiMetricQuery {\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiMetric!, { nullable: true, description: 'Fetch a single KPI metric by its unique identifier.' })\n async kpiMetric(\n @Arg('id', { description: 'Unique identifier of the KPI metric to fetch.' }) id: string,\n @Ctx() context: ResolverContext\n ): Promise<KpiMetric> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(KpiMetric).findOne({\n where: { domain: { id: In(domainIds) }, id },\n relations: ['dataSet']\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiMetricList, { description: 'To fetch multiple KpiMetrics' })\n async kpiMetrics(\n @Args(type => ListParam) params: ListParam,\n @Ctx() context: ResolverContext\n ): Promise<KpiMetricList> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n // closest-wins dedup 을 위해 pagination 은 dedup 이후 직접 적용한다.\n const { pagination, ...rest } = params || ({} as ListParam)\n\n const queryBuilder = getQueryBuilderFromListParams({\n domain: undefined,\n params: rest as ListParam,\n repository: getRepository(KpiMetric),\n searchables: ['name', 'description']\n })\n\n queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {\n ancestorDomainIds: domainIds\n })\n\n const allItems = await queryBuilder.getMany()\n const deduped = dedupByClosestDomain(allItems, domainIds, m => m.name)\n\n const { page = 1, limit } = pagination || {}\n const total = deduped.length\n const items =\n limit && limit > 0 ? deduped.slice(Math.max(0, (page - 1) * limit), Math.max(0, (page - 1) * limit) + limit) : deduped\n\n return { items, total }\n }\n\n @FieldResolver(type => String)\n async thumbnail(@Root() kpiMetric: KpiMetric): Promise<string | undefined> {\n const attachment: Attachment = await getRepository(Attachment).findOne({\n where: {\n domain: { id: kpiMetric.domainId },\n refType: KpiMetric.name,\n refBy: kpiMetric.id\n }\n })\n\n return attachment?.fullpath\n }\n\n @FieldResolver(type => String)\n async dataSet(@Root() kpiMetric: KpiMetric): Promise<DataSet> {\n return kpiMetric.dataSetId && (await getRepository(DataSet).findOneBy({ id: kpiMetric.dataSetId }))\n }\n\n @FieldResolver(type => Domain)\n async domain(@Root() kpiMetric: KpiMetric): Promise<Domain> {\n return kpiMetric.domainId && (await getRepository(Domain).findOneBy({ id: kpiMetric.domainId }))\n }\n\n @FieldResolver(type => User)\n async updater(@Root() kpiMetric: KpiMetric): Promise<User> {\n return kpiMetric.updaterId && (await getRepository(User).findOneBy({ id: kpiMetric.updaterId }))\n }\n\n @FieldResolver(type => User)\n async creator(@Root() kpiMetric: KpiMetric): Promise<User> {\n return kpiMetric.creatorId && (await getRepository(User).findOneBy({ id: kpiMetric.creatorId }))\n }\n}\n"]}
|
|
@@ -2,28 +2,35 @@
|
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.KpiMetricValueQuery = void 0;
|
|
4
4
|
const tslib_1 = require("tslib");
|
|
5
|
+
const typeorm_1 = require("typeorm");
|
|
5
6
|
const type_graphql_1 = require("type-graphql");
|
|
6
7
|
const shell_1 = require("@things-factory/shell");
|
|
7
8
|
const auth_base_1 = require("@things-factory/auth-base");
|
|
8
9
|
const kpi_metric_value_1 = require("./kpi-metric-value");
|
|
9
10
|
const kpi_metric_value_type_1 = require("./kpi-metric-value-type");
|
|
10
11
|
const kpi_metric_1 = require("../kpi-metric/kpi-metric");
|
|
12
|
+
const domain_inheritance_1 = require("../utils/domain-inheritance");
|
|
11
13
|
let KpiMetricValueQuery = class KpiMetricValueQuery {
|
|
12
14
|
async kpiMetricValues(params, context) {
|
|
13
15
|
const { domain } = context.state;
|
|
16
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
14
17
|
const queryBuilder = (0, shell_1.getQueryBuilderFromListParams)({
|
|
15
|
-
domain,
|
|
18
|
+
domain: undefined,
|
|
16
19
|
params,
|
|
17
|
-
repository:
|
|
20
|
+
repository: (0, shell_1.getRepository)(kpi_metric_value_1.KpiMetricValue),
|
|
18
21
|
searchables: ['metricId', 'group', 'valueDate']
|
|
19
22
|
});
|
|
23
|
+
queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {
|
|
24
|
+
ancestorDomainIds: domainIds
|
|
25
|
+
});
|
|
20
26
|
const [items, total] = await queryBuilder.getManyAndCount();
|
|
21
27
|
return { items, total };
|
|
22
28
|
}
|
|
23
29
|
async kpiMetricValue(id, context) {
|
|
24
30
|
const { domain } = context.state;
|
|
31
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
25
32
|
return await (0, shell_1.getRepository)(kpi_metric_value_1.KpiMetricValue).findOne({
|
|
26
|
-
where: { domain: { id:
|
|
33
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, id }
|
|
27
34
|
});
|
|
28
35
|
}
|
|
29
36
|
async domain(metricValue) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"kpi-metric-value-query.js","sourceRoot":"","sources":["../../../server/service/kpi-metric-value/kpi-metric-value-query.ts"],"names":[],"mappings":";;;;AAAA,+CAA8F;AAC9F,iDAAuG;AACvG,yDAAgD;AAChD,yDAAmD;AACnD,mEAA4D;AAC5D,yDAAoD;
|
|
1
|
+
{"version":3,"file":"kpi-metric-value-query.js","sourceRoot":"","sources":["../../../server/service/kpi-metric-value/kpi-metric-value-query.ts"],"names":[],"mappings":"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","sourcesContent":["import { In } from 'typeorm'\nimport { Directive, Resolver, Query, Args, Arg, Ctx, FieldResolver, Root } from 'type-graphql'\nimport { Domain, getQueryBuilderFromListParams, getRepository, ListParam } from '@things-factory/shell'\nimport { User } from '@things-factory/auth-base'\nimport { KpiMetricValue } from './kpi-metric-value'\nimport { KpiMetricValueList } from './kpi-metric-value-type'\nimport { KpiMetric } from '../kpi-metric/kpi-metric'\nimport { getDomainIdsWithAncestors } from '../utils/domain-inheritance'\n\n@Resolver(KpiMetricValue)\nexport class KpiMetricValueQuery {\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiMetricValueList, { description: 'To fetch multiple KpiMetricValues' })\n async kpiMetricValues(\n @Args(type => ListParam) params: ListParam,\n @Ctx() context: ResolverContext\n ): Promise<KpiMetricValueList> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n const queryBuilder = getQueryBuilderFromListParams({\n domain: undefined,\n params,\n repository: getRepository(KpiMetricValue),\n searchables: ['metricId', 'group', 'valueDate']\n })\n\n queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {\n ancestorDomainIds: domainIds\n })\n\n const [items, total] = await queryBuilder.getManyAndCount()\n return { items, total }\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiMetricValue, {\n nullable: true,\n description: 'Fetch a single metric value by its unique identifier.'\n })\n async kpiMetricValue(\n @Arg('id', { description: 'Unique identifier of the metric value to fetch.' }) id: string,\n @Ctx() context: ResolverContext\n ): Promise<KpiMetricValue | null> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n return await getRepository(KpiMetricValue).findOne({\n where: { domain: { id: In(domainIds) }, id }\n })\n }\n\n @FieldResolver(type => Domain)\n async domain(@Root() metricValue: KpiMetricValue): Promise<Domain> {\n return metricValue.domainId && (await getRepository(Domain).findOneBy({ id: metricValue.domainId }))\n }\n\n @FieldResolver(type => User)\n async updater(@Root() metricValue: KpiMetricValue): Promise<User> {\n return metricValue.updaterId && (await getRepository(User).findOneBy({ id: metricValue.updaterId }))\n }\n\n @FieldResolver(type => User)\n async creator(@Root() metricValue: KpiMetricValue): Promise<User> {\n return metricValue.creatorId && (await getRepository(User).findOneBy({ id: metricValue.creatorId }))\n }\n\n @FieldResolver(type => KpiMetric)\n async metric(@Root() metricValue: KpiMetricValue): Promise<KpiMetric | null> {\n if (!metricValue.metricId) return null\n return await getRepository(KpiMetric).findOneBy({ id: metricValue.metricId })\n }\n}\n"]}
|
|
@@ -2,27 +2,34 @@
|
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.KpiValueQuery = void 0;
|
|
4
4
|
const tslib_1 = require("tslib");
|
|
5
|
+
const typeorm_1 = require("typeorm");
|
|
5
6
|
const type_graphql_1 = require("type-graphql");
|
|
6
7
|
const shell_1 = require("@things-factory/shell");
|
|
7
8
|
const auth_base_1 = require("@things-factory/auth-base");
|
|
8
9
|
const kpi_value_1 = require("./kpi-value");
|
|
9
10
|
const kpi_value_type_1 = require("./kpi-value-type");
|
|
10
11
|
const kpi_1 = require("../kpi/kpi");
|
|
12
|
+
const domain_inheritance_1 = require("../utils/domain-inheritance");
|
|
11
13
|
let KpiValueQuery = class KpiValueQuery {
|
|
12
14
|
async kpiValue(id, context) {
|
|
13
15
|
const { domain } = context.state;
|
|
16
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
14
17
|
return await (0, shell_1.getRepository)(kpi_value_1.KpiValue).findOne({
|
|
15
|
-
where: { domain: { id:
|
|
18
|
+
where: { domain: { id: (0, typeorm_1.In)(domainIds) }, id }
|
|
16
19
|
});
|
|
17
20
|
}
|
|
18
21
|
async kpiValues(params, context) {
|
|
19
22
|
const { domain } = context.state;
|
|
23
|
+
const domainIds = await (0, domain_inheritance_1.getDomainIdsWithAncestors)(domain);
|
|
20
24
|
const queryBuilder = (0, shell_1.getQueryBuilderFromListParams)({
|
|
21
|
-
domain,
|
|
25
|
+
domain: undefined,
|
|
22
26
|
params,
|
|
23
|
-
repository:
|
|
27
|
+
repository: (0, shell_1.getRepository)(kpi_value_1.KpiValue),
|
|
24
28
|
searchables: ['kpi', 'group', 'valueDate']
|
|
25
29
|
});
|
|
30
|
+
queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {
|
|
31
|
+
ancestorDomainIds: domainIds
|
|
32
|
+
});
|
|
26
33
|
const [items, total] = await queryBuilder.getManyAndCount();
|
|
27
34
|
return { items, total };
|
|
28
35
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"kpi-value-query.js","sourceRoot":"","sources":["../../../server/service/kpi-value/kpi-value-query.ts"],"names":[],"mappings":";;;;AAAA,+CAAqG;AACrG,iDAAuG;AACvG,yDAAgD;AAChD,2CAAsC;AACtC,qDAA+C;AAC/C,oCAAgC;
|
|
1
|
+
{"version":3,"file":"kpi-value-query.js","sourceRoot":"","sources":["../../../server/service/kpi-value/kpi-value-query.ts"],"names":[],"mappings":"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","sourcesContent":["import { In } from 'typeorm'\nimport { Resolver, Query, FieldResolver, Root, Args, Arg, Ctx, Directive, Float } from 'type-graphql'\nimport { Domain, getQueryBuilderFromListParams, getRepository, ListParam } from '@things-factory/shell'\nimport { User } from '@things-factory/auth-base'\nimport { KpiValue } from './kpi-value'\nimport { KpiValueList } from './kpi-value-type'\nimport { Kpi } from '../kpi/kpi'\nimport { KpiValueScoreService } from './kpi-value-score.service'\nimport { getDomainIdsWithAncestors } from '../utils/domain-inheritance'\n\n@Resolver(KpiValue)\nexport class KpiValueQuery {\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiValue!, { nullable: true, description: 'Fetch a single KPI value by its unique identifier.' })\n async kpiValue(\n @Arg('id', { description: 'Unique identifier of the KPI value to fetch.' }) id: string,\n @Ctx() context: ResolverContext\n ): Promise<KpiValue> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n return await getRepository(KpiValue).findOne({\n where: { domain: { id: In(domainIds) }, id }\n })\n }\n\n @Directive('@privilege(category: \"kpi\", privilege: \"query\", domainOwnerGranted: true, superUserGranted: true)')\n @Query(returns => KpiValueList, { description: 'To fetch multiple KpiValues' })\n async kpiValues(@Args(type => ListParam) params: ListParam, @Ctx() context: ResolverContext): Promise<KpiValueList> {\n const { domain } = context.state\n const domainIds = await getDomainIdsWithAncestors(domain)\n\n const queryBuilder = getQueryBuilderFromListParams({\n domain: undefined,\n params,\n repository: getRepository(KpiValue),\n searchables: ['kpi', 'group', 'valueDate']\n })\n\n queryBuilder.andWhere(`${queryBuilder.alias}.domain IN (:...ancestorDomainIds)`, {\n ancestorDomainIds: domainIds\n })\n\n const [items, total] = await queryBuilder.getManyAndCount()\n\n return { items, total }\n }\n\n @FieldResolver(type => Domain)\n async domain(@Root() kpiValue: KpiValue): Promise<Domain> {\n return kpiValue.domainId && (await getRepository(Domain).findOneBy({ id: kpiValue.domainId }))\n }\n\n @FieldResolver(type => User)\n async updater(@Root() kpiValue: KpiValue): Promise<User> {\n return kpiValue.updaterId && (await getRepository(User).findOneBy({ id: kpiValue.updaterId }))\n }\n\n @FieldResolver(type => User)\n async creator(@Root() kpiValue: KpiValue): Promise<User> {\n return kpiValue.creatorId && (await getRepository(User).findOneBy({ id: kpiValue.creatorId }))\n }\n\n @FieldResolver(type => Kpi)\n async kpi(@Root() kpiValue: KpiValue): Promise<Kpi | null> {\n if (!kpiValue.kpiId) return null\n return await getRepository(Kpi).findOneBy({ id: kpiValue.kpiId })\n }\n}\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { Domain } from '@things-factory/shell';
|
|
2
|
+
export declare function getDomainIdsWithAncestors(domain: Domain): Promise<string[]>;
|
|
3
|
+
/**
|
|
4
|
+
* "가까운 도메인 우선(override)" 의미 구현.
|
|
5
|
+
* 여러 ancestor 도메인의 항목들 중 같은 키(name 등)가 중복 존재하면
|
|
6
|
+
* domainIds 배열에서 더 앞(=가까운 도메인)에 있는 도메인에 속한 항목 1개만 남긴다.
|
|
7
|
+
*
|
|
8
|
+
* - domainIds: closest-first 순서 (getDomainIdsWithAncestors 결과)
|
|
9
|
+
* - keyFn: 중복 판정 키 (예: kpi.name)
|
|
10
|
+
*
|
|
11
|
+
* 정렬은 호출자가 다시 적용해야 한다 (typeorm 의 사전 정렬은 dedup 으로 깨지지 않음 -
|
|
12
|
+
* 같은 키 그룹 내부의 선택만 바뀌고 그룹의 위치는 입력 순서를 유지하므로
|
|
13
|
+
* 안정적이다).
|
|
14
|
+
*/
|
|
15
|
+
export declare function dedupByClosestDomain<T extends {
|
|
16
|
+
domainId?: string;
|
|
17
|
+
}>(items: T[], domainIds: string[], keyFn: (item: T) => string | undefined): T[];
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getDomainIdsWithAncestors = getDomainIdsWithAncestors;
|
|
4
|
+
exports.dedupByClosestDomain = dedupByClosestDomain;
|
|
5
|
+
const shell_1 = require("@things-factory/shell");
|
|
6
|
+
/**
|
|
7
|
+
* 멀티테넌트 도메인 다단계 상속 헬퍼.
|
|
8
|
+
*
|
|
9
|
+
* SYSTEM → PROJECT-TEMPLATE → project 처럼 계층이 깊어질 수 있으므로
|
|
10
|
+
* 현재 도메인 부터 루트(부모가 없는 도메인)까지의 ancestor id 를
|
|
11
|
+
* "가까운 순서대로" 반환한다. (closest-first)
|
|
12
|
+
*
|
|
13
|
+
* - 사용처: 단일 fetch 의 `In([...])`, list 조회의 `andWhere('domain IN (:...ids)')`,
|
|
14
|
+
* 그리고 closest-wins dedup 의 우선순위 키.
|
|
15
|
+
*
|
|
16
|
+
* 깊이는 보통 2~3 이지만 순환 방지 + 최대 깊이(10) 안전장치를 둔다.
|
|
17
|
+
*/
|
|
18
|
+
const MAX_ANCESTOR_DEPTH = 10;
|
|
19
|
+
async function getDomainIdsWithAncestors(domain) {
|
|
20
|
+
const ids = [domain.id];
|
|
21
|
+
const seen = new Set([domain.id]);
|
|
22
|
+
let parentId = domain.parentId;
|
|
23
|
+
let depth = 0;
|
|
24
|
+
while (parentId && !seen.has(parentId) && depth < MAX_ANCESTOR_DEPTH) {
|
|
25
|
+
seen.add(parentId);
|
|
26
|
+
ids.push(parentId);
|
|
27
|
+
const parent = await (0, shell_1.getRepository)(shell_1.Domain).findOne({ where: { id: parentId } });
|
|
28
|
+
parentId = parent?.parentId;
|
|
29
|
+
depth++;
|
|
30
|
+
}
|
|
31
|
+
return ids;
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* "가까운 도메인 우선(override)" 의미 구현.
|
|
35
|
+
* 여러 ancestor 도메인의 항목들 중 같은 키(name 등)가 중복 존재하면
|
|
36
|
+
* domainIds 배열에서 더 앞(=가까운 도메인)에 있는 도메인에 속한 항목 1개만 남긴다.
|
|
37
|
+
*
|
|
38
|
+
* - domainIds: closest-first 순서 (getDomainIdsWithAncestors 결과)
|
|
39
|
+
* - keyFn: 중복 판정 키 (예: kpi.name)
|
|
40
|
+
*
|
|
41
|
+
* 정렬은 호출자가 다시 적용해야 한다 (typeorm 의 사전 정렬은 dedup 으로 깨지지 않음 -
|
|
42
|
+
* 같은 키 그룹 내부의 선택만 바뀌고 그룹의 위치는 입력 순서를 유지하므로
|
|
43
|
+
* 안정적이다).
|
|
44
|
+
*/
|
|
45
|
+
function dedupByClosestDomain(items, domainIds, keyFn) {
|
|
46
|
+
const rankOf = (domainId) => {
|
|
47
|
+
if (!domainId)
|
|
48
|
+
return Number.MAX_SAFE_INTEGER;
|
|
49
|
+
const idx = domainIds.indexOf(domainId);
|
|
50
|
+
return idx < 0 ? Number.MAX_SAFE_INTEGER : idx;
|
|
51
|
+
};
|
|
52
|
+
const byKey = new Map();
|
|
53
|
+
const order = [];
|
|
54
|
+
for (const item of items) {
|
|
55
|
+
const key = keyFn(item);
|
|
56
|
+
if (key == null)
|
|
57
|
+
continue;
|
|
58
|
+
const existing = byKey.get(key);
|
|
59
|
+
if (!existing) {
|
|
60
|
+
byKey.set(key, item);
|
|
61
|
+
order.push(key);
|
|
62
|
+
}
|
|
63
|
+
else if (rankOf(item.domainId) < rankOf(existing.domainId)) {
|
|
64
|
+
byKey.set(key, item);
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
return order.map(k => byKey.get(k));
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=domain-inheritance.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"domain-inheritance.js","sourceRoot":"","sources":["../../../server/service/utils/domain-inheritance.ts"],"names":[],"mappings":";;AAgBA,8DAiBC;AAcD,oDA4BC;AA3ED,iDAA6D;AAE7D;;;;;;;;;;;GAWG;AACH,MAAM,kBAAkB,GAAG,EAAE,CAAA;AAEtB,KAAK,UAAU,yBAAyB,CAAC,MAAc;IAC5D,MAAM,GAAG,GAAa,CAAC,MAAM,CAAC,EAAE,CAAC,CAAA;IACjC,MAAM,IAAI,GAAG,IAAI,GAAG,CAAS,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAA;IAEzC,IAAI,QAAQ,GAAuB,MAAM,CAAC,QAAQ,CAAA;IAClD,IAAI,KAAK,GAAG,CAAC,CAAA;IAEb,OAAO,QAAQ,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,KAAK,GAAG,kBAAkB,EAAE,CAAC;QACrE,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAA;QAClB,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;QAElB,MAAM,MAAM,GAAG,MAAM,IAAA,qBAAa,EAAC,cAAM,CAAC,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAA;QAC/E,QAAQ,GAAG,MAAM,EAAE,QAAQ,CAAA;QAC3B,KAAK,EAAE,CAAA;IACT,CAAC;IAED,OAAO,GAAG,CAAA;AACZ,CAAC;AAED;;;;;;;;;;;GAWG;AACH,SAAgB,oBAAoB,CAClC,KAAU,EACV,SAAmB,EACnB,KAAsC;IAEtC,MAAM,MAAM,GAAG,CAAC,QAA4B,EAAU,EAAE;QACtD,IAAI,CAAC,QAAQ;YAAE,OAAO,MAAM,CAAC,gBAAgB,CAAA;QAC7C,MAAM,GAAG,GAAG,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAA;QACvC,OAAO,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,GAAG,CAAA;IAChD,CAAC,CAAA;IAED,MAAM,KAAK,GAAG,IAAI,GAAG,EAAa,CAAA;IAClC,MAAM,KAAK,GAAa,EAAE,CAAA;IAE1B,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,MAAM,GAAG,GAAG,KAAK,CAAC,IAAI,CAAC,CAAA;QACvB,IAAI,GAAG,IAAI,IAAI;YAAE,SAAQ;QAEzB,MAAM,QAAQ,GAAG,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;QAC/B,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAA;YACpB,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;QACjB,CAAC;aAAM,IAAI,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,MAAM,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7D,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAA;QACtB,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAE,CAAC,CAAA;AACtC,CAAC","sourcesContent":["import { Domain, getRepository } from '@things-factory/shell'\n\n/**\n * 멀티테넌트 도메인 다단계 상속 헬퍼.\n *\n * SYSTEM → PROJECT-TEMPLATE → project 처럼 계층이 깊어질 수 있으므로\n * 현재 도메인 부터 루트(부모가 없는 도메인)까지의 ancestor id 를\n * \"가까운 순서대로\" 반환한다. (closest-first)\n *\n * - 사용처: 단일 fetch 의 `In([...])`, list 조회의 `andWhere('domain IN (:...ids)')`,\n * 그리고 closest-wins dedup 의 우선순위 키.\n *\n * 깊이는 보통 2~3 이지만 순환 방지 + 최대 깊이(10) 안전장치를 둔다.\n */\nconst MAX_ANCESTOR_DEPTH = 10\n\nexport async function getDomainIdsWithAncestors(domain: Domain): Promise<string[]> {\n const ids: string[] = [domain.id]\n const seen = new Set<string>([domain.id])\n\n let parentId: string | undefined = domain.parentId\n let depth = 0\n\n while (parentId && !seen.has(parentId) && depth < MAX_ANCESTOR_DEPTH) {\n seen.add(parentId)\n ids.push(parentId)\n\n const parent = await getRepository(Domain).findOne({ where: { id: parentId } })\n parentId = parent?.parentId\n depth++\n }\n\n return ids\n}\n\n/**\n * \"가까운 도메인 우선(override)\" 의미 구현.\n * 여러 ancestor 도메인의 항목들 중 같은 키(name 등)가 중복 존재하면\n * domainIds 배열에서 더 앞(=가까운 도메인)에 있는 도메인에 속한 항목 1개만 남긴다.\n *\n * - domainIds: closest-first 순서 (getDomainIdsWithAncestors 결과)\n * - keyFn: 중복 판정 키 (예: kpi.name)\n *\n * 정렬은 호출자가 다시 적용해야 한다 (typeorm 의 사전 정렬은 dedup 으로 깨지지 않음 -\n * 같은 키 그룹 내부의 선택만 바뀌고 그룹의 위치는 입력 순서를 유지하므로\n * 안정적이다).\n */\nexport function dedupByClosestDomain<T extends { domainId?: string }>(\n items: T[],\n domainIds: string[],\n keyFn: (item: T) => string | undefined\n): T[] {\n const rankOf = (domainId: string | undefined): number => {\n if (!domainId) return Number.MAX_SAFE_INTEGER\n const idx = domainIds.indexOf(domainId)\n return idx < 0 ? Number.MAX_SAFE_INTEGER : idx\n }\n\n const byKey = new Map<string, T>()\n const order: string[] = []\n\n for (const item of items) {\n const key = keyFn(item)\n if (key == null) continue\n\n const existing = byKey.get(key)\n if (!existing) {\n byKey.set(key, item)\n order.push(key)\n } else if (rankOf(item.domainId) < rankOf(existing.domainId)) {\n byKey.set(key, item)\n }\n }\n\n return order.map(k => byKey.get(k)!)\n}\n"]}
|