mirror of
https://github.com/Unleash/unleash.git
synced 2025-07-12 13:48:35 +02:00
**BREAKING CHANGE**: DEFAULT_ENV changed from `default` (should not be
used anymore) to `development`
## About the changes
- Only delete default env if the install is fresh new.
- Consider development the new default. The main consequence of this
change is that the default is no longer considered `type=production`
environment but also for frontend tokens due to this assumption:
724c4b78a2/src/lib/schema/api-token-schema.test.ts (L54-L59)
(I believe this is mostly due to the [support for admin
tokens](https://github.com/Unleash/unleash/pull/10080#discussion_r2126871567))
- `feature_toggle_update_total` metric reports `n/a` in environment and
environment type as it's not environment specific
263 lines
7.9 KiB
TypeScript
263 lines
7.9 KiB
TypeScript
import dbInit, {
|
|
type ITestDb,
|
|
} from '../../../test/e2e/helpers/database-init.js';
|
|
import {
|
|
type IUnleashTest,
|
|
setupAppWithAuth,
|
|
} from '../../../test/e2e/helpers/test-helper.js';
|
|
import getLogger from '../../../test/fixtures/no-logger.js';
|
|
import {
|
|
CLIENT_METRICS_ADDED,
|
|
FEATURE_ARCHIVED,
|
|
FEATURE_CREATED,
|
|
FEATURE_REVIVED,
|
|
} from '../../events/index.js';
|
|
import type {
|
|
IEventStore,
|
|
IFeatureLifecycleStore,
|
|
StageName,
|
|
} from '../../types/index.js';
|
|
import type EventEmitter from 'events';
|
|
import type { FeatureLifecycleCompletedSchema } from '../../openapi/index.js';
|
|
import { FeatureLifecycleReadModel } from './feature-lifecycle-read-model.js';
|
|
import type { IFeatureLifecycleReadModel } from './feature-lifecycle-read-model-type.js';
|
|
import { STAGE_ENTERED } from '../../metric-events.js';
|
|
import type ClientInstanceService from '../metrics/instance/instance-service.js';
|
|
|
|
let app: IUnleashTest;
|
|
let db: ITestDb;
|
|
let featureLifecycleStore: IFeatureLifecycleStore;
|
|
let eventStore: IEventStore;
|
|
let eventBus: EventEmitter;
|
|
let featureLifecycleReadModel: IFeatureLifecycleReadModel;
|
|
let clientInstanceService: ClientInstanceService;
|
|
|
|
beforeAll(async () => {
|
|
db = await dbInit('feature_lifecycle', getLogger);
|
|
app = await setupAppWithAuth(
|
|
db.stores,
|
|
{
|
|
experimental: {
|
|
flags: {},
|
|
},
|
|
},
|
|
db.rawDatabase,
|
|
);
|
|
eventStore = db.stores.eventStore;
|
|
eventBus = app.config.eventBus;
|
|
featureLifecycleReadModel = new FeatureLifecycleReadModel(db.rawDatabase);
|
|
featureLifecycleStore = db.stores.featureLifecycleStore;
|
|
clientInstanceService = app.services.clientInstanceService;
|
|
|
|
await app.request
|
|
.post(`/auth/demo/login`)
|
|
.send({
|
|
email: 'user@getunleash.io',
|
|
})
|
|
.expect(200);
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await app.destroy();
|
|
await db.destroy();
|
|
});
|
|
|
|
beforeEach(async () => {
|
|
await clientInstanceService.bulkAdd(); // flush
|
|
await featureLifecycleStore.deleteAll();
|
|
});
|
|
|
|
const getFeatureLifecycle = async (featureName: string, expectedCode = 200) => {
|
|
return app.request
|
|
.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,
|
|
expectedCode = 200,
|
|
) => {
|
|
return app.request
|
|
.post(
|
|
`/api/admin/projects/default/features/${featureName}/lifecycle/complete`,
|
|
)
|
|
.send(status)
|
|
.expect(expectedCode);
|
|
};
|
|
|
|
const uncompleteFeature = async (featureName: string, expectedCode = 200) => {
|
|
return app.request
|
|
.post(
|
|
`/api/admin/projects/default/features/${featureName}/lifecycle/uncomplete`,
|
|
)
|
|
.expect(expectedCode);
|
|
};
|
|
|
|
function reachedStage(feature: string, stage: StageName) {
|
|
return new Promise((resolve) =>
|
|
eventBus.on(STAGE_ENTERED, (event) => {
|
|
if (event.stage === stage && event.feature === feature) {
|
|
resolve(stage);
|
|
}
|
|
}),
|
|
);
|
|
}
|
|
|
|
const expectFeatureStage = async (featureName: string, stage: StageName) => {
|
|
const { body: feature } = await app.getProjectFeatures(
|
|
'default',
|
|
featureName,
|
|
);
|
|
expect(feature.lifecycle).toMatchObject({
|
|
stage,
|
|
enteredStageAt: expect.any(String),
|
|
});
|
|
};
|
|
|
|
const getFeaturesLifecycleCount = async () => {
|
|
return app.request.get(`/api/admin/lifecycle/count`).expect(200);
|
|
};
|
|
|
|
test('should return lifecycle stages', async () => {
|
|
const environment = 'production'; // prod environment moves lifecycle to live stage
|
|
await app.createFeature('my_feature_a');
|
|
await app.enableFeature('my_feature_a', environment);
|
|
eventStore.emit(FEATURE_CREATED, { featureName: 'my_feature_a' });
|
|
await reachedStage('my_feature_a', 'initial');
|
|
await expectFeatureStage('my_feature_a', 'initial');
|
|
eventBus.emit(CLIENT_METRICS_ADDED, [
|
|
{
|
|
featureName: 'my_feature_a',
|
|
environment: environment,
|
|
},
|
|
{
|
|
featureName: 'non_existent_feature',
|
|
environment: environment,
|
|
},
|
|
]);
|
|
|
|
// missing feature
|
|
eventBus.emit(CLIENT_METRICS_ADDED, [
|
|
{
|
|
environment: environment,
|
|
yes: 0,
|
|
no: 0,
|
|
},
|
|
]);
|
|
// non existent env
|
|
eventBus.emit(CLIENT_METRICS_ADDED, [
|
|
{
|
|
featureName: 'my_feature_a',
|
|
environment: 'non-existent',
|
|
},
|
|
]);
|
|
await reachedStage('my_feature_a', 'live');
|
|
await expectFeatureStage('my_feature_a', 'live');
|
|
eventStore.emit(FEATURE_ARCHIVED, { featureName: 'my_feature_a' });
|
|
await reachedStage('my_feature_a', 'archived');
|
|
|
|
const { body } = await getFeatureLifecycle('my_feature_a');
|
|
|
|
expect(body).toEqual([
|
|
{
|
|
stage: 'initial',
|
|
enteredStageAt: expect.any(String),
|
|
},
|
|
{ stage: 'pre-live', enteredStageAt: expect.any(String) },
|
|
{
|
|
stage: 'live',
|
|
enteredStageAt: expect.any(String),
|
|
},
|
|
{
|
|
stage: 'archived',
|
|
enteredStageAt: expect.any(String),
|
|
},
|
|
]);
|
|
await expectFeatureStage('my_feature_a', 'archived');
|
|
|
|
eventStore.emit(FEATURE_REVIVED, { featureName: 'my_feature_a' });
|
|
await reachedStage('my_feature_a', 'initial');
|
|
|
|
const { body: lifecycleCount } = await getFeaturesLifecycleCount();
|
|
expect(lifecycleCount).toEqual({
|
|
initial: 1,
|
|
preLive: 0,
|
|
live: 0,
|
|
completed: 0,
|
|
archived: 0,
|
|
});
|
|
});
|
|
|
|
test('should be able to toggle between completed/uncompleted', async () => {
|
|
await app.createFeature('my_feature_b');
|
|
|
|
await completeFeature('my_feature_b', {
|
|
status: 'kept',
|
|
statusValue: 'variant1',
|
|
});
|
|
const currentStage = await getCurrentStage('my_feature_b');
|
|
expect(currentStage).toMatchObject({ stage: 'completed', status: 'kept' });
|
|
|
|
await expectFeatureStage('my_feature_b', 'completed');
|
|
|
|
await uncompleteFeature('my_feature_b');
|
|
|
|
const { body } = await getFeatureLifecycle('my_feature_b');
|
|
|
|
expect(body).toEqual([]);
|
|
});
|
|
|
|
test('should backfill intialized feature', async () => {
|
|
await app.createFeature('my_feature_c');
|
|
await featureLifecycleStore.delete('my_feature_c');
|
|
|
|
await featureLifecycleStore.backfill();
|
|
|
|
const { body } = await getFeatureLifecycle('my_feature_c');
|
|
expect(body).toEqual([
|
|
{ stage: 'initial', enteredStageAt: expect.any(String) },
|
|
]);
|
|
});
|
|
|
|
test('should backfill archived feature', async () => {
|
|
await app.createFeature('my_feature_d');
|
|
await app.archiveFeature('my_feature_d');
|
|
await featureLifecycleStore.delete('my_feature_d');
|
|
|
|
await featureLifecycleStore.backfill();
|
|
|
|
const { body } = await getFeatureLifecycle('my_feature_d');
|
|
expect(body).toEqual([
|
|
{ stage: 'initial', enteredStageAt: expect.any(String) },
|
|
{ stage: 'archived', enteredStageAt: expect.any(String) },
|
|
]);
|
|
});
|
|
|
|
test('should not backfill for existing lifecycle', async () => {
|
|
const environment = 'production'; // prod environment moves lifecycle to live stage
|
|
await app.createFeature('my_feature_e');
|
|
await app.enableFeature('my_feature_e', environment);
|
|
eventStore.emit(FEATURE_CREATED, { featureName: 'my_feature_e' });
|
|
eventBus.emit(CLIENT_METRICS_ADDED, [
|
|
{
|
|
featureName: 'my_feature_e',
|
|
environment: environment,
|
|
},
|
|
]);
|
|
await reachedStage('my_feature_e', 'live');
|
|
|
|
await featureLifecycleStore.backfill();
|
|
|
|
const { body } = await getFeatureLifecycle('my_feature_e');
|
|
expect(body).toEqual([
|
|
{ stage: 'initial', enteredStageAt: expect.any(String) },
|
|
{ stage: 'pre-live', enteredStageAt: expect.any(String) },
|
|
{ stage: 'live', enteredStageAt: expect.any(String) },
|
|
]);
|
|
});
|