mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-25 00:07:47 +01:00
5f3e5729b9
### What Adds the documentation for the instance admin endpoints. --------- Co-authored-by: Thomas Heartman <thomas@getunleash.ai>
267 lines
7.8 KiB
TypeScript
267 lines
7.8 KiB
TypeScript
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<IUnleashConfig, 'getLogger'>,
|
|
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<void> {
|
|
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<number> {
|
|
return this.featureToggleStore.count({
|
|
archived: false,
|
|
});
|
|
}
|
|
|
|
async hasOIDC(): Promise<boolean> {
|
|
const settings = await this.settingStore.get(
|
|
'unleash.enterprise.auth.oidc',
|
|
);
|
|
|
|
return settings?.enabled || false;
|
|
}
|
|
|
|
async hasSAML(): Promise<boolean> {
|
|
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<InstanceStats> {
|
|
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<InstanceStatsSigned> {
|
|
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 };
|
|
}
|
|
}
|