@acorex/connectivity 21.0.0-next.3 → 21.0.0-next.7
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/fesm2022/acorex-connectivity-mock-get-application-versions-chart-data.query-C3eQfF0A.mjs +115 -0
- package/fesm2022/acorex-connectivity-mock-get-application-versions-chart-data.query-C3eQfF0A.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-application-versions-timeline.query-CLzDz9ob.mjs +163 -0
- package/fesm2022/acorex-connectivity-mock-get-application-versions-timeline.query-CLzDz9ob.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-product-chart-data.query-DDVVFeYN.mjs +97 -0
- package/fesm2022/acorex-connectivity-mock-get-product-chart-data.query-DDVVFeYN.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-product-list.query-ChtYlSTt.mjs +157 -0
- package/fesm2022/acorex-connectivity-mock-get-product-list.query-ChtYlSTt.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-subscriptions-chart-data.query-HxX_bCT8.mjs +56 -0
- package/fesm2022/acorex-connectivity-mock-get-subscriptions-chart-data.query-HxX_bCT8.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-subscriptions-timeline.query-dp08JqLP.mjs +168 -0
- package/fesm2022/acorex-connectivity-mock-get-subscriptions-timeline.query-dp08JqLP.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-application-access.query-IATTXcAV.mjs +180 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-application-access.query-IATTXcAV.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-registration-chart-data.query-DlXi4Rcg.mjs +56 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-registration-chart-data.query-DlXi4Rcg.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-registration-timeline.query-hskw8Ioa.mjs +216 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-registration-timeline.query-hskw8Ioa.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-user-distribution.query-Bs5U5tze.mjs +134 -0
- package/fesm2022/acorex-connectivity-mock-get-tenant-user-distribution.query-Bs5U5tze.mjs.map +1 -0
- package/fesm2022/acorex-connectivity-mock.mjs +10595 -11891
- package/fesm2022/acorex-connectivity-mock.mjs.map +1 -1
- package/mock/index.d.ts +80 -279
- package/package.json +2 -2
package/fesm2022/acorex-connectivity-mock-get-tenant-registration-timeline.query-hskw8Ioa.mjs.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"acorex-connectivity-mock-get-tenant-registration-timeline.query-hskw8Ioa.mjs","sources":["../tmp-esm2022/mock/lib/tenant-management/reports/get-tenant-registration-timeline.query.js"],"sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { AXPEntityService } from '@acorex/platform/layout/entity';\nimport { RootConfig as TenantRootConfig } from '@acorex/modules/tenant-management';\nimport { RootConfig as SubscriptionRootConfig } from '@acorex/modules/subscription-management';\nimport { RootConfig as ApplicationRootConfig } from '@acorex/modules/application-management';\nimport * as i0 from \"@angular/core\";\n//#endregion\nexport class GetTenantRegistrationTimelineQuery {\n constructor() {\n this.entityService = inject(AXPEntityService);\n this.tenantService = this.entityService\n .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenant.name}`)\n .data();\n this.tenantUserService = this.entityService\n .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenantUser.name}`)\n .data();\n this.subscriptionService = this.entityService\n .withEntity(`${SubscriptionRootConfig.module.name}.${SubscriptionRootConfig.entities.subscription.name}`)\n .data();\n this.planItemService = this.entityService\n .withEntity(`${SubscriptionRootConfig.module.name}.${SubscriptionRootConfig.entities.subscriptionPlanItem.name}`)\n .data();\n this.applicationService = this.entityService\n .withEntity(`${ApplicationRootConfig.module.name}.${ApplicationRootConfig.entities.application.name}`)\n .data();\n }\n async fetch(input) {\n // Get all tenants\n const tenantsResult = await this.tenantService.query({\n skip: 0,\n take: 10000,\n filter: input.filters && input.filters.length > 0\n ? {\n logic: input.logic || 'and',\n filters: input.filters.filter((f) => f.field !== 'dateRange'),\n }\n : undefined,\n });\n let tenants = tenantsResult.items.filter((t) => t.registrationDate);\n // Apply date range filter if provided\n if (input.filters) {\n for (const filter of input.filters) {\n if (filter.field === 'dateRange' && filter.value) {\n tenants = this.applyDateRangeFilter(tenants, filter.value);\n }\n }\n }\n // Get all tenant users for counting\n const tenantUsersResult = await this.tenantUserService.query({\n skip: 0,\n take: 10000,\n });\n const tenantUsers = tenantUsersResult.items;\n // Get all subscriptions for counting\n const subscriptionsResult = await this.subscriptionService.query({\n skip: 0,\n take: 10000,\n });\n const subscriptions = subscriptionsResult.items.filter((s) => s.subscriberType === 'Tenant');\n // Get all plan items and applications for application counting\n const planItemsResult = await this.planItemService.query({\n skip: 0,\n take: 10000,\n });\n const planItems = planItemsResult.items;\n const applicationsResult = await this.applicationService.query({\n skip: 0,\n take: 10000,\n });\n const applications = applicationsResult.items;\n // Get application counts for tenants\n const applicationCounts = await this.getApplicationCountsForTenants(tenants.map((t) => t.id), subscriptions, planItems, applications);\n // Build results\n const results = [];\n for (const tenant of tenants) {\n if (!tenant.registrationDate)\n continue;\n const registrationDate = new Date(tenant.registrationDate);\n // Extract time period info\n const year = registrationDate.getFullYear();\n const month = registrationDate.getMonth() + 1;\n const monthName = registrationDate.toLocaleString('en-US', { month: 'long' });\n const quarter = Math.ceil(month / 3);\n const quarterName = `Q${quarter}`;\n // Count users for this tenant\n const userCount = tenantUsers.filter((tu) => tu.tenantId === tenant.id).length;\n // Count subscriptions for this tenant\n const subscriptionCount = subscriptions.filter((s) => s.subscriberId === tenant.id).length;\n // Get application count\n const applicationCount = applicationCounts.get(tenant.id) || 0;\n results.push({\n tenantId: tenant.id,\n tenantName: tenant.title,\n registrationDate: registrationDate,\n registrationDateStr: registrationDate.toISOString(),\n status: tenant.statusId || 'Unknown',\n country: tenant.country?.title,\n userCount: userCount,\n applicationCount: applicationCount,\n subscriptionCount: subscriptionCount,\n month: monthName,\n monthNumber: month,\n quarter: quarterName,\n year: year,\n });\n }\n // Apply sorting if provided\n if (input.sort && input.sort.length > 0) {\n results.sort((a, b) => {\n for (const sortField of input.sort) {\n const field = sortField.field;\n const aValue = a[field];\n const bValue = b[field];\n const dir = sortField.dir === 'asc' ? 1 : -1;\n if (aValue === undefined && bValue === undefined)\n continue;\n if (aValue === undefined)\n return 1 * dir;\n if (bValue === undefined)\n return -1 * dir;\n // Handle date comparison\n if (field === 'registrationDate') {\n const aDate = a.registrationDate.getTime();\n const bDate = b.registrationDate.getTime();\n if (aDate < bDate)\n return -1 * dir;\n if (aDate > bDate)\n return 1 * dir;\n continue;\n }\n if (aValue < bValue)\n return -1 * dir;\n if (aValue > bValue)\n return 1 * dir;\n }\n return 0;\n });\n }\n else {\n // Default sort by registration date descending (newest first)\n results.sort((a, b) => {\n return b.registrationDate.getTime() - a.registrationDate.getTime();\n });\n }\n // Apply pagination\n const skip = input.skip || 0;\n const take = input.take || 10;\n const total = results.length;\n const items = results.slice(skip, skip + take);\n return {\n items,\n total,\n };\n }\n //#region ---- Helper Methods ----\n /**\n * Get application counts for tenants\n */\n async getApplicationCountsForTenants(tenantIds, subscriptions, planItems, applications) {\n const counts = new Map();\n for (const tenantId of tenantIds) {\n const tenantSubscriptions = subscriptions.filter((s) => s.subscriberId === tenantId);\n // Get unique applications through subscriptions → plans → plan items → editions → applications\n const applicationSet = new Set();\n for (const subscription of tenantSubscriptions) {\n if (!subscription.planId)\n continue;\n // Get plan items for this plan\n const planPlanItems = planItems.filter((pi) => pi.planId === subscription.planId);\n for (const planItem of planPlanItems) {\n if (!planItem.editionId)\n continue;\n // Find applications that have this edition\n for (const application of applications) {\n if (application.editionIds && application.editionIds.includes(planItem.editionId)) {\n applicationSet.add(application.id);\n }\n }\n }\n }\n counts.set(tenantId, applicationSet.size);\n }\n return counts;\n }\n //#endregion\n //#region ---- Custom Filter Handlers ----\n /**\n * Apply date range filter to tenants\n */\n applyDateRangeFilter(tenants, dateRange) {\n if (!dateRange || !dateRange.start || !dateRange.end) {\n return tenants;\n }\n const startDate = new Date(dateRange.start);\n const endDate = new Date(dateRange.end);\n endDate.setHours(23, 59, 59, 999);\n return tenants.filter((tenant) => {\n if (!tenant.registrationDate)\n return false;\n const regDate = new Date(tenant.registrationDate);\n return regDate >= startDate && regDate <= endDate;\n });\n }\n static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"20.3.12\", ngImport: i0, type: GetTenantRegistrationTimelineQuery, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }\n static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"20.3.12\", ngImport: i0, type: GetTenantRegistrationTimelineQuery, providedIn: 'root' }); }\n}\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"20.3.12\", ngImport: i0, type: GetTenantRegistrationTimelineQuery, decorators: [{\n type: Injectable,\n args: [{\n providedIn: 'root',\n }]\n }] });\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"get-tenant-registration-timeline.query.js","sourceRoot":"","sources":["../../../../../../../../libs/connectivity/mock/src/lib/tenant-management/reports/get-tenant-registration-timeline.query.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { AXPQuery } from '@acorex/platform/runtime';\nimport { AXPEntityService } from '@acorex/platform/layout/entity';\nimport { AXMTenantManagementTenantEntityModel } from '@acorex/modules/tenant-management';\nimport { AXMTenantManagementTenantUserEntityModel } from '@acorex/modules/tenant-management';\nimport { AXMSubscriptionManagementSubscriptionEntityModel } from '@acorex/modules/subscription-management';\nimport { AXMSubscriptionManagementSubscriptionPlanItemEntityModel } from '@acorex/modules/subscription-management';\nimport { AXMApplicationManagementApplicationEntityModel } from '@acorex/modules/application-management';\nimport { RootConfig as TenantRootConfig } from '@acorex/modules/tenant-management';\nimport { RootConfig as SubscriptionRootConfig } from '@acorex/modules/subscription-management';\nimport { RootConfig as ApplicationRootConfig } from '@acorex/modules/application-management';\n\n//#region ----   Query Input/Output Types   ----\n\nexport interface GetTenantRegistrationTimelineQueryInput {\n  skip?: number;\n  take?: number;\n  sort?: Array<{ field: string; dir: 'asc' | 'desc' }>;\n  filters?: Array<{\n    field: string;\n    operator: { type: string };\n    value: any;\n  }>;\n  logic?: 'and' | 'or';\n}\n\nexport interface TenantRegistrationTimelineResult {\n  tenantId: string;\n  tenantName: string;\n  registrationDate: Date;\n  registrationDateStr: string;\n  status: string;\n  country?: string;\n  userCount: number;\n  applicationCount: number;\n  subscriptionCount: number;\n  month: string;\n  monthNumber: number;\n  quarter: string;\n  year: number;\n}\n\nexport interface GetTenantRegistrationTimelineQueryResult {\n  items: TenantRegistrationTimelineResult[];\n  total: number;\n}\n\n//#endregion\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class GetTenantRegistrationTimelineQuery\n  implements AXPQuery<GetTenantRegistrationTimelineQueryInput, GetTenantRegistrationTimelineQueryResult>\n{\n  private readonly entityService = inject(AXPEntityService);\n  private tenantService = this.entityService\n    .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenant.name}`)\n    .data<string, AXMTenantManagementTenantEntityModel>();\n  private tenantUserService = this.entityService\n    .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenantUser.name}`)\n    .data<string, AXMTenantManagementTenantUserEntityModel>();\n  private subscriptionService = this.entityService\n    .withEntity(`${SubscriptionRootConfig.module.name}.${SubscriptionRootConfig.entities.subscription.name}`)\n    .data<string, AXMSubscriptionManagementSubscriptionEntityModel>();\n  private planItemService = this.entityService\n    .withEntity(`${SubscriptionRootConfig.module.name}.${SubscriptionRootConfig.entities.subscriptionPlanItem.name}`)\n    .data<string, AXMSubscriptionManagementSubscriptionPlanItemEntityModel>();\n  private applicationService = this.entityService\n    .withEntity(`${ApplicationRootConfig.module.name}.${ApplicationRootConfig.entities.application.name}`)\n    .data<string, AXMApplicationManagementApplicationEntityModel>();\n\n  async fetch(input: GetTenantRegistrationTimelineQueryInput): Promise<GetTenantRegistrationTimelineQueryResult> {\n    // Get all tenants\n    const tenantsResult = await this.tenantService.query({\n      skip: 0,\n      take: 10000,\n      filter:\n        input.filters && input.filters.length > 0\n          ? ({\n              logic: input.logic || 'and',\n              filters: input.filters.filter((f) => f.field !== 'dateRange'),\n            } as any)\n          : undefined,\n    });\n\n    let tenants = tenantsResult.items.filter((t) => t.registrationDate);\n\n    // Apply date range filter if provided\n    if (input.filters) {\n      for (const filter of input.filters) {\n        if (filter.field === 'dateRange' && filter.value) {\n          tenants = this.applyDateRangeFilter(tenants, filter.value);\n        }\n      }\n    }\n\n    // Get all tenant users for counting\n    const tenantUsersResult = await this.tenantUserService.query({\n      skip: 0,\n      take: 10000,\n    });\n    const tenantUsers = tenantUsersResult.items;\n\n    // Get all subscriptions for counting\n    const subscriptionsResult = await this.subscriptionService.query({\n      skip: 0,\n      take: 10000,\n    });\n    const subscriptions = subscriptionsResult.items.filter((s) => s.subscriberType === 'Tenant');\n\n    // Get all plan items and applications for application counting\n    const planItemsResult = await this.planItemService.query({\n      skip: 0,\n      take: 10000,\n    });\n    const planItems = planItemsResult.items;\n\n    const applicationsResult = await this.applicationService.query({\n      skip: 0,\n      take: 10000,\n    });\n    const applications = applicationsResult.items;\n\n    // Get application counts for tenants\n    const applicationCounts = await this.getApplicationCountsForTenants(\n      tenants.map((t) => t.id),\n      subscriptions,\n      planItems,\n      applications\n    );\n\n    // Build results\n    const results: TenantRegistrationTimelineResult[] = [];\n\n    for (const tenant of tenants) {\n      if (!tenant.registrationDate) continue;\n\n      const registrationDate = new Date(tenant.registrationDate);\n\n      // Extract time period info\n      const year = registrationDate.getFullYear();\n      const month = registrationDate.getMonth() + 1;\n      const monthName = registrationDate.toLocaleString('en-US', { month: 'long' });\n      const quarter = Math.ceil(month / 3);\n      const quarterName = `Q${quarter}`;\n\n      // Count users for this tenant\n      const userCount = tenantUsers.filter((tu) => tu.tenantId === tenant.id).length;\n\n      // Count subscriptions for this tenant\n      const subscriptionCount = subscriptions.filter((s) => s.subscriberId === tenant.id).length;\n\n      // Get application count\n      const applicationCount = applicationCounts.get(tenant.id) || 0;\n\n      results.push({\n        tenantId: tenant.id,\n        tenantName: tenant.title,\n        registrationDate: registrationDate,\n        registrationDateStr: registrationDate.toISOString(),\n        status: tenant.statusId || 'Unknown',\n        country: tenant.country?.title,\n        userCount: userCount,\n        applicationCount: applicationCount,\n        subscriptionCount: subscriptionCount,\n        month: monthName,\n        monthNumber: month,\n        quarter: quarterName,\n        year: year,\n      });\n    }\n\n    // Apply sorting if provided\n    if (input.sort && input.sort.length > 0) {\n      results.sort((a, b) => {\n        for (const sortField of input.sort!) {\n          const field = sortField.field as keyof TenantRegistrationTimelineResult;\n          const aValue = a[field];\n          const bValue = b[field];\n          const dir = sortField.dir === 'asc' ? 1 : -1;\n\n          if (aValue === undefined && bValue === undefined) continue;\n          if (aValue === undefined) return 1 * dir;\n          if (bValue === undefined) return -1 * dir;\n\n          // Handle date comparison\n          if (field === 'registrationDate') {\n            const aDate = a.registrationDate.getTime();\n            const bDate = b.registrationDate.getTime();\n            if (aDate < bDate) return -1 * dir;\n            if (aDate > bDate) return 1 * dir;\n            continue;\n          }\n\n          if (aValue < bValue) return -1 * dir;\n          if (aValue > bValue) return 1 * dir;\n        }\n        return 0;\n      });\n    } else {\n      // Default sort by registration date descending (newest first)\n      results.sort((a, b) => {\n        return b.registrationDate.getTime() - a.registrationDate.getTime();\n      });\n    }\n\n    // Apply pagination\n    const skip = input.skip || 0;\n    const take = input.take || 10;\n    const total = results.length;\n    const items = results.slice(skip, skip + take);\n\n    return {\n      items,\n      total,\n    };\n  }\n\n  //#region ----   Helper Methods   ----\n\n  /**\n   * Get application counts for tenants\n   */\n  private async getApplicationCountsForTenants(\n    tenantIds: string[],\n    subscriptions: AXMSubscriptionManagementSubscriptionEntityModel[],\n    planItems: AXMSubscriptionManagementSubscriptionPlanItemEntityModel[],\n    applications: AXMApplicationManagementApplicationEntityModel[]\n  ): Promise<Map<string, number>> {\n    const counts = new Map<string, number>();\n\n    for (const tenantId of tenantIds) {\n      const tenantSubscriptions = subscriptions.filter((s) => s.subscriberId === tenantId);\n\n      // Get unique applications through subscriptions → plans → plan items → editions → applications\n      const applicationSet = new Set<string>();\n\n      for (const subscription of tenantSubscriptions) {\n        if (!subscription.planId) continue;\n\n        // Get plan items for this plan\n        const planPlanItems = planItems.filter((pi) => pi.planId === subscription.planId);\n\n        for (const planItem of planPlanItems) {\n          if (!planItem.editionId) continue;\n\n          // Find applications that have this edition\n          for (const application of applications) {\n            if (application.editionIds && application.editionIds.includes(planItem.editionId)) {\n              applicationSet.add(application.id);\n            }\n          }\n        }\n      }\n\n      counts.set(tenantId, applicationSet.size);\n    }\n\n    return counts;\n  }\n\n  //#endregion\n\n  //#region ----   Custom Filter Handlers   ----\n\n  /**\n   * Apply date range filter to tenants\n   */\n  private applyDateRangeFilter(\n    tenants: AXMTenantManagementTenantEntityModel[],\n    dateRange: any\n  ): AXMTenantManagementTenantEntityModel[] {\n    if (!dateRange || !dateRange.start || !dateRange.end) {\n      return tenants;\n    }\n\n    const startDate = new Date(dateRange.start);\n    const endDate = new Date(dateRange.end);\n    endDate.setHours(23, 59, 59, 999);\n\n    return tenants.filter((tenant) => {\n      if (!tenant.registrationDate) return false;\n      const regDate = new Date(tenant.registrationDate);\n      return regDate >= startDate && regDate <= endDate;\n    });\n  }\n\n  //#endregion\n}\n\n"]}"],"names":["TenantRootConfig","SubscriptionRootConfig","ApplicationRootConfig"],"mappings":"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"}
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
import * as i0 from '@angular/core';
|
|
2
|
+
import { inject, Injectable } from '@angular/core';
|
|
3
|
+
import { AXPEntityService } from '@acorex/platform/layout/entity';
|
|
4
|
+
import { AXPSystemStatusType } from '@acorex/platform/common';
|
|
5
|
+
import { RootConfig } from '@acorex/modules/tenant-management';
|
|
6
|
+
|
|
7
|
+
//#endregion
|
|
8
|
+
class GetTenantUserDistributionQuery {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.entityService = inject(AXPEntityService);
|
|
11
|
+
this.tenantService = this.entityService
|
|
12
|
+
.withEntity(`${RootConfig.module.name}.${RootConfig.entities.tenant.name}`)
|
|
13
|
+
.data();
|
|
14
|
+
this.tenantUserService = this.entityService
|
|
15
|
+
.withEntity(`${RootConfig.module.name}.${RootConfig.entities.tenantUser.name}`)
|
|
16
|
+
.data();
|
|
17
|
+
}
|
|
18
|
+
async fetch(input) {
|
|
19
|
+
// Get all tenants
|
|
20
|
+
const tenantsResult = await this.tenantService.query({
|
|
21
|
+
skip: 0,
|
|
22
|
+
take: 10000,
|
|
23
|
+
filter: input.filters && input.filters.length > 0
|
|
24
|
+
? {
|
|
25
|
+
logic: input.logic || 'and',
|
|
26
|
+
filters: input.filters.filter((f) => f.field !== 'dateRange'),
|
|
27
|
+
}
|
|
28
|
+
: undefined,
|
|
29
|
+
});
|
|
30
|
+
let tenants = tenantsResult.items;
|
|
31
|
+
// Apply date range filter if provided
|
|
32
|
+
if (input.filters) {
|
|
33
|
+
for (const filter of input.filters) {
|
|
34
|
+
if (filter.field === 'dateRange' && filter.value) {
|
|
35
|
+
tenants = this.applyDateRangeFilter(tenants, filter.value);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
// Get all tenant users
|
|
40
|
+
const tenantUsersResult = await this.tenantUserService.query({
|
|
41
|
+
skip: 0,
|
|
42
|
+
take: 10000,
|
|
43
|
+
});
|
|
44
|
+
const tenantUsers = tenantUsersResult.items;
|
|
45
|
+
// Build results for each tenant
|
|
46
|
+
const results = [];
|
|
47
|
+
for (const tenant of tenants) {
|
|
48
|
+
// Get tenant's users
|
|
49
|
+
const tenantUsersList = tenantUsers.filter((tu) => tu.tenantId === tenant.id);
|
|
50
|
+
// Count active and inactive users
|
|
51
|
+
const activeUsers = tenantUsersList.filter((tu) => tu.statusId === AXPSystemStatusType.Active);
|
|
52
|
+
const inactiveUsers = tenantUsersList.filter((tu) => tu.statusId !== AXPSystemStatusType.Active);
|
|
53
|
+
// Calculate utilization percentage
|
|
54
|
+
const maxUsers = tenant.maxUsers;
|
|
55
|
+
const utilizationPercent = maxUsers && maxUsers > 0 ? Math.round((tenantUsersList.length / maxUsers) * 100) : 0;
|
|
56
|
+
results.push({
|
|
57
|
+
tenantId: tenant.id,
|
|
58
|
+
tenantName: tenant.title,
|
|
59
|
+
tenantStatus: tenant.statusId || 'Unknown',
|
|
60
|
+
totalUsersCount: tenantUsersList.length,
|
|
61
|
+
activeUsersCount: activeUsers.length,
|
|
62
|
+
inactiveUsersCount: inactiveUsers.length,
|
|
63
|
+
maxUsers: maxUsers,
|
|
64
|
+
utilizationPercent: utilizationPercent,
|
|
65
|
+
registrationDate: tenant.registrationDate,
|
|
66
|
+
registrationDateStr: tenant.registrationDate?.toISOString(),
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
// Apply sorting if provided
|
|
70
|
+
if (input.sort && input.sort.length > 0) {
|
|
71
|
+
results.sort((a, b) => {
|
|
72
|
+
for (const sortField of input.sort) {
|
|
73
|
+
const field = sortField.field;
|
|
74
|
+
const aValue = a[field];
|
|
75
|
+
const bValue = b[field];
|
|
76
|
+
const dir = sortField.dir === 'asc' ? 1 : -1;
|
|
77
|
+
if (aValue === undefined && bValue === undefined)
|
|
78
|
+
continue;
|
|
79
|
+
if (aValue === undefined)
|
|
80
|
+
return 1 * dir;
|
|
81
|
+
if (bValue === undefined)
|
|
82
|
+
return -1 * dir;
|
|
83
|
+
if (aValue < bValue)
|
|
84
|
+
return -1 * dir;
|
|
85
|
+
if (aValue > bValue)
|
|
86
|
+
return 1 * dir;
|
|
87
|
+
}
|
|
88
|
+
return 0;
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
// Default sort by tenant name
|
|
93
|
+
results.sort((a, b) => a.tenantName.localeCompare(b.tenantName));
|
|
94
|
+
}
|
|
95
|
+
// Apply pagination
|
|
96
|
+
const skip = input.skip || 0;
|
|
97
|
+
const take = input.take || 10;
|
|
98
|
+
const total = results.length;
|
|
99
|
+
const items = results.slice(skip, skip + take);
|
|
100
|
+
return {
|
|
101
|
+
items,
|
|
102
|
+
total,
|
|
103
|
+
};
|
|
104
|
+
}
|
|
105
|
+
//#region ---- Custom Filter Handlers ----
|
|
106
|
+
/**
|
|
107
|
+
* Apply date range filter to tenants
|
|
108
|
+
*/
|
|
109
|
+
applyDateRangeFilter(tenants, dateRange) {
|
|
110
|
+
if (!dateRange || !dateRange.start || !dateRange.end) {
|
|
111
|
+
return tenants;
|
|
112
|
+
}
|
|
113
|
+
const startDate = new Date(dateRange.start);
|
|
114
|
+
const endDate = new Date(dateRange.end);
|
|
115
|
+
endDate.setHours(23, 59, 59, 999);
|
|
116
|
+
return tenants.filter((tenant) => {
|
|
117
|
+
if (!tenant.registrationDate)
|
|
118
|
+
return false;
|
|
119
|
+
const regDate = new Date(tenant.registrationDate);
|
|
120
|
+
return regDate >= startDate && regDate <= endDate;
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.12", ngImport: i0, type: GetTenantUserDistributionQuery, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
124
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.12", ngImport: i0, type: GetTenantUserDistributionQuery, providedIn: 'root' }); }
|
|
125
|
+
}
|
|
126
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.12", ngImport: i0, type: GetTenantUserDistributionQuery, decorators: [{
|
|
127
|
+
type: Injectable,
|
|
128
|
+
args: [{
|
|
129
|
+
providedIn: 'root',
|
|
130
|
+
}]
|
|
131
|
+
}] });
|
|
132
|
+
|
|
133
|
+
export { GetTenantUserDistributionQuery };
|
|
134
|
+
//# sourceMappingURL=acorex-connectivity-mock-get-tenant-user-distribution.query-Bs5U5tze.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"acorex-connectivity-mock-get-tenant-user-distribution.query-Bs5U5tze.mjs","sources":["../tmp-esm2022/mock/lib/tenant-management/reports/get-tenant-user-distribution.query.js"],"sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { AXPEntityService } from '@acorex/platform/layout/entity';\nimport { AXPSystemStatusType } from '@acorex/platform/common';\nimport { RootConfig as TenantRootConfig } from '@acorex/modules/tenant-management';\nimport * as i0 from \"@angular/core\";\n//#endregion\nexport class GetTenantUserDistributionQuery {\n constructor() {\n this.entityService = inject(AXPEntityService);\n this.tenantService = this.entityService\n .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenant.name}`)\n .data();\n this.tenantUserService = this.entityService\n .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenantUser.name}`)\n .data();\n }\n async fetch(input) {\n // Get all tenants\n const tenantsResult = await this.tenantService.query({\n skip: 0,\n take: 10000,\n filter: input.filters && input.filters.length > 0\n ? {\n logic: input.logic || 'and',\n filters: input.filters.filter((f) => f.field !== 'dateRange'),\n }\n : undefined,\n });\n let tenants = tenantsResult.items;\n // Apply date range filter if provided\n if (input.filters) {\n for (const filter of input.filters) {\n if (filter.field === 'dateRange' && filter.value) {\n tenants = this.applyDateRangeFilter(tenants, filter.value);\n }\n }\n }\n // Get all tenant users\n const tenantUsersResult = await this.tenantUserService.query({\n skip: 0,\n take: 10000,\n });\n const tenantUsers = tenantUsersResult.items;\n // Build results for each tenant\n const results = [];\n for (const tenant of tenants) {\n // Get tenant's users\n const tenantUsersList = tenantUsers.filter((tu) => tu.tenantId === tenant.id);\n // Count active and inactive users\n const activeUsers = tenantUsersList.filter((tu) => tu.statusId === AXPSystemStatusType.Active);\n const inactiveUsers = tenantUsersList.filter((tu) => tu.statusId !== AXPSystemStatusType.Active);\n // Calculate utilization percentage\n const maxUsers = tenant.maxUsers;\n const utilizationPercent = maxUsers && maxUsers > 0 ? Math.round((tenantUsersList.length / maxUsers) * 100) : 0;\n results.push({\n tenantId: tenant.id,\n tenantName: tenant.title,\n tenantStatus: tenant.statusId || 'Unknown',\n totalUsersCount: tenantUsersList.length,\n activeUsersCount: activeUsers.length,\n inactiveUsersCount: inactiveUsers.length,\n maxUsers: maxUsers,\n utilizationPercent: utilizationPercent,\n registrationDate: tenant.registrationDate,\n registrationDateStr: tenant.registrationDate?.toISOString(),\n });\n }\n // Apply sorting if provided\n if (input.sort && input.sort.length > 0) {\n results.sort((a, b) => {\n for (const sortField of input.sort) {\n const field = sortField.field;\n const aValue = a[field];\n const bValue = b[field];\n const dir = sortField.dir === 'asc' ? 1 : -1;\n if (aValue === undefined && bValue === undefined)\n continue;\n if (aValue === undefined)\n return 1 * dir;\n if (bValue === undefined)\n return -1 * dir;\n if (aValue < bValue)\n return -1 * dir;\n if (aValue > bValue)\n return 1 * dir;\n }\n return 0;\n });\n }\n else {\n // Default sort by tenant name\n results.sort((a, b) => a.tenantName.localeCompare(b.tenantName));\n }\n // Apply pagination\n const skip = input.skip || 0;\n const take = input.take || 10;\n const total = results.length;\n const items = results.slice(skip, skip + take);\n return {\n items,\n total,\n };\n }\n //#region ---- Custom Filter Handlers ----\n /**\n * Apply date range filter to tenants\n */\n applyDateRangeFilter(tenants, dateRange) {\n if (!dateRange || !dateRange.start || !dateRange.end) {\n return tenants;\n }\n const startDate = new Date(dateRange.start);\n const endDate = new Date(dateRange.end);\n endDate.setHours(23, 59, 59, 999);\n return tenants.filter((tenant) => {\n if (!tenant.registrationDate)\n return false;\n const regDate = new Date(tenant.registrationDate);\n return regDate >= startDate && regDate <= endDate;\n });\n }\n static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"20.3.12\", ngImport: i0, type: GetTenantUserDistributionQuery, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }\n static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"20.3.12\", ngImport: i0, type: GetTenantUserDistributionQuery, providedIn: 'root' }); }\n}\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"20.3.12\", ngImport: i0, type: GetTenantUserDistributionQuery, decorators: [{\n type: Injectable,\n args: [{\n providedIn: 'root',\n }]\n }] });\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"get-tenant-user-distribution.query.js","sourceRoot":"","sources":["../../../../../../../../libs/connectivity/mock/src/lib/tenant-management/reports/get-tenant-user-distribution.query.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { AXPQuery } from '@acorex/platform/runtime';\nimport { AXPEntityService } from '@acorex/platform/layout/entity';\nimport { AXMTenantManagementTenantEntityModel } from '@acorex/modules/tenant-management';\nimport { AXMTenantManagementTenantUserEntityModel } from '@acorex/modules/tenant-management';\nimport { AXPSystemStatusType } from '@acorex/platform/common';\nimport { RootConfig as TenantRootConfig } from '@acorex/modules/tenant-management';\n\n//#region ----   Query Input/Output Types   ----\n\nexport interface GetTenantUserDistributionQueryInput {\n  skip?: number;\n  take?: number;\n  sort?: Array<{ field: string; dir: 'asc' | 'desc' }>;\n  filters?: Array<{\n    field: string;\n    operator: { type: string };\n    value: any;\n  }>;\n  logic?: 'and' | 'or';\n}\n\nexport interface TenantUserDistributionResult {\n  tenantId: string;\n  tenantName: string;\n  tenantStatus: string;\n  totalUsersCount: number;\n  activeUsersCount: number;\n  inactiveUsersCount: number;\n  maxUsers?: number;\n  utilizationPercent: number;\n  registrationDate?: Date;\n  registrationDateStr?: string;\n}\n\nexport interface GetTenantUserDistributionQueryResult {\n  items: TenantUserDistributionResult[];\n  total: number;\n}\n\n//#endregion\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class GetTenantUserDistributionQuery\n  implements AXPQuery<GetTenantUserDistributionQueryInput, GetTenantUserDistributionQueryResult>\n{\n  private readonly entityService = inject(AXPEntityService);\n  private tenantService = this.entityService\n    .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenant.name}`)\n    .data<string, AXMTenantManagementTenantEntityModel>();\n  private tenantUserService = this.entityService\n    .withEntity(`${TenantRootConfig.module.name}.${TenantRootConfig.entities.tenantUser.name}`)\n    .data<string, AXMTenantManagementTenantUserEntityModel>();\n\n  async fetch(input: GetTenantUserDistributionQueryInput): Promise<GetTenantUserDistributionQueryResult> {\n    // Get all tenants\n    const tenantsResult = await this.tenantService.query({\n      skip: 0,\n      take: 10000,\n      filter:\n        input.filters && input.filters.length > 0\n          ? ({\n              logic: input.logic || 'and',\n              filters: input.filters.filter((f) => f.field !== 'dateRange'),\n            } as any)\n          : undefined,\n    });\n\n    let tenants = tenantsResult.items;\n\n    // Apply date range filter if provided\n    if (input.filters) {\n      for (const filter of input.filters) {\n        if (filter.field === 'dateRange' && filter.value) {\n          tenants = this.applyDateRangeFilter(tenants, filter.value);\n        }\n      }\n    }\n\n    // Get all tenant users\n    const tenantUsersResult = await this.tenantUserService.query({\n      skip: 0,\n      take: 10000,\n    });\n    const tenantUsers = tenantUsersResult.items;\n\n    // Build results for each tenant\n    const results: TenantUserDistributionResult[] = [];\n\n    for (const tenant of tenants) {\n      // Get tenant's users\n      const tenantUsersList = tenantUsers.filter((tu) => tu.tenantId === tenant.id);\n\n      // Count active and inactive users\n      const activeUsers = tenantUsersList.filter(\n        (tu) => tu.statusId === AXPSystemStatusType.Active\n      );\n      const inactiveUsers = tenantUsersList.filter(\n        (tu) => tu.statusId !== AXPSystemStatusType.Active\n      );\n\n      // Calculate utilization percentage\n      const maxUsers = tenant.maxUsers;\n      const utilizationPercent =\n        maxUsers && maxUsers > 0 ? Math.round((tenantUsersList.length / maxUsers) * 100) : 0;\n\n      results.push({\n        tenantId: tenant.id,\n        tenantName: tenant.title,\n        tenantStatus: tenant.statusId || 'Unknown',\n        totalUsersCount: tenantUsersList.length,\n        activeUsersCount: activeUsers.length,\n        inactiveUsersCount: inactiveUsers.length,\n        maxUsers: maxUsers,\n        utilizationPercent: utilizationPercent,\n        registrationDate: tenant.registrationDate,\n        registrationDateStr: tenant.registrationDate?.toISOString(),\n      });\n    }\n\n    // Apply sorting if provided\n    if (input.sort && input.sort.length > 0) {\n      results.sort((a, b) => {\n        for (const sortField of input.sort!) {\n          const field = sortField.field as keyof TenantUserDistributionResult;\n          const aValue = a[field];\n          const bValue = b[field];\n          const dir = sortField.dir === 'asc' ? 1 : -1;\n\n          if (aValue === undefined && bValue === undefined) continue;\n          if (aValue === undefined) return 1 * dir;\n          if (bValue === undefined) return -1 * dir;\n\n          if (aValue < bValue) return -1 * dir;\n          if (aValue > bValue) return 1 * dir;\n        }\n        return 0;\n      });\n    } else {\n      // Default sort by tenant name\n      results.sort((a, b) => a.tenantName.localeCompare(b.tenantName));\n    }\n\n    // Apply pagination\n    const skip = input.skip || 0;\n    const take = input.take || 10;\n    const total = results.length;\n    const items = results.slice(skip, skip + take);\n\n    return {\n      items,\n      total,\n    };\n  }\n\n  //#region ----   Custom Filter Handlers   ----\n\n  /**\n   * Apply date range filter to tenants\n   */\n  private applyDateRangeFilter(\n    tenants: AXMTenantManagementTenantEntityModel[],\n    dateRange: any\n  ): AXMTenantManagementTenantEntityModel[] {\n    if (!dateRange || !dateRange.start || !dateRange.end) {\n      return tenants;\n    }\n\n    const startDate = new Date(dateRange.start);\n    const endDate = new Date(dateRange.end);\n    endDate.setHours(23, 59, 59, 999);\n\n    return tenants.filter((tenant) => {\n      if (!tenant.registrationDate) return false;\n      const regDate = new Date(tenant.registrationDate);\n      return regDate >= startDate && regDate <= endDate;\n    });\n  }\n\n  //#endregion\n}\n\n"]}"],"names":["TenantRootConfig"],"mappings":"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"}
|