1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-25 00:07:47 +01:00
unleash.unleash/frontend/src/component/common/util.ts

187 lines
5.3 KiB
TypeScript

import { weightTypes } from 'constants/weightTypes';
import type { IUiConfig } from 'interfaces/uiConfig';
import type { INavigationMenuItem } from 'interfaces/route';
import type { IFeatureVariant } from 'interfaces/featureToggle';
import { format, isValid, parseISO } from 'date-fns';
import type { IFeatureVariantEdit } from 'component/feature/FeatureView/FeatureVariants/FeatureEnvironmentVariants/EnvironmentVariantsModal/EnvironmentVariantsModal';
import { formatDateYMD } from '../../utils/formatDate';
/**
* Handle feature flags and configuration for different plans.
*/
export const filterByConfig =
(config: IUiConfig) => (r: INavigationMenuItem) => {
if (r.flag) {
// Check if the route's `flag` is enabled in IUiConfig.flags.
const flags = config.flags as unknown as Record<string, boolean>;
return Boolean(flags[r.flag]);
}
if (r.notFlag) {
const flags = config.flags as unknown as Record<string, boolean>;
return !(flags[r.notFlag] === true);
}
if (r.configFlag) {
// Check if the route's `configFlag` is enabled in IUiConfig.
return Boolean(config[r.configFlag]);
}
const isOss = !config?.versionInfo?.current?.enterprise;
if (isOss && r.enterprise) {
return false;
}
return true;
};
export const scrollToTop = () => {
window.scrollTo(0, 0);
};
export const mapRouteLink = (route: INavigationMenuItem) => ({
...route,
path: route.path.replace('/*', ''),
route: route.path,
});
export const trim = (value: string): string => {
if (value?.trim) {
return value.trim();
} else {
return value;
}
};
export const getLocalizedDateString = (
value: Date | string | null | undefined,
locale: string,
) => {
const date = value
? value instanceof Date
? formatDateYMD(value, locale)
: formatDateYMD(parseISO(value), locale)
: undefined;
return date;
};
export const parseDateValue = (value: string) => {
const date = new Date(value);
return `${format(date, 'yyyy-MM-dd')}T${format(date, 'HH:mm')}`;
};
export const parseValidDate = (value: string): Date | undefined => {
const parsed = new Date(value);
if (isValid(parsed)) {
return parsed;
}
};
export const calculateVariantWeight = (weight: number) => {
return weight / 10.0;
};
export function updateWeight(variants: IFeatureVariant[], totalWeight: number) {
if (variants.length === 0) {
return [];
}
const variantMetadata = variants.reduce(
({ remainingPercentage, variableVariantCount }, variant) => {
if (variant.weight && variant.weightType === weightTypes.FIX) {
remainingPercentage -= Number(variant.weight);
} else {
variableVariantCount += 1;
}
return {
remainingPercentage,
variableVariantCount,
};
},
{ remainingPercentage: totalWeight, variableVariantCount: 0 },
);
const { remainingPercentage, variableVariantCount } = variantMetadata;
if (remainingPercentage < 0) {
throw new Error('The traffic distribution total must equal 100%');
}
if (!variableVariantCount) {
throw new Error('There must be at least one variable variant');
}
const percentage = Number.parseInt(
String(remainingPercentage / variableVariantCount),
);
return variants.map((variant) => {
if (variant.weightType !== weightTypes.FIX) {
variant.weight = percentage;
}
return variant;
});
}
export function updateWeightEdit(
variants: IFeatureVariantEdit[],
totalWeight: number,
) {
if (variants.length === 0) {
return [];
}
let { remainingPercentage, variableVariantCount } = variants.reduce(
({ remainingPercentage, variableVariantCount }, variant) => {
if (variant.weight && variant.weightType === weightTypes.FIX) {
remainingPercentage -= Number(variant.weight);
}
if (variant.weightType === weightTypes.VARIABLE) {
variableVariantCount += 1;
}
return {
remainingPercentage,
variableVariantCount,
};
},
{ remainingPercentage: totalWeight, variableVariantCount: 0 },
);
const getPercentage = () =>
Math.max(Math.round(remainingPercentage / variableVariantCount), 0);
return variants.map((variant) => {
if (variant.weightType !== weightTypes.FIX) {
const percentage = getPercentage(); // round "as we go" - clean best effort approach
remainingPercentage -= percentage;
variableVariantCount -= 1;
variant.weight = percentage;
}
return variant;
});
}
export const modalStyles = {
overlay: {
position: 'absolute',
top: 0,
left: 0,
right: 0,
bottom: 0,
backgroundColor: 'rgba(0, 0, 0, 0.25)',
zIndex: 5,
},
content: {
width: '500px',
maxWidth: '90%',
margin: '0',
top: '50%',
left: '50%',
right: 'auto',
bottom: 'auto',
transform: 'translate(-50%, -50%)',
},
};