Skip to content

Commit

Permalink
Add feature attribution (#296)
Browse files Browse the repository at this point in the history
Signed-off-by: Jackie Han <[email protected]>
  • Loading branch information
jackiehanyang authored Aug 5, 2022
1 parent 6f736d7 commit d973b8e
Show file tree
Hide file tree
Showing 9 changed files with 239 additions and 40 deletions.
12 changes: 11 additions & 1 deletion public/models/interfaces.ts
Original file line number Diff line number Diff line change
Expand Up @@ -234,14 +234,23 @@ export type AnomalyData = {
plotTime?: number;
entity?: EntityData[];
features?: { [key: string]: FeatureAggregationData };
aggInterval?: string;
contributions?: { [key: string]: FeatureContributionData };
aggInterval?: string;
};

export type FeatureAggregationData = {
data: number;
name?: string;
endTime: number;
startTime: number;
plotTime?: number;
attribution?: number;
expectedValue?: number;
};

export type FeatureContributionData = {
name: string;
attribution: number;
};

export type Anomalies = {
Expand Down Expand Up @@ -279,6 +288,7 @@ export type AnomalySummary = {
minConfidence: number;
maxConfidence: number;
lastAnomalyOccurrence: string;
contributions?: string;
};

export type DateRange = {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -205,6 +205,7 @@ export const FeatureChart = (props: FeatureChartProps) => {
* and thus show different annotations per feature chart (currently all annotations
* shown equally across all enabled feature charts for a given detector).
*/}

{props.feature.featureEnabled ? (
<RectAnnotation
dataValues={flattenData(props.annotations)}
Expand Down Expand Up @@ -257,13 +258,14 @@ export const FeatureChart = (props: FeatureChartProps) => {
}
{featureData.map(
(featureTimeSeries: FeatureAggregationData[], index) => {
const timeSeriesList: any[] = [];
const seriesKey = props.isHCDetector
? `${props.featureDataSeriesName} (${convertToEntityString(
props.entityData[index],
', '
)})`
: props.featureDataSeriesName;
return (
timeSeriesList.push(
<LineSeries
id={seriesKey}
name={seriesKey}
Expand All @@ -278,9 +280,29 @@ export const FeatureChart = (props: FeatureChartProps) => {
yAccessors={[CHART_FIELDS.DATA]}
data={featureTimeSeries}
/>
);
)
if (featureTimeSeries.map(
(item: FeatureAggregationData) => {
if(item.hasOwnProperty('expectedValue')) {
timeSeriesList.push(
<LineSeries
id={"ExpectedValue"}
name={"Expected Value"}
color={"#0475a2"}
xScaleType={ScaleType.Time}
yScaleType={ScaleType.Linear}
xAccessor={CHART_FIELDS.PLOT_TIME}
yAccessors={[CHART_FIELDS.EXPECTED_VALUE]}
data={featureTimeSeries}
/>
)
}
}
))
return timeSeriesList;
}
)}

</Chart>
{showCustomExpression ? (
<CodeModal
Expand Down
135 changes: 106 additions & 29 deletions public/pages/AnomalyCharts/containers/AnomalyDetailsChart.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import {
niceTimeFormatter,
Position,
RectAnnotation,
RectAnnotationDatum,
ScaleType,
Settings,
XYBrushArea,
Expand All @@ -29,8 +30,9 @@ import {
EuiLoadingChart,
EuiStat,
EuiButtonGroup,
EuiText,
} from '@elastic/eui';
import { get } from 'lodash';
import { forEach, get } from 'lodash';
import moment from 'moment';
import React, { useEffect, useState } from 'react';
import { useDispatch, useSelector } from 'react-redux';
Expand All @@ -41,6 +43,7 @@ import {
Monitor,
MonitorAlert,
AnomalyData,
Anomalies,
} from '../../../models/interfaces';
import { AppState } from '../../../redux/reducers';
import { searchAlerts } from '../../../redux/reducers/alerting';
Expand All @@ -52,6 +55,8 @@ import {
getHistoricalAggQuery,
parseHistoricalAggregatedAnomalies,
convertToEntityString,
flattenData,
generateAnomalyAnnotations,
} from '../../utils/anomalyResultUtils';
import { AlertsFlyout } from '../components/AlertsFlyout/AlertsFlyout';
import {
Expand Down Expand Up @@ -155,7 +160,6 @@ export const AnomalyDetailsChart = React.memo(

const taskId = get(props, 'detector.taskId');
const taskState = get(props, 'detector.taskState');

const isRequestingAnomalyResults = useSelector(
(state: AppState) => state.anomalyResults.requesting
);
Expand Down Expand Up @@ -344,6 +348,69 @@ export const AnomalyDetailsChart = React.memo(
zoomRange.endDate,
]);


const customAnomalyContributionTooltip = (details?: string) => {
const anomaly = details ? JSON.parse(details) : undefined;
const contributionData = get(anomaly, `contributions`, [])

const featureData = get(anomaly, `features`, {})
let featureAttributionList = [] as any[];
if (Array.isArray(contributionData)) {
contributionData.map((contribution: any) => {
const featureName = get(get(featureData, contribution.feature_id, ""), "name", "")
const dataString = (contribution.data * 100) + "%"
featureAttributionList.push(
<div>
{featureName}: {dataString} <br />
</div>
)
})
} else {
for (const [, value] of Object.entries(contributionData)) {
featureAttributionList.push(
<div>
{value.name}: {value.attribution} <br />
</div>
)
}
}
return (
<div>
<EuiText size="xs">
<EuiIcon type="list" /> <b>Feature Contribution: </b>
{anomaly ? (
<p>
<hr style={{ color: '#E0E0E0' }}></hr>
{featureAttributionList}
</p>
) : null}
</EuiText>
</div>
);
};


const generateContributionAnomalyAnnotations = (
anomalies: AnomalyData[][]
): any[][] => {
let annotations = [] as any[];
anomalies.forEach((anomalyTimeSeries: AnomalyData[]) => {
annotations.push(
anomalyTimeSeries
.filter((anomaly: AnomalyData) => anomaly.anomalyGrade > 0)
.map((anomaly: AnomalyData) => (
{
coordinates: {
x0: anomaly.startTime,
x1: anomaly.endTime + (anomaly.endTime - anomaly.startTime),
},
details: `${JSON.stringify(anomaly)}`
}))
);
});
return annotations;
};

const isLoading =
props.isLoading || isLoadingAlerts || isRequestingAnomalyResults;
const isInitializingHistorical = taskState === DETECTOR_STATE.INIT;
Expand Down Expand Up @@ -531,7 +598,18 @@ export const AnomalyDetailsChart = React.memo(
}}
/>
)}

<RectAnnotation
dataValues={flattenData(generateContributionAnomalyAnnotations(zoomedAnomalies))}
id="featureAttributionAnnotation"
style={{
stroke: CHART_COLORS.ANOMALY_GRADE_COLOR,
strokeWidth: 1,
opacity: 0.1,
fill: CHART_COLORS.ANOMALY_GRADE_COLOR,
}}
renderTooltip={customAnomalyContributionTooltip}
/>

{alertAnnotations ? (
<LineAnnotation
id="alertAnnotation"
Expand Down Expand Up @@ -603,32 +681,31 @@ export const AnomalyDetailsChart = React.memo(
get(anomalySeries, '0.entity', []),
', '
)})`
: props.anomalyGradeSeriesName;

return (
<LineSeries
id={seriesKey}
name={seriesKey}
color={
multipleTimeSeries
? ENTITY_COLORS[index]
: CHART_COLORS.ANOMALY_GRADE_COLOR
}
data={anomalySeries}
xScaleType={
showAggregateResults
? ScaleType.Ordinal
: ScaleType.Time
}
yScaleType={ScaleType.Linear}
xAccessor={
showAggregateResults
? CHART_FIELDS.AGG_INTERVAL
: CHART_FIELDS.PLOT_TIME
}
yAccessors={[CHART_FIELDS.ANOMALY_GRADE]}
/>
);
: props.anomalyGradeSeriesName;
return (
<LineSeries
id={seriesKey}
name={seriesKey}
color={
multipleTimeSeries
? ENTITY_COLORS[index]
: CHART_COLORS.ANOMALY_GRADE_COLOR
}
data={anomalySeries}
xScaleType={
showAggregateResults
? ScaleType.Ordinal
: ScaleType.Time
}
yScaleType={ScaleType.Linear}
xAccessor={
showAggregateResults
? CHART_FIELDS.AGG_INTERVAL
: CHART_FIELDS.PLOT_TIME
}
yAccessors={[CHART_FIELDS.ANOMALY_GRADE]}
/>
)
}
)}
</Chart>
Expand Down
4 changes: 3 additions & 1 deletion public/pages/AnomalyCharts/utils/anomalyChartUtils.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -640,7 +640,9 @@ export const getFeatureBreakdownWording = (
return isNotSample ? 'Feature breakdown' : 'Sample feature breakdown';
};

export const getFeatureDataWording = (isNotSample: boolean | undefined) => {
export const getFeatureDataWording = (
isNotSample: boolean | undefined
) => {
return isNotSample ? 'Feature output' : 'Sample feature output';
};

Expand Down
4 changes: 3 additions & 1 deletion public/pages/AnomalyCharts/utils/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,11 +24,13 @@ export enum CHART_FIELDS {
CONFIDENCE = 'confidence',
DATA = 'data',
AGG_INTERVAL = 'aggInterval',
EXPECTED_VALUE = 'expectedValue'
}

export enum CHART_COLORS {
ANOMALY_GRADE_COLOR = '#D13212',
FEATURE_DATA_COLOR = '#16191F',
FEATURE_COLOR = '#fcd529',
CONFIDENCE_COLOR = '#017F75',
LIGHT_BACKGROUND = '#FFFFFF',
DARK_BACKGROUND = '#1D1E24',
Expand Down Expand Up @@ -89,7 +91,7 @@ export const DEFAULT_ANOMALY_SUMMARY = {
maxAnomalyGrade: 0,
minConfidence: 0,
maxConfidence: 0,
lastAnomalyOccurrence: '-',
lastAnomalyOccurrence: '-'
};

export const HEATMAP_CHART_Y_AXIS_WIDTH = 30;
12 changes: 6 additions & 6 deletions public/pages/DetectorResults/containers/AnomalyHistory.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -136,7 +136,7 @@ export const AnomalyHistory = (props: AnomalyHistoryProps) => {
endDate: initialEndDate,
});
const [selectedTabId, setSelectedTabId] = useState<string>(
ANOMALY_HISTORY_TABS.ANOMALY_OCCURRENCE
ANOMALY_HISTORY_TABS.FEATURE_BREAKDOWN
);
const [isLoadingAnomalyResults, setIsLoadingAnomalyResults] =
useState<boolean>(false);
Expand Down Expand Up @@ -425,7 +425,6 @@ export const AnomalyHistory = (props: AnomalyHistoryProps) => {
setIsLoadingAnomalyResults(false);
errorFetchingResults = true;
});

const rawAnomaliesData = get(detectorResultResponse, 'response', []);
const rawAnomaliesResult = {
anomalies: get(rawAnomaliesData, 'results', []),
Expand Down Expand Up @@ -610,6 +609,7 @@ export const AnomalyHistory = (props: AnomalyHistoryProps) => {
fetchBucketizedEntityAnomalyData(entityLists);
} else {
fetchAllEntityAnomalyData(dateRange, entityLists);
// no visit
setBucketizedAnomalyResults(undefined);
}
} catch (err) {
Expand Down Expand Up @@ -807,13 +807,13 @@ export const AnomalyHistory = (props: AnomalyHistoryProps) => {

const tabs = [
{
id: ANOMALY_HISTORY_TABS.ANOMALY_OCCURRENCE,
name: 'Anomaly occurrences',
id: ANOMALY_HISTORY_TABS.FEATURE_BREAKDOWN,
name: 'Feature breakdown',
disabled: false,
},
{
id: ANOMALY_HISTORY_TABS.FEATURE_BREAKDOWN,
name: 'Feature breakdown',
id: ANOMALY_HISTORY_TABS.ANOMALY_OCCURRENCE,
name: 'Anomaly occurrences',
disabled: false,
},
];
Expand Down
Loading

0 comments on commit d973b8e

Please sign in to comment.