import { sha256 } from 'js-sha256'; import { Logger } from '../logger'; import { IUnleashConfig } from '../types/option'; import { IClientInstanceStore, IEventStore, IUnleashStores, } from '../types/stores'; import { IContextFieldStore } from '../types/stores/context-field-store'; import { IEnvironmentStore } from '../types/stores/environment-store'; import { IFeatureToggleStore } from '../types/stores/feature-toggle-store'; import { IGroupStore } from '../types/stores/group-store'; import { IProjectStore } from '../types/stores/project-store'; import { IStrategyStore } from '../types/stores/strategy-store'; import { IUserStore } from '../types/stores/user-store'; import { ISegmentStore } from '../types/stores/segment-store'; import { IRoleStore } from '../types/stores/role-store'; import VersionService from './version-service'; import { ISettingStore } from '../types/stores/settings-store'; import { FEATURES_EXPORTED, FEATURES_IMPORTED } from '../types'; export type TimeRange = 'allTime' | '30d' | '7d'; export interface InstanceStats { instanceId: string; timestamp: Date; versionOSS: string; versionEnterprise?: string; users: number; featureToggles: number; projects: number; contextFields: number; roles: number; featureExports: number; featureImports: number; groups: number; environments: number; segments: number; strategies: number; SAMLenabled: boolean; OIDCenabled: boolean; clientApps: { range: TimeRange; count: number }[]; } export interface InstanceStatsSigned extends InstanceStats { sum: string; } export class InstanceStatsService { private logger: Logger; private strategyStore: IStrategyStore; private userStore: IUserStore; private featureToggleStore: IFeatureToggleStore; private contextFieldStore: IContextFieldStore; private projectStore: IProjectStore; private groupStore: IGroupStore; private environmentStore: IEnvironmentStore; private segmentStore: ISegmentStore; private roleStore: IRoleStore; private eventStore: IEventStore; private versionService: VersionService; private settingStore: ISettingStore; private clientInstanceStore: IClientInstanceStore; private snapshot?: InstanceStats; private appCount?: Partial<{ [key in TimeRange]: number }>; constructor( { featureToggleStore, userStore, projectStore, environmentStore, strategyStore, contextFieldStore, groupStore, segmentStore, roleStore, settingStore, clientInstanceStore, eventStore, }: Pick< IUnleashStores, | 'featureToggleStore' | 'userStore' | 'projectStore' | 'environmentStore' | 'strategyStore' | 'contextFieldStore' | 'groupStore' | 'segmentStore' | 'roleStore' | 'settingStore' | 'clientInstanceStore' | 'eventStore' >, { getLogger }: Pick, versionService: VersionService, ) { this.strategyStore = strategyStore; this.userStore = userStore; this.featureToggleStore = featureToggleStore; this.environmentStore = environmentStore; this.projectStore = projectStore; this.groupStore = groupStore; this.contextFieldStore = contextFieldStore; this.segmentStore = segmentStore; this.roleStore = roleStore; this.versionService = versionService; this.settingStore = settingStore; this.eventStore = eventStore; this.clientInstanceStore = clientInstanceStore; this.logger = getLogger('services/stats-service.js'); } async refreshStatsSnapshot(): Promise { try { this.snapshot = await this.getStats(); const appCountReplacement = {}; this.snapshot.clientApps?.forEach((appCount) => { appCountReplacement[appCount.range] = appCount.count; }); this.appCount = appCountReplacement; } catch (error) { this.logger.warn( 'Unable to retrieve statistics. This will be retried', error, ); } } getToggleCount(): Promise { return this.featureToggleStore.count({ archived: false, }); } async hasOIDC(): Promise { const settings = await this.settingStore.get( 'unleash.enterprise.auth.oidc', ); return settings?.enabled || false; } async hasSAML(): Promise { const settings = await this.settingStore.get( 'unleash.enterprise.auth.saml', ); return settings?.enabled || false; } /** * use getStatsSnapshot for low latency, sacrificing data-freshness */ async getStats(): Promise { const versionInfo = this.versionService.getVersionInfo(); const [ featureToggles, users, projects, contextFields, groups, roles, environments, segments, strategies, SAMLenabled, OIDCenabled, clientApps, featureExports, featureImports, ] = await Promise.all([ this.getToggleCount(), this.userStore.count(), this.projectStore.count(), this.contextFieldStore.count(), this.groupStore.count(), this.roleStore.count(), this.environmentStore.count(), this.segmentStore.count(), this.strategyStore.count(), this.hasSAML(), this.hasOIDC(), this.getLabeledAppCounts(), this.eventStore.filteredCount({ type: FEATURES_EXPORTED }), this.eventStore.filteredCount({ type: FEATURES_IMPORTED }), ]); return { timestamp: new Date(), instanceId: versionInfo.instanceId, versionOSS: versionInfo.current.oss, versionEnterprise: versionInfo.current.enterprise, users, featureToggles, projects, contextFields, roles, groups, environments, segments, strategies, SAMLenabled, OIDCenabled, clientApps, featureExports, featureImports, }; } getStatsSnapshot(): InstanceStats | undefined { return this.snapshot; } async getLabeledAppCounts(): Promise< { range: TimeRange; count: number }[] > { return [ { range: 'allTime', count: await this.clientInstanceStore.getDistinctApplicationsCount(), }, { range: '30d', count: await this.clientInstanceStore.getDistinctApplicationsCount( 30, ), }, { range: '7d', count: await this.clientInstanceStore.getDistinctApplicationsCount( 7, ), }, ]; } getAppCountSnapshot(range: TimeRange): number | undefined { return this.appCount?.[range]; } async getSignedStats(): Promise { const instanceStats = await this.getStats(); const sum = sha256( `${instanceStats.instanceId}${instanceStats.users}${instanceStats.featureToggles}${instanceStats.projects}${instanceStats.roles}${instanceStats.groups}${instanceStats.environments}${instanceStats.segments}`, ); return { ...instanceStats, sum }; } }