1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-04-06 01:15:28 +02:00

fix: update potentially-stale status dynamically (#4905)

Fixes 2 bugs:

- project-health-service keeping the feature types as an instance
variable and only updating it once was preventing real calculation to
happen if the lifetime value changed for a feature toggle type
- the ui was reading from a predefined map for the lifetime values so
they would never reflect the BE change

Closes #
[SR-66](https://linear.app/unleash/issue/SR-66/slack-question-around-potentially-stale-and-its-uses)

<img width="1680" alt="Screenshot 2023-10-02 at 14 37 17"
src="https://github.com/Unleash/unleash/assets/104830839/7bee8d4a-9054-4214-a1a2-11ad8169c3d5">
<img width="1660" alt="Screenshot 2023-10-02 at 14 37 06"
src="https://github.com/Unleash/unleash/assets/104830839/23bf55c7-a380-4423-a732-205ad81d5c3c">

---------

Signed-off-by: andreas-unleash <andreas@getunleash.ai>
This commit is contained in:
andreas-unleash 2023-10-04 12:47:16 +03:00 committed by GitHub
parent bd8b54b5bd
commit b07c032d56
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 50 additions and 45 deletions

View File

@ -1,14 +1,23 @@
import { IFeatureToggleListItem } from 'interfaces/featureToggle'; import { IFeatureToggleListItem } from 'interfaces/featureToggle';
import { PERMISSION, KILLSWITCH } from 'constants/featureToggleTypes'; import { KILLSWITCH, PERMISSION } from 'constants/featureToggleTypes';
import { getDiffInDays, expired, toggleExpiryByTypeMap } from '../utils'; import { expired, getDiffInDays } from '../utils';
import { subDays, parseISO } from 'date-fns'; import { parseISO, subDays } from 'date-fns';
import { FeatureTypeSchema } from 'openapi';
export const formatExpiredAt = ( export const formatExpiredAt = (
feature: IFeatureToggleListItem, feature: IFeatureToggleListItem,
featureTypes: FeatureTypeSchema[],
): string | undefined => { ): string | undefined => {
const { type, createdAt } = feature; const { type, createdAt } = feature;
if (type === KILLSWITCH || type === PERMISSION) { const featureType = featureTypes.find(
(featureType) => featureType.name === type,
);
if (
featureType &&
(featureType.name === KILLSWITCH || featureType.name === PERMISSION)
) {
return; return;
} }
@ -16,8 +25,8 @@ export const formatExpiredAt = (
const now = new Date(); const now = new Date();
const diff = getDiffInDays(date, now); const diff = getDiffInDays(date, now);
if (expired(diff, type)) { if (featureType && expired(diff, featureType)) {
const result = diff - toggleExpiryByTypeMap[type]; const result = diff - (featureType?.lifetimeDays?.valueOf() || 0);
return subDays(now, result).toISOString(); return subDays(now, result).toISOString();
} }

View File

@ -1,19 +1,30 @@
import { IFeatureToggleListItem } from 'interfaces/featureToggle'; import { IFeatureToggleListItem } from 'interfaces/featureToggle';
import { getDiffInDays, expired } from '../utils'; import { expired, getDiffInDays } from '../utils';
import { PERMISSION, KILLSWITCH } from 'constants/featureToggleTypes'; import { KILLSWITCH, PERMISSION } from 'constants/featureToggleTypes';
import { parseISO } from 'date-fns'; import { parseISO } from 'date-fns';
import { FeatureTypeSchema } from 'openapi';
export type ReportingStatus = 'potentially-stale' | 'healthy'; export type ReportingStatus = 'potentially-stale' | 'healthy';
export const formatStatus = ( export const formatStatus = (
feature: IFeatureToggleListItem, feature: IFeatureToggleListItem,
featureTypes: FeatureTypeSchema[],
): ReportingStatus => { ): ReportingStatus => {
const { type, createdAt } = feature; const { type, createdAt } = feature;
const featureType = featureTypes.find(
(featureType) => featureType.name === type,
);
const date = parseISO(createdAt); const date = parseISO(createdAt);
const now = new Date(); const now = new Date();
const diff = getDiffInDays(date, now); const diff = getDiffInDays(date, now);
if (expired(diff, type) && type !== KILLSWITCH && type !== PERMISSION) { if (
featureType &&
expired(diff, featureType) &&
type !== KILLSWITCH &&
type !== PERMISSION
) {
return 'potentially-stale'; return 'potentially-stale';
} }

View File

@ -9,10 +9,10 @@ import { SearchHighlightProvider } from 'component/common/Table/SearchHighlightC
import { PageHeader } from 'component/common/PageHeader/PageHeader'; import { PageHeader } from 'component/common/PageHeader/PageHeader';
import { sortTypes } from 'utils/sortTypes'; import { sortTypes } from 'utils/sortTypes';
import { import {
useSortBy,
useGlobalFilter,
useTable,
useFlexLayout, useFlexLayout,
useGlobalFilter,
useSortBy,
useTable,
} from 'react-table'; } from 'react-table';
import { useMediaQuery, useTheme } from '@mui/material'; import { useMediaQuery, useTheme } from '@mui/material';
import { FeatureSeenCell } from 'component/common/Table/cells/FeatureSeenCell/FeatureSeenCell'; import { FeatureSeenCell } from 'component/common/Table/cells/FeatureSeenCell/FeatureSeenCell';
@ -29,6 +29,7 @@ import { formatExpiredAt } from './ReportExpiredCell/formatExpiredAt';
import { useConditionallyHiddenColumns } from 'hooks/useConditionallyHiddenColumns'; import { useConditionallyHiddenColumns } from 'hooks/useConditionallyHiddenColumns';
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig'; import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
import { FeatureEnvironmentSeenCell } from 'component/common/Table/cells/FeatureSeenCell/FeatureEnvironmentSeenCell'; import { FeatureEnvironmentSeenCell } from 'component/common/Table/cells/FeatureSeenCell/FeatureEnvironmentSeenCell';
import useFeatureTypes from 'hooks/api/getters/useFeatureTypes/useFeatureTypes';
interface IReportTableProps { interface IReportTableProps {
projectId: string; projectId: string;
@ -56,6 +57,7 @@ export const ReportTable = ({ projectId, features }: IReportTableProps) => {
const showEnvironmentLastSeen = Boolean( const showEnvironmentLastSeen = Boolean(
uiConfig.flags.lastSeenByEnvironment, uiConfig.flags.lastSeenByEnvironment,
); );
const { featureTypes } = useFeatureTypes();
const data: IReportTableRow[] = useMemo<IReportTableRow[]>( const data: IReportTableRow[] = useMemo<IReportTableRow[]>(
() => () =>
@ -65,10 +67,10 @@ export const ReportTable = ({ projectId, features }: IReportTableProps) => {
type: report.type, type: report.type,
stale: report.stale, stale: report.stale,
environments: report.environments, environments: report.environments,
status: formatStatus(report), status: formatStatus(report, featureTypes),
lastSeenAt: report.lastSeenAt, lastSeenAt: report.lastSeenAt,
createdAt: report.createdAt, createdAt: report.createdAt,
expiredAt: formatExpiredAt(report), expiredAt: formatExpiredAt(report, featureTypes),
})), })),
[projectId, features], [projectId, features],
); );

View File

@ -1,19 +1,12 @@
import differenceInDays from 'date-fns/differenceInDays'; import differenceInDays from 'date-fns/differenceInDays';
import { EXPERIMENT, OPERATIONAL, RELEASE } from 'constants/featureToggleTypes'; import { FeatureTypeSchema } from 'openapi';
const FORTY_DAYS = 40;
const SEVEN_DAYS = 7;
export const toggleExpiryByTypeMap: Record<string, number> = {
[EXPERIMENT]: FORTY_DAYS,
[RELEASE]: FORTY_DAYS,
[OPERATIONAL]: SEVEN_DAYS,
};
export const getDiffInDays = (date: Date, now: Date) => { export const getDiffInDays = (date: Date, now: Date) => {
return Math.abs(differenceInDays(date, now)); return Math.abs(differenceInDays(date, now));
}; };
export const expired = (diff: number, type: string) => { export const expired = (diff: number, type: FeatureTypeSchema) => {
return diff >= toggleExpiryByTypeMap[type]; if (type.lifetimeDays) return diff >= type?.lifetimeDays?.valueOf();
return false;
}; };

View File

@ -1,8 +1,8 @@
import useSWR, { mutate, SWRConfiguration } from 'swr'; import useSWR, { mutate, SWRConfiguration } from 'swr';
import { useState, useEffect } from 'react'; import { useEffect, useState } from 'react';
import { formatApiPath } from 'utils/formatPath'; import { formatApiPath } from 'utils/formatPath';
import { IFeatureType } from 'interfaces/featureTypes';
import handleErrorResponses from '../httpErrorResponseHandler'; import handleErrorResponses from '../httpErrorResponseHandler';
import { FeatureTypeSchema } from '../../../../openapi';
const useFeatureTypes = (options: SWRConfiguration = {}) => { const useFeatureTypes = (options: SWRConfiguration = {}) => {
const fetcher = async () => { const fetcher = async () => {
@ -27,7 +27,7 @@ const useFeatureTypes = (options: SWRConfiguration = {}) => {
}, [data, error]); }, [data, error]);
return { return {
featureTypes: (data?.types as IFeatureType[]) || [], featureTypes: (data?.types as FeatureTypeSchema[]) || [],
error, error,
loading, loading,
refetch, refetch,

View File

@ -3,15 +3,12 @@ import { IUnleashConfig } from '../types/option';
import { Logger } from '../logger'; import { Logger } from '../logger';
import type { IProject, IProjectHealthReport } from '../types/model'; import type { IProject, IProjectHealthReport } from '../types/model';
import type { IFeatureToggleStore } from '../types/stores/feature-toggle-store'; import type { IFeatureToggleStore } from '../types/stores/feature-toggle-store';
import type { import type { IFeatureTypeStore } from '../types/stores/feature-type-store';
IFeatureType,
IFeatureTypeStore,
} from '../types/stores/feature-type-store';
import type { IProjectStore } from '../types/stores/project-store'; import type { IProjectStore } from '../types/stores/project-store';
import ProjectService from './project-service'; import ProjectService from './project-service';
import { import {
calculateProjectHealth,
calculateHealthRating, calculateHealthRating,
calculateProjectHealth,
} from '../domain/project-health/project-health'; } from '../domain/project-health/project-health';
export default class ProjectHealthService { export default class ProjectHealthService {
@ -23,8 +20,6 @@ export default class ProjectHealthService {
private featureToggleStore: IFeatureToggleStore; private featureToggleStore: IFeatureToggleStore;
private featureTypes: IFeatureType[];
private projectService: ProjectService; private projectService: ProjectService;
constructor( constructor(
@ -43,7 +38,6 @@ export default class ProjectHealthService {
this.projectStore = projectStore; this.projectStore = projectStore;
this.featureTypeStore = featureTypeStore; this.featureTypeStore = featureTypeStore;
this.featureToggleStore = featureToggleStore; this.featureToggleStore = featureToggleStore;
this.featureTypes = [];
this.projectService = projectService; this.projectService = projectService;
} }
@ -51,9 +45,7 @@ export default class ProjectHealthService {
async getProjectHealthReport( async getProjectHealthReport(
projectId: string, projectId: string,
): Promise<IProjectHealthReport> { ): Promise<IProjectHealthReport> {
if (this.featureTypes.length === 0) { const featureTypes = await this.featureTypeStore.getAll();
this.featureTypes = await this.featureTypeStore.getAll();
}
const overview = await this.projectService.getProjectOverview( const overview = await this.projectService.getProjectOverview(
projectId, projectId,
@ -63,7 +55,7 @@ export default class ProjectHealthService {
const healthRating = calculateProjectHealth( const healthRating = calculateProjectHealth(
overview.features, overview.features,
this.featureTypes, featureTypes,
); );
return { return {
@ -73,16 +65,14 @@ export default class ProjectHealthService {
} }
async calculateHealthRating(project: IProject): Promise<number> { async calculateHealthRating(project: IProject): Promise<number> {
if (this.featureTypes.length === 0) { const featureTypes = await this.featureTypeStore.getAll();
this.featureTypes = await this.featureTypeStore.getAll();
}
const toggles = await this.featureToggleStore.getAll({ const toggles = await this.featureToggleStore.getAll({
project: project.id, project: project.id,
archived: false, archived: false,
}); });
return calculateHealthRating(toggles, this.featureTypes); return calculateHealthRating(toggles, featureTypes);
} }
async setHealthRating(): Promise<void> { async setHealthRating(): Promise<void> {