diff --git a/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureLifecycle/populateCurrentStage.ts b/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureLifecycle/populateCurrentStage.ts index 6798f91433..d62dfd7ccb 100644 --- a/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureLifecycle/populateCurrentStage.ts +++ b/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureLifecycle/populateCurrentStage.ts @@ -43,7 +43,10 @@ export const populateCurrentStage = ( case 'completed': return { name: 'completed', - status: 'kept', + status: + feature.lifecycle.status === 'discarded' + ? 'discarded' + : 'kept', environments: getFilteredEnvironments(() => true), enteredStageAt, }; diff --git a/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureOverviewMetaData/FeatureOverviewMetaData.tsx b/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureOverviewMetaData/FeatureOverviewMetaData.tsx index c4904320ac..7d28eedf19 100644 --- a/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureOverviewMetaData/FeatureOverviewMetaData.tsx +++ b/frontend/src/component/feature/FeatureView/FeatureOverview/FeatureOverviewMetaData/FeatureOverviewMetaData.tsx @@ -137,7 +137,10 @@ const FeatureOverviewMetaData = () => { {project} Lifecycle: diff --git a/frontend/src/interfaces/featureToggle.ts b/frontend/src/interfaces/featureToggle.ts index 4d5edd72c0..5d3d3d3821 100644 --- a/frontend/src/interfaces/featureToggle.ts +++ b/frontend/src/interfaces/featureToggle.ts @@ -34,6 +34,7 @@ export type ILastSeenEnvironments = Pick< export type Lifecycle = { stage: 'initial' | 'pre-live' | 'live' | 'completed' | 'archived'; + status?: string; enteredStageAt: string; }; diff --git a/src/lib/features/feature-lifecycle/fake-feature-lifecycle-store.ts b/src/lib/features/feature-lifecycle/fake-feature-lifecycle-store.ts index 74ff64cf6a..d15b1f9f49 100644 --- a/src/lib/features/feature-lifecycle/fake-feature-lifecycle-store.ts +++ b/src/lib/features/feature-lifecycle/fake-feature-lifecycle-store.ts @@ -27,6 +27,9 @@ export class FakeFeatureLifecycleStore implements IFeatureLifecycleStore { ...existingStages, { stage: featureLifecycleStage.stage, + ...(featureLifecycleStage.status + ? { status: featureLifecycleStage.status } + : {}), enteredStageAt: new Date(), }, ]; diff --git a/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.ts b/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.ts index 1f6e5deee8..a0dff07d00 100644 --- a/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.ts +++ b/src/lib/features/feature-lifecycle/feature-lifecycle-read-model.ts @@ -6,6 +6,7 @@ import type { IFeatureLifecycleStage, StageName } from '../../types'; type DBType = { feature: string; stage: StageName; + status: string | null; created_at: Date; }; @@ -23,8 +24,9 @@ export class FeatureLifecycleReadModel implements IFeatureLifecycleReadModel { .where({ feature }) .orderBy('created_at', 'asc'); - const stages = results.map(({ stage, created_at }: DBType) => ({ + const stages = results.map(({ stage, status, created_at }: DBType) => ({ stage, + ...(status ? { status } : {}), enteredStageAt: created_at, })); diff --git a/src/lib/features/feature-lifecycle/feature-lifecycle-store.ts b/src/lib/features/feature-lifecycle/feature-lifecycle-store.ts index 925e47af08..89076e7f30 100644 --- a/src/lib/features/feature-lifecycle/feature-lifecycle-store.ts +++ b/src/lib/features/feature-lifecycle/feature-lifecycle-store.ts @@ -10,8 +10,8 @@ import type { StageName } from '../../types'; type DBType = { stage: StageName; created_at: string; - status?: string; - status_value?: string; + status: string | null; + status_value: string | null; }; type DBProjectType = DBType & { @@ -64,8 +64,9 @@ export class FeatureLifecycleStore implements IFeatureLifecycleStore { .where({ feature }) .orderBy('created_at', 'asc'); - return results.map(({ stage, created_at }: DBType) => ({ + return results.map(({ stage, status, created_at }: DBType) => ({ stage, + ...(status ? { status } : {}), enteredStageAt: new Date(created_at), })); } diff --git a/src/lib/features/feature-lifecycle/feature-lifecycle.e2e.test.ts b/src/lib/features/feature-lifecycle/feature-lifecycle.e2e.test.ts index ec05cbb0ed..65e207958a 100644 --- a/src/lib/features/feature-lifecycle/feature-lifecycle.e2e.test.ts +++ b/src/lib/features/feature-lifecycle/feature-lifecycle.e2e.test.ts @@ -18,12 +18,15 @@ import { STAGE_ENTERED, } from './feature-lifecycle-service'; import type { FeatureLifecycleCompletedSchema } from '../../openapi'; +import { FeatureLifecycleReadModel } from './feature-lifecycle-read-model'; +import type { IFeatureLifecycleReadModel } from './feature-lifecycle-read-model-type'; let app: IUnleashTest; let db: ITestDb; let featureLifecycleService: FeatureLifecycleService; let eventStore: IEventStore; let eventBus: EventEmitter; +let featureLifecycleReadModel: IFeatureLifecycleReadModel; beforeAll(async () => { db = await dbInit('feature_lifecycle', getLogger); @@ -41,6 +44,7 @@ beforeAll(async () => { eventStore = db.stores.eventStore; eventBus = app.config.eventBus; featureLifecycleService = app.services.featureLifecycleService; + featureLifecycleReadModel = new FeatureLifecycleReadModel(db.rawDatabase); await app.request .post(`/auth/demo/login`) @@ -62,6 +66,11 @@ const getFeatureLifecycle = async (featureName: string, expectedCode = 200) => { .get(`/api/admin/projects/default/features/${featureName}/lifecycle`) .expect(expectedCode); }; + +const getCurrentStage = async (featureName: string) => { + return featureLifecycleReadModel.findCurrentStage(featureName); +}; + const completeFeature = async ( featureName: string, status: FeatureLifecycleCompletedSchema, @@ -166,6 +175,8 @@ test('should be able to toggle between completed/uncompleted', async () => { status: 'kept', statusValue: 'variant1', }); + const currentStage = await getCurrentStage('my_feature_b'); + expect(currentStage).toMatchObject({ stage: 'completed', status: 'kept' }); await expectFeatureStage('my_feature_b', 'completed'); diff --git a/src/lib/features/feature-lifecycle/get-current-stage.test.ts b/src/lib/features/feature-lifecycle/get-current-stage.test.ts index d4e9646448..31bc61154a 100644 --- a/src/lib/features/feature-lifecycle/get-current-stage.test.ts +++ b/src/lib/features/feature-lifecycle/get-current-stage.test.ts @@ -12,6 +12,7 @@ describe('getCurrentStage', () => { }, { stage: 'completed', + status: 'kept', enteredStageAt: irrelevantDate, }, { diff --git a/src/lib/features/feature-search/feature-search-store.ts b/src/lib/features/feature-search/feature-search-store.ts index 40cffd7a3f..2b7aa9a955 100644 --- a/src/lib/features/feature-search/feature-search-store.ts +++ b/src/lib/features/feature-search/feature-search-store.ts @@ -79,6 +79,7 @@ class FeatureSearchStore implements IFeatureSearchStore { .select( 'feature as stage_feature', 'stage as latest_stage', + 'status as stage_status', 'created_at as entered_stage_at', ) .distinctOn('stage_feature') @@ -416,6 +417,9 @@ class FeatureSearchStore implements IFeatureSearchStore { entry.lifecycle = row.latest_stage ? { stage: row.latest_stage, + ...(row.stage_status + ? { status: row.stage_status } + : {}), enteredStageAt: row.entered_stage_at, } : undefined; diff --git a/src/lib/features/feature-search/feature.search.e2e.test.ts b/src/lib/features/feature-search/feature.search.e2e.test.ts index c303ce0609..33d78ba3fc 100644 --- a/src/lib/features/feature-search/feature.search.e2e.test.ts +++ b/src/lib/features/feature-search/feature.search.e2e.test.ts @@ -962,7 +962,7 @@ test('should return environment usage metrics and lifecycle', async () => { { feature: 'my_feature_b', stage: 'initial' }, ]); await stores.featureLifecycleStore.insert([ - { feature: 'my_feature_b', stage: 'pre-live' }, + { feature: 'my_feature_b', stage: 'completed', status: 'discarded' }, ]); const { body } = await searchFeatures({ @@ -972,7 +972,7 @@ test('should return environment usage metrics and lifecycle', async () => { features: [ { name: 'my_feature_b', - lifecycle: { stage: 'pre-live' }, + lifecycle: { stage: 'completed', status: 'discarded' }, environments: [ { name: 'default', diff --git a/src/lib/openapi/spec/feature-lifecycle-schema.ts b/src/lib/openapi/spec/feature-lifecycle-schema.ts index 7586fc04e9..88878f8a75 100644 --- a/src/lib/openapi/spec/feature-lifecycle-schema.ts +++ b/src/lib/openapi/spec/feature-lifecycle-schema.ts @@ -16,6 +16,12 @@ export const featureLifecycleSchema = { description: 'The name of the lifecycle stage that got recorded for a given feature', }, + status: { + type: 'string', + example: 'kept', + description: + 'The name of the detailed status of a given stage. E.g. completed stage can be kept or discarded.', + }, enteredStageAt: { type: 'string', format: 'date-time', diff --git a/src/lib/openapi/spec/feature-search-response-schema.ts b/src/lib/openapi/spec/feature-search-response-schema.ts index b517e331b9..8c125a6411 100644 --- a/src/lib/openapi/spec/feature-search-response-schema.ts +++ b/src/lib/openapi/spec/feature-search-response-schema.ts @@ -161,6 +161,13 @@ export const featureSearchResponseSchema = { ], example: 'initial', }, + status: { + type: 'string', + nullable: true, + example: 'kept', + description: + 'The name of the detailed status of a given stage. E.g. completed stage can be kept or discarded.', + }, enteredStageAt: { description: 'When the feature entered this stage', type: 'string', diff --git a/src/lib/types/model.ts b/src/lib/types/model.ts index 7d08845447..7fcf38e636 100644 --- a/src/lib/types/model.ts +++ b/src/lib/types/model.ts @@ -163,6 +163,7 @@ export type StageName = export interface IFeatureLifecycleStage { stage: StageName; enteredStageAt: Date; + status?: string; } export type IProjectLifecycleStageDuration = {