@spinnaker/kayenta 0.0.0-2025.1-0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (289) hide show
  1. package/.editorconfig +9 -0
  2. package/.eslintrc.js +1 -0
  3. package/.huskyrc +5 -0
  4. package/.lintstagedrc.json +4 -0
  5. package/.prettierignore +4 -0
  6. package/.prettierrc.js +1 -0
  7. package/LICENSE.txt +203 -0
  8. package/README.md +81 -0
  9. package/__mocks__/styleMock.js +1 -0
  10. package/__mocks__/version.json +4 -0
  11. package/babel.config.js +3 -0
  12. package/build.gradle +67 -0
  13. package/build_scripts/checkLicenses.js +79 -0
  14. package/gradle.properties +0 -0
  15. package/jest.config.js +204 -0
  16. package/jest.setup.js +5 -0
  17. package/package.json +166 -0
  18. package/rollup-plugin-angularjs-template-loader.js +82 -0
  19. package/rollup.config.js +30 -0
  20. package/src/index.ts +2 -0
  21. package/src/kayenta/actions/creators.ts +163 -0
  22. package/src/kayenta/actions/index.ts +98 -0
  23. package/src/kayenta/canary.dataSource.bridge.ts +53 -0
  24. package/src/kayenta/canary.dataSource.stub.ts +64 -0
  25. package/src/kayenta/canary.help.ts +136 -0
  26. package/src/kayenta/canary.less +168 -0
  27. package/src/kayenta/canary.settings.ts +26 -0
  28. package/src/kayenta/canary.tsx +67 -0
  29. package/src/kayenta/components/canaryScore.component.less +77 -0
  30. package/src/kayenta/components/canaryScore.component.ts +12 -0
  31. package/src/kayenta/components/canaryScore.tsx +63 -0
  32. package/src/kayenta/components/canaryScores.component.ts +20 -0
  33. package/src/kayenta/components/canaryScores.less +22 -0
  34. package/src/kayenta/components/canaryScores.tsx +163 -0
  35. package/src/kayenta/components/loadStates.tsx +52 -0
  36. package/src/kayenta/domain/ICanaryConfig.ts +57 -0
  37. package/src/kayenta/domain/ICanaryConfigSummary.ts +7 -0
  38. package/src/kayenta/domain/ICanaryConfigUpdateResponse.ts +3 -0
  39. package/src/kayenta/domain/ICanaryExecutionStatusResult.ts +72 -0
  40. package/src/kayenta/domain/ICanaryJudgeResult.ts +51 -0
  41. package/src/kayenta/domain/ICanaryJudgeResultSummary.ts +5 -0
  42. package/src/kayenta/domain/ICanaryScoreThresholds.ts +4 -0
  43. package/src/kayenta/domain/IJudge.ts +5 -0
  44. package/src/kayenta/domain/IKayentaAccount.ts +14 -0
  45. package/src/kayenta/domain/IKayentaStageConfig.ts +58 -0
  46. package/src/kayenta/domain/IMetricSetPair.ts +17 -0
  47. package/src/kayenta/domain/IMetricsServiceMetadata.ts +2 -0
  48. package/src/kayenta/domain/ISetupCanaryStage.ts +11 -0
  49. package/src/kayenta/domain/MetricClassificationLabel.ts +8 -0
  50. package/src/kayenta/domain/ScoreClassificationLabel.ts +7 -0
  51. package/src/kayenta/domain/index.ts +15 -0
  52. package/src/kayenta/edit/changeMetricGroupModal.tsx +107 -0
  53. package/src/kayenta/edit/configDetail.tsx +31 -0
  54. package/src/kayenta/edit/configDetailActionButtons.tsx +24 -0
  55. package/src/kayenta/edit/configDetailHeader.tsx +104 -0
  56. package/src/kayenta/edit/configDetailLoadStates.tsx +36 -0
  57. package/src/kayenta/edit/configDetailLoader.tsx +60 -0
  58. package/src/kayenta/edit/configJson.less +42 -0
  59. package/src/kayenta/edit/configJsonModal.tsx +158 -0
  60. package/src/kayenta/edit/configList.less +7 -0
  61. package/src/kayenta/edit/configList.tsx +57 -0
  62. package/src/kayenta/edit/copyConfigButton.tsx +34 -0
  63. package/src/kayenta/edit/createConfigButton.tsx +34 -0
  64. package/src/kayenta/edit/deleteModal.tsx +87 -0
  65. package/src/kayenta/edit/edit.tsx +24 -0
  66. package/src/kayenta/edit/editMetricEffectSizes.tsx +186 -0
  67. package/src/kayenta/edit/editMetricModal.less +9 -0
  68. package/src/kayenta/edit/editMetricModal.spec.tsx +129 -0
  69. package/src/kayenta/edit/editMetricModal.tsx +294 -0
  70. package/src/kayenta/edit/editMetricValidation.spec.ts +63 -0
  71. package/src/kayenta/edit/editMetricValidation.ts +50 -0
  72. package/src/kayenta/edit/filterTemplateSelector.less +15 -0
  73. package/src/kayenta/edit/filterTemplateSelector.spec.tsx +106 -0
  74. package/src/kayenta/edit/filterTemplateSelector.tsx +194 -0
  75. package/src/kayenta/edit/filterTemplatesValidation.spec.ts +108 -0
  76. package/src/kayenta/edit/filterTemplatesValidation.ts +95 -0
  77. package/src/kayenta/edit/footer.less +30 -0
  78. package/src/kayenta/edit/footer.tsx +12 -0
  79. package/src/kayenta/edit/groupName.tsx +80 -0
  80. package/src/kayenta/edit/groupTabs.tsx +106 -0
  81. package/src/kayenta/edit/groupWeight.tsx +80 -0
  82. package/src/kayenta/edit/groupWeights.tsx +38 -0
  83. package/src/kayenta/edit/inlineTemplateEditor.spec.tsx +42 -0
  84. package/src/kayenta/edit/inlineTemplateEditor.tsx +61 -0
  85. package/src/kayenta/edit/judgeSelect.tsx +82 -0
  86. package/src/kayenta/edit/metricConfigurerDelegator.tsx +30 -0
  87. package/src/kayenta/edit/metricList.less +21 -0
  88. package/src/kayenta/edit/metricList.tsx +215 -0
  89. package/src/kayenta/edit/metricStoreSelector.tsx +66 -0
  90. package/src/kayenta/edit/nameAndDescription.tsx +90 -0
  91. package/src/kayenta/edit/openConfigJsonModalButton.tsx +33 -0
  92. package/src/kayenta/edit/openDeleteModalButton.tsx +50 -0
  93. package/src/kayenta/edit/ownedBy.tsx +34 -0
  94. package/src/kayenta/edit/save.tsx +19 -0
  95. package/src/kayenta/edit/saveConfigButton.tsx +65 -0
  96. package/src/kayenta/edit/saveConfigError.tsx +59 -0
  97. package/src/kayenta/edit/scoring.tsx +35 -0
  98. package/src/kayenta/edit/selectConfig.tsx +10 -0
  99. package/src/kayenta/edit/validationErrors.tsx +39 -0
  100. package/src/kayenta/index.ts +6 -0
  101. package/src/kayenta/layout/addNewButton.tsx +20 -0
  102. package/src/kayenta/layout/centeredDetail.tsx +13 -0
  103. package/src/kayenta/layout/deleteButton.tsx +11 -0
  104. package/src/kayenta/layout/disableable.tsx +87 -0
  105. package/src/kayenta/layout/formList.tsx +26 -0
  106. package/src/kayenta/layout/formRow.tsx +36 -0
  107. package/src/kayenta/layout/formattedDate.tsx +14 -0
  108. package/src/kayenta/layout/index.ts +2 -0
  109. package/src/kayenta/layout/keyValueList.less +20 -0
  110. package/src/kayenta/layout/keyValueList.tsx +114 -0
  111. package/src/kayenta/layout/list.less +9 -0
  112. package/src/kayenta/layout/list.spec.tsx +83 -0
  113. package/src/kayenta/layout/list.tsx +73 -0
  114. package/src/kayenta/layout/listDetail.tsx +33 -0
  115. package/src/kayenta/layout/radioChoice.tsx +29 -0
  116. package/src/kayenta/layout/styleguide.tsx +16 -0
  117. package/src/kayenta/layout/table/index.ts +5 -0
  118. package/src/kayenta/layout/table/nativeTable.tsx +51 -0
  119. package/src/kayenta/layout/table/nativeTableHeader.tsx +26 -0
  120. package/src/kayenta/layout/table/table.tsx +56 -0
  121. package/src/kayenta/layout/table/tableColumn.ts +7 -0
  122. package/src/kayenta/layout/table/tableHeader.tsx +23 -0
  123. package/src/kayenta/layout/tabs.tsx +26 -0
  124. package/src/kayenta/layout/titledSection.less +16 -0
  125. package/src/kayenta/layout/titledSection.tsx +20 -0
  126. package/src/kayenta/layout/titledSubsection.less +11 -0
  127. package/src/kayenta/layout/titledSubsection.tsx +22 -0
  128. package/src/kayenta/manualAnalysis/ManualAnalysisModal.tsx +716 -0
  129. package/src/kayenta/metricStore/atlas/atlasMetricConfigurer.tsx +130 -0
  130. package/src/kayenta/metricStore/atlas/index.ts +8 -0
  131. package/src/kayenta/metricStore/datadog/domain/IDatadogMetricDescriptor.ts +5 -0
  132. package/src/kayenta/metricStore/datadog/index.ts +9 -0
  133. package/src/kayenta/metricStore/datadog/metricConfigurer.tsx +90 -0
  134. package/src/kayenta/metricStore/datadog/metricTypeSelector.spec.tsx +59 -0
  135. package/src/kayenta/metricStore/datadog/metricTypeSelector.tsx +73 -0
  136. package/src/kayenta/metricStore/graphite/domain/IGraphiteMetricDescriptor.ts +5 -0
  137. package/src/kayenta/metricStore/graphite/index.ts +8 -0
  138. package/src/kayenta/metricStore/graphite/metricConfigurer.tsx +54 -0
  139. package/src/kayenta/metricStore/graphite/metricTypeSelector.tsx +80 -0
  140. package/src/kayenta/metricStore/graphite/typeahead.less +3 -0
  141. package/src/kayenta/metricStore/index.ts +8 -0
  142. package/src/kayenta/metricStore/metricStoreConfig.service.ts +12 -0
  143. package/src/kayenta/metricStore/newrelic/domain/INewRelicMetricDescriptor.ts +5 -0
  144. package/src/kayenta/metricStore/newrelic/index.ts +8 -0
  145. package/src/kayenta/metricStore/newrelic/metricConfigurer.tsx +58 -0
  146. package/src/kayenta/metricStore/prometheus/domain/IPrometheusCanaryMetricSetQueryConfig.ts +14 -0
  147. package/src/kayenta/metricStore/prometheus/domain/IPrometheusMetricDescriptor.ts +5 -0
  148. package/src/kayenta/metricStore/prometheus/index.ts +12 -0
  149. package/src/kayenta/metricStore/prometheus/metricConfigurer.tsx +157 -0
  150. package/src/kayenta/metricStore/prometheus/metricTypeSelector.less +5 -0
  151. package/src/kayenta/metricStore/prometheus/metricTypeSelector.spec.tsx +62 -0
  152. package/src/kayenta/metricStore/prometheus/metricTypeSelector.tsx +144 -0
  153. package/src/kayenta/metricStore/prometheus/queryTypeSelectors.spec.ts +61 -0
  154. package/src/kayenta/metricStore/prometheus/queryTypeSelectors.ts +38 -0
  155. package/src/kayenta/metricStore/signalfx/domain/ISignalFxCanaryMetricSetQueryConfig.ts +7 -0
  156. package/src/kayenta/metricStore/signalfx/index.ts +8 -0
  157. package/src/kayenta/metricStore/signalfx/metricConfigurer.less +10 -0
  158. package/src/kayenta/metricStore/signalfx/metricConfigurer.tsx +187 -0
  159. package/src/kayenta/metricStore/stackdriver/domain/IStackdriverCanaryMetricSetQueryConfig.ts +9 -0
  160. package/src/kayenta/metricStore/stackdriver/domain/IStackdriverMetricDescriptor.ts +17 -0
  161. package/src/kayenta/metricStore/stackdriver/index.ts +12 -0
  162. package/src/kayenta/metricStore/stackdriver/metricConfigurer.tsx +144 -0
  163. package/src/kayenta/metricStore/stackdriver/metricTypeSelector.spec.tsx +92 -0
  164. package/src/kayenta/metricStore/stackdriver/metricTypeSelector.tsx +113 -0
  165. package/src/kayenta/middleware/actionInterceptor.ts +29 -0
  166. package/src/kayenta/middleware/asyncDispatch.ts +37 -0
  167. package/src/kayenta/middleware/epics.ts +211 -0
  168. package/src/kayenta/middleware/index.ts +3 -0
  169. package/src/kayenta/navigation/canary.states.stub.ts +28 -0
  170. package/src/kayenta/navigation/canary.states.ts +182 -0
  171. package/src/kayenta/reducers/app.ts +56 -0
  172. package/src/kayenta/reducers/asyncRequest.ts +5 -0
  173. package/src/kayenta/reducers/data.ts +169 -0
  174. package/src/kayenta/reducers/editingTemplate.ts +54 -0
  175. package/src/kayenta/reducers/group.ts +82 -0
  176. package/src/kayenta/reducers/index.ts +245 -0
  177. package/src/kayenta/reducers/prometheusMetricConfig.spec.ts +33 -0
  178. package/src/kayenta/reducers/prometheusMetricConfig.ts +56 -0
  179. package/src/kayenta/reducers/selectedConfig.spec.ts +190 -0
  180. package/src/kayenta/reducers/selectedConfig.ts +566 -0
  181. package/src/kayenta/reducers/selectedRun.ts +101 -0
  182. package/src/kayenta/reducers/signalFxMetricConfig.ts +36 -0
  183. package/src/kayenta/reducers/stackdriverMetricConfig.spec.ts +33 -0
  184. package/src/kayenta/reducers/stackdriverMetricConfig.ts +41 -0
  185. package/src/kayenta/reducers/templates.spec.ts +192 -0
  186. package/src/kayenta/reducers/validators.ts +118 -0
  187. package/src/kayenta/report/detail/allMetricResultsHeader.tsx +32 -0
  188. package/src/kayenta/report/detail/clickableHeader.tsx +21 -0
  189. package/src/kayenta/report/detail/colors.ts +47 -0
  190. package/src/kayenta/report/detail/detail.less +16 -0
  191. package/src/kayenta/report/detail/detail.tsx +48 -0
  192. package/src/kayenta/report/detail/detailLoader.tsx +55 -0
  193. package/src/kayenta/report/detail/graph/graph.tsx +37 -0
  194. package/src/kayenta/report/detail/graph/metricSetPairGraph.service.ts +35 -0
  195. package/src/kayenta/report/detail/graph/semiotic/boxplot.less +45 -0
  196. package/src/kayenta/report/detail/graph/semiotic/boxplot.tsx +283 -0
  197. package/src/kayenta/report/detail/graph/semiotic/chartHeader.tsx +19 -0
  198. package/src/kayenta/report/detail/graph/semiotic/chartLegend.less +26 -0
  199. package/src/kayenta/report/detail/graph/semiotic/chartLegend.tsx +42 -0
  200. package/src/kayenta/report/detail/graph/semiotic/circleIcon.tsx +16 -0
  201. package/src/kayenta/report/detail/graph/semiotic/config.less +5 -0
  202. package/src/kayenta/report/detail/graph/semiotic/config.ts +38 -0
  203. package/src/kayenta/report/detail/graph/semiotic/customAxisTickLabel.tsx +17 -0
  204. package/src/kayenta/report/detail/graph/semiotic/declarations/labella.d.ts +16 -0
  205. package/src/kayenta/report/detail/graph/semiotic/declarations/react-container-dimensions.d.ts +3 -0
  206. package/src/kayenta/report/detail/graph/semiotic/declarations/semiotic.d.ts +160 -0
  207. package/src/kayenta/report/detail/graph/semiotic/differenceArea.less +17 -0
  208. package/src/kayenta/report/detail/graph/semiotic/differenceArea.tsx +186 -0
  209. package/src/kayenta/report/detail/graph/semiotic/histogram.less +22 -0
  210. package/src/kayenta/report/detail/graph/semiotic/histogram.tsx +251 -0
  211. package/src/kayenta/report/detail/graph/semiotic/index.tsx +19 -0
  212. package/src/kayenta/report/detail/graph/semiotic/noValidDataSign.less +5 -0
  213. package/src/kayenta/report/detail/graph/semiotic/noValidDataSign.tsx +10 -0
  214. package/src/kayenta/report/detail/graph/semiotic/secondaryTSXAxis.less +6 -0
  215. package/src/kayenta/report/detail/graph/semiotic/secondaryTSXAxis.tsx +58 -0
  216. package/src/kayenta/report/detail/graph/semiotic/semiotic.service.ts +32 -0
  217. package/src/kayenta/report/detail/graph/semiotic/semioticGraph.less +53 -0
  218. package/src/kayenta/report/detail/graph/semiotic/semioticGraph.tsx +49 -0
  219. package/src/kayenta/report/detail/graph/semiotic/timeSeries.less +42 -0
  220. package/src/kayenta/report/detail/graph/semiotic/timeSeries.tsx +473 -0
  221. package/src/kayenta/report/detail/graph/semiotic/tooltip.tsx +55 -0
  222. package/src/kayenta/report/detail/graph/semiotic/utils.ts +90 -0
  223. package/src/kayenta/report/detail/graphTypeSelector.less +4 -0
  224. package/src/kayenta/report/detail/graphTypeSelector.tsx +50 -0
  225. package/src/kayenta/report/detail/groupScores.tsx +68 -0
  226. package/src/kayenta/report/detail/header.less +70 -0
  227. package/src/kayenta/report/detail/header.tsx +39 -0
  228. package/src/kayenta/report/detail/headerArrow.tsx +13 -0
  229. package/src/kayenta/report/detail/loadStates.tsx +31 -0
  230. package/src/kayenta/report/detail/metricResultActions.less +29 -0
  231. package/src/kayenta/report/detail/metricResultActions.tsx +87 -0
  232. package/src/kayenta/report/detail/metricResultClassification.tsx +22 -0
  233. package/src/kayenta/report/detail/metricResultDetail.tsx +20 -0
  234. package/src/kayenta/report/detail/metricResultDetailLayout.tsx +19 -0
  235. package/src/kayenta/report/detail/metricResultDeviation.tsx +25 -0
  236. package/src/kayenta/report/detail/metricResultStats.less +9 -0
  237. package/src/kayenta/report/detail/metricResultStats.tsx +120 -0
  238. package/src/kayenta/report/detail/metricResults.less +12 -0
  239. package/src/kayenta/report/detail/metricResults.tsx +52 -0
  240. package/src/kayenta/report/detail/metricResultsClassificationFilters.tsx +65 -0
  241. package/src/kayenta/report/detail/metricResultsColumns.tsx +27 -0
  242. package/src/kayenta/report/detail/metricResultsList.less +44 -0
  243. package/src/kayenta/report/detail/metricResultsList.tsx +120 -0
  244. package/src/kayenta/report/detail/metricSetPairLoadStates.tsx +22 -0
  245. package/src/kayenta/report/detail/multipleResultsTable.tsx +81 -0
  246. package/src/kayenta/report/detail/reportException.tsx +57 -0
  247. package/src/kayenta/report/detail/reportExplanation.less +12 -0
  248. package/src/kayenta/report/detail/reportExplanation.tsx +32 -0
  249. package/src/kayenta/report/detail/reportMetadata.tsx +167 -0
  250. package/src/kayenta/report/detail/reportScores.less +47 -0
  251. package/src/kayenta/report/detail/reportScores.tsx +80 -0
  252. package/src/kayenta/report/detail/score.tsx +33 -0
  253. package/src/kayenta/report/detail/sourceLinks.tsx +69 -0
  254. package/src/kayenta/report/list/configLink.tsx +32 -0
  255. package/src/kayenta/report/list/executionList.less +7 -0
  256. package/src/kayenta/report/list/loadStates.tsx +32 -0
  257. package/src/kayenta/report/list/pipelineLink.tsx +15 -0
  258. package/src/kayenta/report/list/reportLink.tsx +33 -0
  259. package/src/kayenta/report/list/table.tsx +309 -0
  260. package/src/kayenta/report/report.tsx +11 -0
  261. package/src/kayenta/selectors/filterTemplatesSelectors.ts +87 -0
  262. package/src/kayenta/selectors/index.ts +62 -0
  263. package/src/kayenta/service/canaryConfig.service.ts +122 -0
  264. package/src/kayenta/service/canaryRun.service.ts +60 -0
  265. package/src/kayenta/service/delegateFactory.ts +24 -0
  266. package/src/kayenta/service/metricsServiceMetadata.service.ts +9 -0
  267. package/src/kayenta/stages/kayentaStage/AnalysisType.spec.tsx +47 -0
  268. package/src/kayenta/stages/kayentaStage/AnalysisType.tsx +49 -0
  269. package/src/kayenta/stages/kayentaStage/CanaryExecutionLabel.tsx +26 -0
  270. package/src/kayenta/stages/kayentaStage/analysisType.component.ts +12 -0
  271. package/src/kayenta/stages/kayentaStage/canaryRunSummaries.component.ts +12 -0
  272. package/src/kayenta/stages/kayentaStage/canaryRunSummaries.less +5 -0
  273. package/src/kayenta/stages/kayentaStage/canaryRunSummaries.tsx +136 -0
  274. package/src/kayenta/stages/kayentaStage/forAnalysisType.component.ts +45 -0
  275. package/src/kayenta/stages/kayentaStage/kayentaStage.controller.ts +789 -0
  276. package/src/kayenta/stages/kayentaStage/kayentaStage.html +528 -0
  277. package/src/kayenta/stages/kayentaStage/kayentaStage.less +5 -0
  278. package/src/kayenta/stages/kayentaStage/kayentaStage.transformer.ts +179 -0
  279. package/src/kayenta/stages/kayentaStage/kayentaStage.ts +221 -0
  280. package/src/kayenta/stages/kayentaStage/kayentaStageConfigSection.component.ts +21 -0
  281. package/src/kayenta/stages/kayentaStage/kayentaStageExecutionDetails.controller.ts +88 -0
  282. package/src/kayenta/stages/kayentaStage/kayentaStageExecutionDetails.html +114 -0
  283. package/src/kayenta/stages/kayentaStage/kayentaStageExecutionDetails.less +6 -0
  284. package/src/kayenta/stages/kayentaStage/stageTypes.ts +5 -0
  285. package/src/kayenta/utils/duration.spec.ts +69 -0
  286. package/src/kayenta/utils/duration.ts +48 -0
  287. package/src/lazy.ts +29 -0
  288. package/src/stub.ts +60 -0
  289. package/tsconfig.json +11 -0
@@ -0,0 +1,65 @@
1
+ import * as React from 'react';
2
+ import { connect, Dispatch } from 'react-redux';
3
+
4
+ import * as Creators from '../../actions/creators';
5
+ import { ICanaryAnalysisResult, MetricClassificationLabel } from '../../domain';
6
+ import { ICanaryState } from '../../reducers';
7
+
8
+ export interface IMetricFiltersOwnProps {
9
+ results: ICanaryAnalysisResult[];
10
+ }
11
+
12
+ interface IMetricFiltersStateProps {
13
+ metricFilters: MetricClassificationLabel[];
14
+ }
15
+
16
+ interface IMetricFiltersDispatchProps {
17
+ toggle: (classification: MetricClassificationLabel) => void;
18
+ }
19
+
20
+ const MetricFilters = ({
21
+ results,
22
+ metricFilters,
23
+ toggle,
24
+ }: IMetricFiltersOwnProps & IMetricFiltersStateProps & IMetricFiltersDispatchProps) => {
25
+ const options = [
26
+ MetricClassificationLabel.Error,
27
+ MetricClassificationLabel.High,
28
+ MetricClassificationLabel.Low,
29
+ MetricClassificationLabel.Nodata,
30
+ MetricClassificationLabel.Pass,
31
+ ]
32
+ .map((classification) => ({
33
+ classification,
34
+ count: results.filter((r) => r.classification === classification).length,
35
+ }))
36
+ .filter((option) => option.count);
37
+
38
+ const checkboxes = options.map((o) => (
39
+ <label key={o.classification}>
40
+ <input
41
+ type="checkbox"
42
+ checked={metricFilters.includes(o.classification)}
43
+ onChange={() => toggle(o.classification)}
44
+ />{' '}
45
+ {o.classification} ({o.count}){' '}
46
+ </label>
47
+ ));
48
+
49
+ return <div className="metric-filter-options horizontal center">{checkboxes}</div>;
50
+ };
51
+
52
+ const mapStateToProps = (state: ICanaryState): IMetricFiltersStateProps => ({
53
+ metricFilters: state.selectedRun.metricFilters,
54
+ });
55
+
56
+ const mapDispatchToProps = (
57
+ dispatch: Dispatch<ICanaryState>,
58
+ ownProps: IMetricFiltersOwnProps,
59
+ ): IMetricFiltersOwnProps & IMetricFiltersDispatchProps => ({
60
+ toggle: (classification: MetricClassificationLabel) =>
61
+ dispatch(Creators.toggleMetricClassificationFilter({ classification })),
62
+ ...ownProps,
63
+ });
64
+
65
+ export default connect(mapStateToProps, mapDispatchToProps)(MetricFilters);
@@ -0,0 +1,27 @@
1
+ import { ITableColumn } from 'kayenta/layout/table';
2
+ import * as React from 'react';
3
+
4
+ import { BreakString } from '@spinnaker/core';
5
+
6
+ import MetricResultClassification from './metricResultClassification';
7
+ import MetricResultDeviation from './metricResultDeviation';
8
+ import { IMetricResultsTableRow } from './metricResultsList';
9
+
10
+ export const metricResultsColumns: Array<ITableColumn<IMetricResultsTableRow>> = [
11
+ {
12
+ label: 'metric name',
13
+ getContent: ({ metricName }) => <BreakString>{metricName}</BreakString>,
14
+ width: 5,
15
+ },
16
+ {
17
+ label: 'deviation',
18
+ getContent: ({ results }) =>
19
+ results[0].resultMetadata && <MetricResultDeviation ratio={results[0].resultMetadata.ratio} />,
20
+ width: 1,
21
+ },
22
+ {
23
+ label: 'result',
24
+ getContent: ({ results }) => <MetricResultClassification classification={results[0].classification} />,
25
+ width: 1,
26
+ },
27
+ ];
@@ -0,0 +1,44 @@
1
+ .metric-results-list {
2
+ .table-header {
3
+ h6 {
4
+ padding-left: 5px;
5
+ }
6
+ }
7
+
8
+ .tabs-vertical {
9
+ overflow-y: auto;
10
+ display: block;
11
+ li {
12
+ padding: 8px 0 8px 5px;
13
+ }
14
+ }
15
+
16
+ .metric-result-classification {
17
+ margin-right: 3px;
18
+ }
19
+
20
+ .table-row {
21
+ cursor: pointer;
22
+ }
23
+
24
+ .list-group {
25
+ margin-bottom: 0;
26
+ }
27
+
28
+ li.multiple-results {
29
+ padding-bottom: 0;
30
+ border: none;
31
+
32
+ &:hover {
33
+ border: none;
34
+ }
35
+ }
36
+
37
+ .multiple-results-table {
38
+ padding-left: 8px;
39
+
40
+ h6 {
41
+ margin-top: 7px;
42
+ }
43
+ }
44
+ }
@@ -0,0 +1,120 @@
1
+ import classNames from 'classnames';
2
+ import * as Creators from 'kayenta/actions/creators';
3
+ import { ICanaryAnalysisResult } from 'kayenta/domain/ICanaryJudgeResult';
4
+ import { Table } from 'kayenta/layout/table';
5
+ import { ICanaryState } from 'kayenta/reducers';
6
+ import * as React from 'react';
7
+ import { connect, Dispatch } from 'react-redux';
8
+
9
+ import { MetricClassificationLabel } from '../../domain';
10
+ import MetricFilters from './metricResultsClassificationFilters';
11
+ import { metricResultsColumns } from './metricResultsColumns';
12
+ import MultipleResultsTable from './multipleResultsTable';
13
+
14
+ import './metricResultsList.less';
15
+
16
+ export interface IResultsListOwnProps {
17
+ results: ICanaryAnalysisResult[];
18
+ }
19
+
20
+ interface IResultsListDispatchProps {
21
+ select: (metric: string) => void;
22
+ }
23
+
24
+ interface IResultsListStateProps {
25
+ selectedMetric: string;
26
+ metricFilters: MetricClassificationLabel[];
27
+ }
28
+
29
+ export interface IMetricResultsTableRow {
30
+ metricName: string;
31
+ results: ICanaryAnalysisResult[];
32
+ hasMultipleRows: boolean;
33
+ }
34
+
35
+ const buildTableRows = (
36
+ results: ICanaryAnalysisResult[],
37
+ metricFilters: MetricClassificationLabel[],
38
+ ): IMetricResultsTableRow[] => {
39
+ const unfilteredCounts = results.reduce((map, result) => {
40
+ if (!map.has(result.name)) {
41
+ map.set(result.name, 0);
42
+ }
43
+ map.set(result.name, map.get(result.name) + 1);
44
+ return map;
45
+ }, new Map<string, number>());
46
+
47
+ const tableRowsByMetricName = results
48
+ .filter((result) => !metricFilters.length || metricFilters.includes(result.classification))
49
+ .reduce(
50
+ (map, result) =>
51
+ map.has(result.name)
52
+ ? map.set(result.name, {
53
+ metricName: result.name,
54
+ results: map.get(result.name).results.concat(result),
55
+ hasMultipleRows: unfilteredCounts.get(result.name) > 1,
56
+ })
57
+ : map.set(result.name, {
58
+ metricName: result.name,
59
+ results: [result],
60
+ hasMultipleRows: unfilteredCounts.get(result.name) > 1,
61
+ }),
62
+ new Map<string, IMetricResultsTableRow>(),
63
+ );
64
+
65
+ return Array.from(tableRowsByMetricName.values());
66
+ };
67
+
68
+ const buildRowForMetricWithMultipleResults = (row: IMetricResultsTableRow) => {
69
+ if (!row.hasMultipleRows) {
70
+ return null;
71
+ }
72
+
73
+ return (
74
+ <li className="horizontal multiple-results">
75
+ <section className="vertical flex-fill">
76
+ <div>{row.metricName}</div>
77
+ <MultipleResultsTable results={row.results} />
78
+ </section>
79
+ </li>
80
+ );
81
+ };
82
+
83
+ const ResultsList = ({
84
+ results,
85
+ select,
86
+ selectedMetric,
87
+ metricFilters,
88
+ }: IResultsListOwnProps & IResultsListDispatchProps & IResultsListStateProps) => {
89
+ const rows = buildTableRows(results, metricFilters);
90
+ return (
91
+ <section className="vertical metric-results-list flex-1">
92
+ <MetricFilters results={results} />
93
+ <Table
94
+ rowKey={(r) => r.metricName}
95
+ tableBodyClassName="list-unstyled tabs-vertical flex-1 sp-padding-xl-bottom"
96
+ rowClassName={(r) => classNames('horizontal', { selected: r.results[0].id === selectedMetric })}
97
+ rows={rows}
98
+ columns={metricResultsColumns}
99
+ onRowClick={(r) => select(r.results[0].id)}
100
+ className="flex-1 vertical"
101
+ customRow={buildRowForMetricWithMultipleResults}
102
+ />
103
+ </section>
104
+ );
105
+ };
106
+
107
+ const mapStateToProps = (state: ICanaryState): IResultsListStateProps => ({
108
+ selectedMetric: state.selectedRun.selectedMetric,
109
+ metricFilters: state.selectedRun.metricFilters,
110
+ });
111
+
112
+ const mapDispatchToProps = (
113
+ dispatch: Dispatch<ICanaryState>,
114
+ ownProps: IResultsListOwnProps,
115
+ ): IResultsListOwnProps & IResultsListDispatchProps => ({
116
+ select: (metricId: string) => dispatch(Creators.selectReportMetric({ metricId })),
117
+ ...ownProps,
118
+ });
119
+
120
+ export default connect(mapStateToProps, mapDispatchToProps)(ResultsList);
@@ -0,0 +1,22 @@
1
+ import LoadStatesBuilder from 'kayenta/components/loadStates';
2
+ import { AsyncRequestState } from 'kayenta/reducers/asyncRequest';
3
+ import { ICanaryState } from 'kayenta/reducers/index';
4
+ import * as React from 'react';
5
+ import { connect } from 'react-redux';
6
+
7
+ import MetricResultDetailLayout from './metricResultDetailLayout';
8
+
9
+ const MetricSetPairLoadStates = ({ state }: { state: AsyncRequestState }) => {
10
+ const LoadStates = new LoadStatesBuilder()
11
+ .onFulfilled(<MetricResultDetailLayout />)
12
+ .onFailed(<h3 className="heading-3 text-center">Could not load metrics.</h3>)
13
+ .build();
14
+
15
+ return <LoadStates state={state} />;
16
+ };
17
+
18
+ const mapStateToProps = (state: ICanaryState) => ({
19
+ state: state.selectedRun.metricSetPair.load,
20
+ });
21
+
22
+ export default connect(mapStateToProps)(MetricSetPairLoadStates);
@@ -0,0 +1,81 @@
1
+ import classNames from 'classnames';
2
+ import * as Creators from 'kayenta/actions/creators';
3
+ import { ICanaryAnalysisResult } from 'kayenta/domain/ICanaryJudgeResult';
4
+ import { ITableColumn, Table } from 'kayenta/layout/table';
5
+ import { ICanaryState } from 'kayenta/reducers';
6
+ import { selectedMetricResultIdSelector } from 'kayenta/selectors';
7
+ import { chain } from 'lodash';
8
+ import * as React from 'react';
9
+ import { connect, Dispatch } from 'react-redux';
10
+
11
+ import { BreakString } from '@spinnaker/core';
12
+
13
+ import MetricResultClassification from './metricResultClassification';
14
+ import MetricResultDeviation from './metricResultDeviation';
15
+
16
+ interface IMultipleResultsTableOwnProps {
17
+ results: ICanaryAnalysisResult[];
18
+ }
19
+
20
+ interface IMultipleResultsTableStateProps {
21
+ selectedResult: string;
22
+ }
23
+
24
+ interface IMultipleResultsTableDispatchProps {
25
+ select: (metricId: string) => void;
26
+ }
27
+
28
+ const MultipleResultsTable = ({
29
+ results,
30
+ select,
31
+ selectedResult,
32
+ }: IMultipleResultsTableOwnProps & IMultipleResultsTableStateProps & IMultipleResultsTableDispatchProps) => {
33
+ const tagKeys = chain(results)
34
+ .flatMap((r) => Object.keys(r.tags || {}))
35
+ .uniq()
36
+ .value();
37
+
38
+ let columns: Array<ITableColumn<ICanaryAnalysisResult>> = tagKeys.map((key) => ({
39
+ label: key,
40
+ width: 5,
41
+ getContent: (result: ICanaryAnalysisResult) => <BreakString>{result.tags[key]}</BreakString>,
42
+ }));
43
+
44
+ columns = columns.concat([
45
+ {
46
+ width: 1,
47
+ getContent: ({ resultMetadata }) => resultMetadata && <MetricResultDeviation ratio={resultMetadata.ratio} />,
48
+ },
49
+ {
50
+ width: 1,
51
+ getContent: ({ classification }) => <MetricResultClassification classification={classification} />,
52
+ },
53
+ ]);
54
+
55
+ return (
56
+ <Table
57
+ rows={results}
58
+ columns={columns}
59
+ className="multiple-results-table"
60
+ headerClassName="sticky-header-2"
61
+ rowClassName={(r) => classNames('horizontal', { selected: r.id === selectedResult })}
62
+ rowKey={(r) =>
63
+ Object.entries(r.tags || {})
64
+ .map(([key, value]) => `${key}:${value}`)
65
+ .join(':')
66
+ }
67
+ onRowClick={(r) => select(r.id)}
68
+ />
69
+ );
70
+ };
71
+
72
+ const mapStateToProps = (state: ICanaryState) => ({
73
+ selectedResult: selectedMetricResultIdSelector(state),
74
+ });
75
+
76
+ const mapDispatchToProps = (dispatch: Dispatch<ICanaryState>, ownProps: IMultipleResultsTableOwnProps) => ({
77
+ ...ownProps,
78
+ select: (metricId: string) => dispatch(Creators.selectReportMetric({ metricId })),
79
+ });
80
+
81
+ export default connect(mapStateToProps, mapDispatchToProps)(MultipleResultsTable);
@@ -0,0 +1,57 @@
1
+ import { ICanaryExecutionException } from 'kayenta/domain';
2
+ import { ICanaryState } from 'kayenta/reducers';
3
+ import * as React from 'react';
4
+ import { connect } from 'react-redux';
5
+
6
+ import SourceLinks from './sourceLinks';
7
+
8
+ import './reportExplanation.less';
9
+
10
+ interface IReportExceptionProps {
11
+ exception: ICanaryExecutionException;
12
+ }
13
+
14
+ const ReportException = ({ exception }: IReportExceptionProps) => {
15
+ const errorMessages = [];
16
+ if (exception.details) {
17
+ // error messages can be in an array on the details, or a single field, or possibly both?
18
+ // it's unclear based on Orca where the error message might be
19
+ const { error, errors } = exception.details;
20
+ if (error) {
21
+ errorMessages.push(error);
22
+ }
23
+ if (errors?.length) {
24
+ errors.forEach((m: string) => errorMessages.push(m));
25
+ }
26
+ }
27
+ // if there were no errors, set a default message
28
+ if (!errorMessages.length) {
29
+ errorMessages.push('No error message provided. Click the "Report" link to see more details.');
30
+ }
31
+
32
+ return (
33
+ <>
34
+ <h3 className="text-center">Canary report failed</h3>
35
+ <dl>
36
+ <dt>Details</dt>
37
+ <dd>
38
+ {errorMessages.map((message, i) => (
39
+ <div key={i}>
40
+ <code>{message}</code>
41
+ </div>
42
+ ))}
43
+ </dd>
44
+ <dt>Source</dt>
45
+ <dd>
46
+ <SourceLinks />
47
+ </dd>
48
+ </dl>
49
+ </>
50
+ );
51
+ };
52
+
53
+ const mapStateToProps = (state: ICanaryState) => ({
54
+ exception: state.selectedRun.run.exception,
55
+ });
56
+
57
+ export default connect(mapStateToProps)(ReportException);
@@ -0,0 +1,12 @@
1
+ .report-explanation {
2
+ background-color: var(--color-warning);
3
+
4
+ height: 30px;
5
+ margin-bottom: 5px;
6
+ margin-left: 15px;
7
+ padding-top: 4px;
8
+
9
+ font-weight: bold;
10
+ font-size: 14px;
11
+ text-align: center;
12
+ }
@@ -0,0 +1,32 @@
1
+ import { ICanaryExecutionStatusResult } from 'kayenta/domain/ICanaryExecutionStatusResult';
2
+ import { ICanaryState } from 'kayenta/reducers';
3
+ import * as React from 'react';
4
+ import { connect } from 'react-redux';
5
+
6
+ import './reportExplanation.less';
7
+
8
+ interface IReportMetadata {
9
+ run: ICanaryExecutionStatusResult;
10
+ }
11
+
12
+ const getReason = (run: ICanaryExecutionStatusResult): string =>
13
+ run?.result?.judgeResult?.score?.classificationReason ?? null;
14
+
15
+ const ReportExplanation = ({ run }: IReportMetadata) => {
16
+ const classificationReason = getReason(run);
17
+
18
+ if (classificationReason) {
19
+ return (
20
+ <section>
21
+ <div className="report-explanation">{classificationReason}</div>
22
+ </section>
23
+ );
24
+ }
25
+ return null;
26
+ };
27
+
28
+ const mapStateToProps = (state: ICanaryState) => ({
29
+ run: state.selectedRun.run,
30
+ });
31
+
32
+ export default connect(mapStateToProps)(ReportExplanation);
@@ -0,0 +1,167 @@
1
+ import { ICanaryExecutionStatusResult } from 'kayenta/domain/ICanaryExecutionStatusResult';
2
+ import FormattedDate from 'kayenta/layout/formattedDate';
3
+ import { ICanaryState } from 'kayenta/reducers';
4
+ import * as React from 'react';
5
+ import { connect } from 'react-redux';
6
+
7
+ import { HoverablePopover } from '@spinnaker/core';
8
+
9
+ import SourceLinks from './sourceLinks';
10
+
11
+ interface IReportMetadata {
12
+ run: ICanaryExecutionStatusResult;
13
+ }
14
+
15
+ interface IMetadataGroup {
16
+ label?: string;
17
+ entries: IMetadataEntry[];
18
+ }
19
+
20
+ interface IMetadataEntry {
21
+ label: string;
22
+ getContent: () => JSX.Element;
23
+ popover?: boolean;
24
+ }
25
+
26
+ const Label = ({ label, extraClass }: { label: string; extraClass?: string }) => (
27
+ <label className={`label uppercase color-text-primary ${extraClass}`}>{label}</label>
28
+ );
29
+
30
+ // TODO(dpeach): this only supports canary runs with a single scope.
31
+ const buildScopeMetadataEntries = (run: ICanaryExecutionStatusResult): IMetadataGroup[] => {
32
+ const request = run.canaryExecutionRequest;
33
+ const scopes = Object.values(request.scopes);
34
+ if (scopes.length > 1) {
35
+ return null;
36
+ }
37
+
38
+ const {
39
+ controlScope: { step, location: controlLocation, scope: controlScope },
40
+ experimentScope: {
41
+ // Since baseline starttime may be offset in canaries from Orca,
42
+ // Choose the experiment start and end to represent
43
+ // the canary start and end times
44
+ start: experimentStart,
45
+ end: experimentEnd,
46
+ location: experimentLocation,
47
+ scope: experimentScope,
48
+ },
49
+ } = scopes[0];
50
+
51
+ return [
52
+ {
53
+ label: 'baseline',
54
+ entries: [
55
+ {
56
+ label: 'scope',
57
+ popover: true,
58
+ getContent: () => <p className="kayenta-scope">{controlScope}</p>,
59
+ },
60
+ {
61
+ label: 'location',
62
+ getContent: () => <p>{controlLocation}</p>,
63
+ },
64
+ ],
65
+ },
66
+ {
67
+ label: 'canary',
68
+ entries: [
69
+ {
70
+ label: 'scope',
71
+ popover: true,
72
+ getContent: () => <p className="kayenta-scope">{experimentScope}</p>,
73
+ },
74
+ {
75
+ label: 'location',
76
+ getContent: () => <p>{experimentLocation}</p>,
77
+ },
78
+ ],
79
+ },
80
+ {
81
+ label: 'time',
82
+ entries: [
83
+ {
84
+ label: 'start',
85
+ getContent: () => (
86
+ <p>
87
+ <FormattedDate dateIso={experimentStart} />
88
+ </p>
89
+ ),
90
+ },
91
+ {
92
+ label: 'end',
93
+ getContent: () => (
94
+ <p>
95
+ <FormattedDate dateIso={experimentEnd} />
96
+ </p>
97
+ ),
98
+ },
99
+ {
100
+ label: 'step',
101
+ getContent: () => {
102
+ const mins = step / 60;
103
+ return (
104
+ <p>
105
+ {mins} min
106
+ {mins === 1 ? '' : 's'}
107
+ </p>
108
+ );
109
+ },
110
+ },
111
+ ],
112
+ },
113
+ ];
114
+ };
115
+
116
+ const ReportMetadata = ({ run }: IReportMetadata) => {
117
+ const {
118
+ thresholds: { marginal, pass },
119
+ } = run.canaryExecutionRequest;
120
+
121
+ const metadataGroups = buildScopeMetadataEntries(run) || [];
122
+ metadataGroups.push({
123
+ label: 'threshold',
124
+ entries: [
125
+ {
126
+ label: 'marginal',
127
+ getContent: () => <p>{marginal}</p>,
128
+ },
129
+ {
130
+ label: 'pass',
131
+ getContent: () => <p>{pass}</p>,
132
+ },
133
+ ],
134
+ });
135
+
136
+ return (
137
+ <section className="report-metadata">
138
+ {metadataGroups.map((group, index) => (
139
+ <div className={`group group-${group.label}`} key={group.label || index}>
140
+ <Label label={group.label || ''} extraClass="label-lg" />
141
+ <ul className="list-unstyled">
142
+ {group.entries.map((e) => (
143
+ <li key={e.label || index}>
144
+ <Label label={e.label} />
145
+ {e.popover ? (
146
+ <HoverablePopover template={e.getContent()}>{e.getContent()}</HoverablePopover>
147
+ ) : (
148
+ <>{e.getContent()}</>
149
+ )}
150
+ </li>
151
+ ))}
152
+ </ul>
153
+ </div>
154
+ ))}
155
+ <div className="group group-source" key="source">
156
+ <Label label="Source" extraClass="label-lgf" />
157
+ <SourceLinks />
158
+ </div>
159
+ </section>
160
+ );
161
+ };
162
+
163
+ const mapStateToProps = (state: ICanaryState) => ({
164
+ run: state.selectedRun.run,
165
+ });
166
+
167
+ export default connect(mapStateToProps)(ReportMetadata);
@@ -0,0 +1,47 @@
1
+ .report-scores {
2
+ height: 40px;
3
+ margin-bottom: 20px;
4
+ .group-score,
5
+ .all-metric-results-header {
6
+ height: 30px;
7
+ color: white;
8
+ border-right: 1px solid white;
9
+ padding-top: 4px;
10
+
11
+ h3 {
12
+ font-weight: 600;
13
+ }
14
+ }
15
+
16
+ .group-score:last-of-type {
17
+ border-right: none;
18
+ }
19
+
20
+ .report-score {
21
+ &.active {
22
+ .arrow-down {
23
+ width: 0;
24
+ height: 0;
25
+ margin: 5px auto 0 auto;
26
+ border-left: 8px solid transparent;
27
+ border-right: 8px solid transparent;
28
+ border-top: 8px solid transparent;
29
+ }
30
+ }
31
+ }
32
+
33
+ .all-metric-results-header {
34
+ border: 1px solid var(--color-titanium);
35
+ width: 70%;
36
+ margin: 0 auto;
37
+
38
+ .arrow-down {
39
+ &.inner {
40
+ margin-top: -9px;
41
+ }
42
+ &.outer {
43
+ margin-top: 4px;
44
+ }
45
+ }
46
+ }
47
+ }