From 72de574012c03e8dc550a574b8e6b637d0342af4 Mon Sep 17 00:00:00 2001 From: Mateusz Kwasniewski Date: Wed, 26 Jun 2024 16:09:08 +0200 Subject: [PATCH] feat: largest projects and features metric (#7459) --- src/lib/db/index.ts | 2 + .../feature-lifecycle-read-model.test.ts | 8 +- .../fake-largest-resources-read-model.ts | 16 ++++ .../largest-resources-read-model-type.ts | 8 ++ .../largest-resources-read-model.test.ts | 94 +++++++++++++++++++ .../sizes/largest-resources-read-model.ts | 76 +++++++++++++++ src/lib/metrics.ts | 40 ++++++++ src/lib/types/stores.ts | 3 + src/test/fixtures/store.ts | 2 + 9 files changed, 245 insertions(+), 4 deletions(-) create mode 100644 src/lib/features/metrics/sizes/fake-largest-resources-read-model.ts create mode 100644 src/lib/features/metrics/sizes/largest-resources-read-model-type.ts create mode 100644 src/lib/features/metrics/sizes/largest-resources-read-model.test.ts create mode 100644 src/lib/features/metrics/sizes/largest-resources-read-model.ts diff --git a/src/lib/db/index.ts b/src/lib/db/index.ts index 1da459eb91..d7e97cf6db 100644 --- a/src/lib/db/index.ts +++ b/src/lib/db/index.ts @@ -48,6 +48,7 @@ import { FeatureLifecycleStore } from '../features/feature-lifecycle/feature-lif import { ProjectFlagCreatorsReadModel } from '../features/project/project-flag-creators-read-model'; import { FeatureStrategiesReadModel } from '../features/feature-toggle/feature-strategies-read-model'; import { FeatureLifecycleReadModel } from '../features/feature-lifecycle/feature-lifecycle-read-model'; +import { LargestResourcesReadModel } from '../features/metrics/sizes/largest-resources-read-model'; export const createStores = ( config: IUnleashConfig, @@ -166,6 +167,7 @@ export const createStores = ( db, config.flagResolver, ), + largestResourcesReadModel: new LargestResourcesReadModel(db), }; }; diff --git a/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.test.ts b/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.test.ts index e76813afc6..7522dc164c 100644 --- a/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.test.ts +++ b/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.test.ts @@ -7,7 +7,7 @@ import type { IFeatureToggleStore } from '../feature-toggle/types/feature-toggle import type { IFlagResolver } from '../../types'; let db: ITestDb; -let featureLifeycycleReadModel: IFeatureLifecycleReadModel; +let featureLifecycleReadModel: IFeatureLifecycleReadModel; let featureLifecycleStore: IFeatureLifecycleStore; let featureToggleStore: IFeatureToggleStore; @@ -19,7 +19,7 @@ const alwaysOnFlagResolver = { beforeAll(async () => { db = await dbInit('feature_lifecycle_read_model', getLogger); - featureLifeycycleReadModel = new FeatureLifecycleReadModel( + featureLifecycleReadModel = new FeatureLifecycleReadModel( db.rawDatabase, alwaysOnFlagResolver, ); @@ -59,14 +59,14 @@ test('can return stage count', async () => { { feature: 'featureA', stage: 'pre-live' }, ]); - const stageCount = await featureLifeycycleReadModel.getStageCount(); + const stageCount = await featureLifecycleReadModel.getStageCount(); expect(stageCount).toMatchObject([ { stage: 'pre-live', count: 1 }, { stage: 'initial', count: 2 }, ]); const stageCountByProject = - await featureLifeycycleReadModel.getStageCountByProject(); + await featureLifecycleReadModel.getStageCountByProject(); expect(stageCountByProject).toMatchObject([ { project: 'default', stage: 'pre-live', count: 1 }, { project: 'default', stage: 'initial', count: 2 }, diff --git a/src/lib/features/metrics/sizes/fake-largest-resources-read-model.ts b/src/lib/features/metrics/sizes/fake-largest-resources-read-model.ts new file mode 100644 index 0000000000..385e77e409 --- /dev/null +++ b/src/lib/features/metrics/sizes/fake-largest-resources-read-model.ts @@ -0,0 +1,16 @@ +import type { ILargestResourcesReadModel } from './largest-resources-read-model-type'; + +export class FakeLargestResourcesReadModel + implements ILargestResourcesReadModel +{ + async getLargestProjectEnvironments( + limit: number, + ): Promise<{ project: string; environment: string; size: number }[]> { + return []; + } + async getLargestFeatureEnvironments( + limit: number, + ): Promise<{ feature: string; environment: string; size: number }[]> { + return []; + } +} diff --git a/src/lib/features/metrics/sizes/largest-resources-read-model-type.ts b/src/lib/features/metrics/sizes/largest-resources-read-model-type.ts new file mode 100644 index 0000000000..a417c5d780 --- /dev/null +++ b/src/lib/features/metrics/sizes/largest-resources-read-model-type.ts @@ -0,0 +1,8 @@ +export interface ILargestResourcesReadModel { + getLargestProjectEnvironments( + limit: number, + ): Promise>; + getLargestFeatureEnvironments( + limit: number, + ): Promise>; +} diff --git a/src/lib/features/metrics/sizes/largest-resources-read-model.test.ts b/src/lib/features/metrics/sizes/largest-resources-read-model.test.ts new file mode 100644 index 0000000000..c1623db5aa --- /dev/null +++ b/src/lib/features/metrics/sizes/largest-resources-read-model.test.ts @@ -0,0 +1,94 @@ +import type { ILargestResourcesReadModel } from './largest-resources-read-model-type'; +import dbInit, { + type ITestDb, +} from '../../../../test/e2e/helpers/database-init'; +import type { IFeatureToggleStore } from '../../feature-toggle/types/feature-toggle-store-type'; +import getLogger from '../../../../test/fixtures/no-logger'; +import type { IFeatureStrategiesStore } from '../../feature-toggle/types/feature-toggle-strategies-store-type'; +import type { IFeatureStrategy } from '../../../types'; + +let db: ITestDb; +let largestResourcesReadModel: ILargestResourcesReadModel; +let featureToggleStore: IFeatureToggleStore; +let featureStrategiesStore: IFeatureStrategiesStore; + +beforeAll(async () => { + db = await dbInit('largest_resources_read_model', getLogger); + featureToggleStore = db.stores.featureToggleStore; + featureStrategiesStore = db.stores.featureStrategiesStore; + largestResourcesReadModel = db.stores.largestResourcesReadModel; +}); + +afterAll(async () => { + if (db) { + await db.destroy(); + } +}); + +beforeEach(async () => { + await featureToggleStore.deleteAll(); +}); + +type FeatureConfig = Pick< + IFeatureStrategy, + 'featureName' | 'constraints' | 'parameters' | 'variants' +>; +const createFeature = async (config: FeatureConfig) => { + await featureToggleStore.create('default', { + name: config.featureName, + createdByUserId: 9999, + }); + await featureStrategiesStore.createStrategyFeatureEnv({ + strategyName: 'flexibleRollout', + projectId: 'default', + environment: 'default', + featureName: config.featureName, + constraints: config.constraints, + parameters: config.parameters, + variants: config.variants, + }); +}; + +test('can calculate resource size', async () => { + await createFeature({ + featureName: 'featureA', + parameters: { + groupId: 'flag_init_test_1', + rollout: '25', + stickiness: 'default', + }, + constraints: [ + { + contextName: 'clientId', + operator: 'IN', + values: ['1', '2', '3', '4', '5', '6'], + caseInsensitive: false, + inverted: false, + }, + ], + variants: [ + { + name: 'a', + weight: 1000, + weightType: 'fix', + stickiness: 'default', + }, + ], + }); + + await createFeature({ + featureName: 'featureB', + parameters: {}, + constraints: [], + variants: [], + }); + + const [project] = + await largestResourcesReadModel.getLargestProjectEnvironments(1); + const [feature1, feature2] = + await largestResourcesReadModel.getLargestFeatureEnvironments(2); + + expect(project.size).toBeGreaterThan(400); + expect(project.size).toBe(feature1.size + feature2.size); + expect(feature1.size).toBeGreaterThan(feature2.size); +}); diff --git a/src/lib/features/metrics/sizes/largest-resources-read-model.ts b/src/lib/features/metrics/sizes/largest-resources-read-model.ts new file mode 100644 index 0000000000..bf4623fd4b --- /dev/null +++ b/src/lib/features/metrics/sizes/largest-resources-read-model.ts @@ -0,0 +1,76 @@ +import type { Db } from '../../../db/db'; +import type { ILargestResourcesReadModel } from './largest-resources-read-model-type'; + +export class LargestResourcesReadModel implements ILargestResourcesReadModel { + private db: Db; + + constructor(db: Db) { + this.db = db; + } + + async getLargestProjectEnvironments( + limit: number, + ): Promise> { + const { rows } = await this.db.raw(` + WITH ProjectSizes AS ( + SELECT + project_name, + environment, + SUM(pg_column_size(constraints) + pg_column_size(variants) + pg_column_size(parameters)) AS total_size + FROM + feature_strategies + GROUP BY + project_name, + environment + ) + SELECT + project_name, + environment, + total_size + FROM + ProjectSizes + ORDER BY + total_size DESC + LIMIT ${limit} + `); + + return rows.map((row) => ({ + project: row.project_name, + environment: row.environment, + size: Number(row.total_size), + })); + } + + async getLargestFeatureEnvironments( + limit: number, + ): Promise> { + const { rows } = await this.db.raw(` + WITH FeatureSizes AS ( + SELECT + feature_name, + environment, + SUM(pg_column_size(constraints) + pg_column_size(variants) + pg_column_size(parameters)) AS total_size + FROM + feature_strategies + GROUP BY + feature_name, + environment + ) + SELECT + feature_name, + environment, + total_size + FROM + FeatureSizes + ORDER BY + total_size DESC + LIMIT ${limit} + `); + + return rows.map((row) => ({ + feature: row.feature_name, + environment: row.environment, + size: Number(row.total_size), + })); + } +} diff --git a/src/lib/metrics.ts b/src/lib/metrics.ts index 861a547d4e..18e1ea04b1 100644 --- a/src/lib/metrics.ts +++ b/src/lib/metrics.ts @@ -129,6 +129,16 @@ export default class MetricsMonitor { help: 'Maximum number of constraints used on a single strategy', labelNames: ['feature', 'environment'], }); + const largestProjectEnvironment = createGauge({ + name: 'largest_project_environment_size', + help: 'The largest project environment size (bytes) based on strategies, constraints, variants and parameters', + labelNames: ['project', 'environment'], + }); + const largestFeatureEnvironment = createGauge({ + name: 'largest_feature_environment_size', + help: 'The largest feature environment size (bytes) base on strategies, constraints, variants and parameters', + labelNames: ['feature', 'environment'], + }); const featureTogglesArchivedTotal = createGauge({ name: 'feature_toggles_archived_total', @@ -313,6 +323,8 @@ export default class MetricsMonitor { maxConstraintsPerStrategyResult, stageCountByProjectResult, stageDurationByProject, + largestProjectEnvironments, + largestFeatureEnvironments, ] = await Promise.all([ stores.featureStrategiesReadModel.getMaxFeatureStrategies(), stores.featureStrategiesReadModel.getMaxFeatureEnvironmentStrategies(), @@ -320,6 +332,12 @@ export default class MetricsMonitor { stores.featureStrategiesReadModel.getMaxConstraintsPerStrategy(), stores.featureLifecycleReadModel.getStageCountByProject(), stores.featureLifecycleReadModel.getAllWithStageDuration(), + stores.largestResourcesReadModel.getLargestProjectEnvironments( + 1, + ), + stores.largestResourcesReadModel.getLargestFeatureEnvironments( + 1, + ), ]); featureFlagsTotal.reset(); @@ -403,6 +421,28 @@ export default class MetricsMonitor { .set(maxConstraintsPerStrategyResult.count); } + if (largestProjectEnvironments.length > 0) { + const projectEnvironment = largestProjectEnvironments[0]; + largestProjectEnvironment.reset(); + largestProjectEnvironment + .labels({ + project: projectEnvironment.project, + environment: projectEnvironment.environment, + }) + .set(projectEnvironment.size); + } + + if (largestFeatureEnvironments.length > 0) { + const featureEnvironment = largestFeatureEnvironments[0]; + largestFeatureEnvironment.reset(); + largestFeatureEnvironment + .labels({ + feature: featureEnvironment.feature, + environment: featureEnvironment.environment, + }) + .set(featureEnvironment.size); + } + enabledMetricsBucketsPreviousDay.reset(); enabledMetricsBucketsPreviousDay.set( stats.previousDayMetricsBucketsCount.enabledCount, diff --git a/src/lib/types/stores.ts b/src/lib/types/stores.ts index aeba11153d..4b982bf1ae 100644 --- a/src/lib/types/stores.ts +++ b/src/lib/types/stores.ts @@ -45,6 +45,7 @@ import { IFeatureLifecycleStore } from '../features/feature-lifecycle/feature-li import { IProjectFlagCreatorsReadModel } from '../features/project/project-flag-creators-read-model.type'; import { IFeatureStrategiesReadModel } from '../features/feature-toggle/types/feature-strategies-read-model-type'; import { IFeatureLifecycleReadModel } from '../features/feature-lifecycle/feature-lifecycle-read-model-type'; +import { ILargestResourcesReadModel } from '../features/metrics/sizes/largest-resources-read-model-type'; export interface IUnleashStores { accessStore: IAccessStore; @@ -94,6 +95,7 @@ export interface IUnleashStores { featureLifecycleStore: IFeatureLifecycleStore; featureStrategiesReadModel: IFeatureStrategiesReadModel; featureLifecycleReadModel: IFeatureLifecycleReadModel; + largestResourcesReadModel: ILargestResourcesReadModel; } export { @@ -142,4 +144,5 @@ export { IProjectFlagCreatorsReadModel, IFeatureStrategiesReadModel, IFeatureLifecycleReadModel, + ILargestResourcesReadModel, }; diff --git a/src/test/fixtures/store.ts b/src/test/fixtures/store.ts index 9a8c72dbc3..e8a44761d2 100644 --- a/src/test/fixtures/store.ts +++ b/src/test/fixtures/store.ts @@ -48,6 +48,7 @@ import { FakeFeatureLifecycleStore } from '../../lib/features/feature-lifecycle/ import { FakeProjectFlagCreatorsReadModel } from '../../lib/features/project/fake-project-flag-creators-read-model'; import { FakeFeatureStrategiesReadModel } from '../../lib/features/feature-toggle/fake-feature-strategies-read-model'; import { FakeFeatureLifecycleReadModel } from '../../lib/features/feature-lifecycle/fake-feature-lifecycle-read-model'; +import { FakeLargestResourcesReadModel } from '../../lib/features/metrics/sizes/fake-largest-resources-read-model'; const db = { select: () => ({ @@ -105,6 +106,7 @@ const createStores: () => IUnleashStores = () => { featureLifecycleStore: new FakeFeatureLifecycleStore(), featureStrategiesReadModel: new FakeFeatureStrategiesReadModel(), featureLifecycleReadModel: new FakeFeatureLifecycleReadModel(), + largestResourcesReadModel: new FakeLargestResourcesReadModel(), }; };