@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,77 @@
1
+ .score.score-label {
2
+ font-size: 110%;
3
+ padding: 3px 6px;
4
+ display: inline-block;
5
+ height: 20px;
6
+ line-height: 14px;
7
+ color: white;
8
+ &.inverse {
9
+ padding: 0;
10
+ }
11
+ &.label-failing {
12
+ background-color: var(--color-alert-light);
13
+ &.inverse {
14
+ color: var(--color-alert-light);
15
+ }
16
+ }
17
+ &.label-healthy {
18
+ background-color: var(--color-success);
19
+ &.inverse {
20
+ color: var(--color-success);
21
+ }
22
+ }
23
+ &.label-unhealthy {
24
+ background-color: var(--color-danger);
25
+ &.inverse {
26
+ color: var(--color-danger);
27
+ }
28
+ }
29
+ &.label-unknown {
30
+ background-color: var(--color-text-caption);
31
+ &.inverse {
32
+ color: var(--color-text-caption);
33
+ }
34
+ }
35
+ &.inverse {
36
+ background-color: transparent;
37
+ font-size: 100%;
38
+ padding: 0;
39
+ }
40
+ .score-classification {
41
+ font-size: 45%;
42
+ display: block;
43
+ margin-top: 15px;
44
+ font-weight: 400;
45
+ text-transform: uppercase;
46
+ }
47
+ }
48
+
49
+ .score-large.score {
50
+ height: 48px;
51
+ }
52
+
53
+ .score-large .score.score-label {
54
+ height: auto;
55
+ line-height: 48px;
56
+ padding: 0 10px;
57
+ font-size: 28px;
58
+ font-weight: bold;
59
+ margin: 0;
60
+ border-radius: 0;
61
+ display: inline-block;
62
+ }
63
+
64
+ .score.score-report {
65
+ align-items: center;
66
+ padding: 12px 12px 6px;
67
+ font-size: 28px;
68
+ font-weight: bold;
69
+ margin-bottom: 10px;
70
+ height: 100%;
71
+ }
72
+
73
+ .table-row {
74
+ .score.score-report {
75
+ display: inline-block;
76
+ }
77
+ }
@@ -0,0 +1,12 @@
1
+ import { module } from 'angular';
2
+ import { react2angular } from 'react2angular';
3
+
4
+ import { withErrorBoundary } from '@spinnaker/core';
5
+
6
+ import { CanaryScore } from './canaryScore';
7
+
8
+ export const CANARY_SCORE_COMPONENT = 'spinnaker.kayenta.score.component';
9
+ module(CANARY_SCORE_COMPONENT, []).component(
10
+ 'kayentaCanaryScore',
11
+ react2angular(withErrorBoundary(CanaryScore, 'kayentaCanaryScore'), ['score', 'health', 'result', 'inverse']),
12
+ );
@@ -0,0 +1,63 @@
1
+ import classNames from 'classnames';
2
+ import * as React from 'react';
3
+
4
+ import { ScoreClassificationLabel } from '../domain/ScoreClassificationLabel';
5
+ import { getHealthLabel } from '../service/canaryRun.service';
6
+
7
+ import './canaryScore.component.less';
8
+
9
+ export interface ICanaryScoreProps {
10
+ score: number | string;
11
+ health?: string;
12
+ result: string;
13
+ inverse: boolean;
14
+ className?: string;
15
+ classification?: ScoreClassificationLabel;
16
+ }
17
+
18
+ export interface ICanaryScoreState {
19
+ score: number | string;
20
+ healthLabel: string;
21
+ }
22
+
23
+ export class CanaryScore extends React.Component<ICanaryScoreProps, ICanaryScoreState> {
24
+ public static defaultProps: Partial<ICanaryScoreProps> = {
25
+ health: '',
26
+ };
27
+
28
+ constructor(props: ICanaryScoreProps) {
29
+ super(props);
30
+ this.state = this.getLabelState(props);
31
+ }
32
+
33
+ private getLabelState(props: ICanaryScoreProps): ICanaryScoreState {
34
+ const score = props.score === 0 || (props.score && props.score > 0) ? props.score : 'N/A';
35
+
36
+ return {
37
+ score,
38
+ healthLabel: getHealthLabel(props.health, props.result),
39
+ };
40
+ }
41
+
42
+ public componentWillReceiveProps(props: ICanaryScoreProps) {
43
+ this.setState(this.getLabelState(props));
44
+ }
45
+
46
+ public render() {
47
+ const className = [
48
+ this.props.inverse ? 'inverse' : '',
49
+ 'score',
50
+ 'score-label',
51
+ 'label-default',
52
+ `label-${this.state.healthLabel}`,
53
+ ].join(' ');
54
+ return (
55
+ <span className={classNames(className, this.props.className)}>
56
+ {this.state.score}
57
+ {this.props.classification && (
58
+ <span className="score-classification">{this.props.classification.toString()}</span>
59
+ )}
60
+ </span>
61
+ );
62
+ }
63
+ }
@@ -0,0 +1,20 @@
1
+ import { module } from 'angular';
2
+ import { react2angular } from 'react2angular';
3
+
4
+ import { withErrorBoundary } from '@spinnaker/core';
5
+
6
+ import { CanaryScores } from './canaryScores';
7
+
8
+ export const CANARY_SCORES_CONFIG_COMPONENT = 'spinnaker.kayenta.canaryScores.component';
9
+ module(CANARY_SCORES_CONFIG_COMPONENT, []).component(
10
+ 'kayentaCanaryScores',
11
+ react2angular(withErrorBoundary(CanaryScores, 'kayentaCanaryScores'), [
12
+ 'onChange',
13
+ 'successfulHelpFieldId',
14
+ 'successfulLabel',
15
+ 'successfulScore',
16
+ 'unhealthyHelpFieldId',
17
+ 'unhealthyLabel',
18
+ 'unhealthyScore',
19
+ ]),
20
+ );
@@ -0,0 +1,22 @@
1
+ div.canary-score {
2
+ div.progress {
3
+ border-radius: 0;
4
+ box-shadow: none;
5
+ -webkit-box-shadow: none;
6
+ height: 10px;
7
+ margin: 20px 0 10px;
8
+ }
9
+ div.progress-bar {
10
+ box-shadow: none;
11
+ -webkit-box-shadow: none;
12
+ }
13
+ div.progress-bar-danger {
14
+ background-color: var(--color-danger);
15
+ }
16
+ div.progress-bar.progress-bar-warning {
17
+ background-color: var(--color-alto);
18
+ }
19
+ div.progress-bar.progress-bar-success {
20
+ background-color: var(--color-success-light);
21
+ }
22
+ }
@@ -0,0 +1,163 @@
1
+ import classNames from 'classnames';
2
+ import { isString } from 'lodash';
3
+ import * as React from 'react';
4
+
5
+ import { HelpField } from '@spinnaker/core';
6
+
7
+ import './canaryScores.less';
8
+
9
+ export interface IScoreConfig {
10
+ successfulScore: string;
11
+ unhealthyScore: string;
12
+ }
13
+
14
+ export interface ICanaryScoresProps {
15
+ onChange: (scoreConfig: IScoreConfig) => void;
16
+ successfulHelpFieldId?: string;
17
+ successfulLabel?: string;
18
+ successfulScore: string;
19
+ unhealthyHelpFieldId?: string;
20
+ unhealthyLabel?: string;
21
+ unhealthyScore: string;
22
+ disabled?: boolean;
23
+ }
24
+
25
+ export interface ICanaryScoresState {
26
+ successfulTouched: boolean;
27
+ unhealthyTouched: boolean;
28
+ }
29
+
30
+ export function CanaryScores(props: ICanaryScoresProps) {
31
+ const [successfulTouched, setSuccessfulTouched] = React.useState(!!props.successfulScore);
32
+ const [unhealthyTouched, setUnhealthyTouched] = React.useState(!!props.unhealthyScore);
33
+
34
+ const { successfulScore, unhealthyScore } = props;
35
+
36
+ const isExpression = (scoreValue: string): boolean => {
37
+ return isString(scoreValue) && scoreValue.includes('${');
38
+ };
39
+
40
+ const hasExpressions = isExpression(unhealthyScore) || isExpression(successfulScore);
41
+
42
+ let successful: number;
43
+ let unhealthy: number;
44
+ if (!hasExpressions) {
45
+ successful = Number(successfulScore);
46
+ unhealthy = Number(unhealthyScore);
47
+ }
48
+
49
+ const handleSuccessfulChange = (event: React.ChangeEvent<HTMLInputElement>): void => {
50
+ props.onChange({
51
+ successfulScore: event.target.value,
52
+ unhealthyScore,
53
+ });
54
+ };
55
+
56
+ const handleUnhealthyChange = (event: React.ChangeEvent<HTMLInputElement>): void => {
57
+ props.onChange({
58
+ successfulScore,
59
+ unhealthyScore: event.target.value,
60
+ });
61
+ };
62
+
63
+ const errors = {
64
+ successful: '',
65
+ unhealthy: '',
66
+ };
67
+
68
+ if (successfulTouched) {
69
+ if (successful <= 0) {
70
+ errors.successful = 'Must be positive';
71
+ }
72
+ if (successful >= 100) {
73
+ errors.successful = 'Must be less than 100';
74
+ }
75
+ }
76
+
77
+ if (unhealthyTouched) {
78
+ if (unhealthy <= 0) {
79
+ errors.unhealthy = 'Must be positive';
80
+ }
81
+ if (unhealthy >= 100) {
82
+ errors.unhealthy = 'Must be less than 100';
83
+ }
84
+ }
85
+
86
+ if (successfulTouched && unhealthyTouched) {
87
+ if (successful <= unhealthy) {
88
+ errors.successful = `Must be greater than ${props.unhealthyLabel}`;
89
+ }
90
+ }
91
+
92
+ const invalid = !(successful && unhealthy);
93
+ return (
94
+ <div>
95
+ {hasExpressions && (
96
+ <div className="form-group">
97
+ <div className="col-md-2 col-md-offset-1 sm-label-right">Canary Scores</div>
98
+ <div className="col-md-9 form-control-static">Expressions are currently being used for canary scores.</div>
99
+ </div>
100
+ )}
101
+ {!hasExpressions && (
102
+ <div className="canary-score">
103
+ <div className="form-group">
104
+ <div className="col-md-2 col-md-offset-1 sm-label-right">
105
+ {props.unhealthyLabel || 'Unhealthy Score '}
106
+ <HelpField id={props.unhealthyHelpFieldId || 'pipeline.config.canary.unhealthyScore'} />
107
+ </div>
108
+ <div className="col-md-2">
109
+ <input
110
+ type="number"
111
+ required={true}
112
+ disabled={props.disabled}
113
+ onBlur={() => setUnhealthyTouched(true)}
114
+ value={Number.isNaN(unhealthy) ? '' : unhealthy}
115
+ onChange={handleUnhealthyChange}
116
+ className={classNames('form-control', 'input-sm', {
117
+ 'ng-invalid': !!errors.unhealthy,
118
+ 'ng-invalid-validate-min': !!errors.unhealthy,
119
+ })}
120
+ />
121
+ {errors.unhealthy && <div className="error-message">{errors.unhealthy}</div>}
122
+ </div>
123
+ <div className="col-md-2 col-md-offset-1 sm-label-right">
124
+ {props.successfulLabel || 'Successful Score '}
125
+ <HelpField id={props.successfulHelpFieldId || 'pipeline.config.canary.successfulScore'} />
126
+ </div>
127
+ <div className="col-md-2">
128
+ <input
129
+ type="number"
130
+ required={true}
131
+ disabled={props.disabled}
132
+ onBlur={() => setSuccessfulTouched(true)}
133
+ value={Number.isNaN(successful) ? '' : successful}
134
+ onChange={handleSuccessfulChange}
135
+ className={classNames('form-control', 'input-sm', {
136
+ 'ng-invalid': !!errors.successful,
137
+ 'ng-invalid-validate-max': !!errors.successful,
138
+ })}
139
+ />
140
+ {errors.successful && <div className="error-message">{errors.successful}</div>}
141
+ </div>
142
+ </div>
143
+ <div className="row">
144
+ <div className="col-md-offset-1 col-md-10">
145
+ <div className="progress">
146
+ <div className="progress-bar progress-bar-danger" style={{ width: `${invalid ? 0 : unhealthy}%` }} />
147
+ <div
148
+ className="progress-bar progress-bar-warning"
149
+ style={{ width: `${invalid ? 0 : 100 - (unhealthy + (100 - successful))}%` }}
150
+ />
151
+ <div
152
+ className="progress-bar progress-bar-success"
153
+ style={{ width: `${invalid ? 0 : 100 - successful}%` }}
154
+ />
155
+ <div className="progress-bar progress-bar-warning" style={{ width: `${invalid ? 100 : 0}%` }} />
156
+ </div>
157
+ </div>
158
+ </div>
159
+ </div>
160
+ )}
161
+ </div>
162
+ );
163
+ }
@@ -0,0 +1,52 @@
1
+ import { AsyncRequestState } from 'kayenta/reducers/asyncRequest';
2
+ import * as React from 'react';
3
+
4
+ import { Spinner } from '@spinnaker/core';
5
+
6
+ export interface ILoadStatesProps {
7
+ state: AsyncRequestState;
8
+ }
9
+
10
+ /*
11
+ * Builder for rendering load states. Defaults to standard spinner during
12
+ * `Requesting` state.
13
+ * */
14
+ export default class LoadStatesBuilder {
15
+ private requesting: JSX.Element;
16
+ private fulfilled: JSX.Element;
17
+ private failed: JSX.Element;
18
+
19
+ public onRequesting(requesting: JSX.Element): LoadStatesBuilder {
20
+ this.requesting = requesting;
21
+ return this;
22
+ }
23
+
24
+ public onFulfilled(fulfilled: JSX.Element): LoadStatesBuilder {
25
+ this.fulfilled = fulfilled;
26
+ return this;
27
+ }
28
+
29
+ public onFailed(failed: JSX.Element): LoadStatesBuilder {
30
+ this.failed = failed;
31
+ return this;
32
+ }
33
+
34
+ public build(): React.SFC<ILoadStatesProps> {
35
+ return ({ state }: ILoadStatesProps) => {
36
+ switch (state) {
37
+ case AsyncRequestState.Requesting:
38
+ return (
39
+ this.requesting || (
40
+ <div className="horizontal center middle spinner-container">
41
+ <Spinner />
42
+ </div>
43
+ )
44
+ );
45
+ case AsyncRequestState.Fulfilled:
46
+ return this.fulfilled;
47
+ case AsyncRequestState.Failed:
48
+ return this.failed;
49
+ }
50
+ };
51
+ }
52
+ }
@@ -0,0 +1,57 @@
1
+ export interface ICanaryConfig {
2
+ applications: string[];
3
+ id?: string;
4
+ createdTimestamp?: number;
5
+ updatedTimestamp?: number;
6
+ createdTimestampIso?: string;
7
+ updatedTimestampIso?: string;
8
+ isNew?: boolean;
9
+ name: string;
10
+ description: string;
11
+ configVersion: string;
12
+ metrics: ICanaryMetricConfig[];
13
+ templates: { [key: string]: string };
14
+ classifier: ICanaryClassifierConfig;
15
+ judge: ICanaryJudgeConfig;
16
+ }
17
+
18
+ export interface ICanaryMetricConfig<T extends ICanaryMetricSetQueryConfig = any> {
19
+ id: string;
20
+ name: string;
21
+ query: T;
22
+ groups: string[];
23
+ analysisConfigurations: {
24
+ [key: string]: any;
25
+ effectSize?: ICanaryMetricEffectSizeConfig;
26
+ };
27
+ scopeName: string;
28
+ isNew?: boolean;
29
+ }
30
+
31
+ export interface ICanaryMetricSetQueryConfig {
32
+ type: string;
33
+ serviceType: string;
34
+ customInlineTemplate?: string;
35
+ customFilterTemplate?: string;
36
+ }
37
+
38
+ export interface IGroupWeights {
39
+ [group: string]: number;
40
+ }
41
+
42
+ export interface ICanaryClassifierConfig {
43
+ groupWeights: IGroupWeights;
44
+ }
45
+
46
+ export interface ICanaryMetricEffectSizeConfig {
47
+ allowedIncrease?: number;
48
+ allowedDecrease?: number;
49
+ criticalIncrease?: number;
50
+ criticalDecrease?: number;
51
+ measure?: string;
52
+ }
53
+
54
+ export interface ICanaryJudgeConfig {
55
+ name: string;
56
+ judgeConfigurations: { [key: string]: any };
57
+ }
@@ -0,0 +1,7 @@
1
+ export interface ICanaryConfigSummary {
2
+ updatedTimestampIso: string;
3
+ name: string;
4
+ id: string;
5
+ updatedTimestamp: number;
6
+ applications: string[];
7
+ }
@@ -0,0 +1,3 @@
1
+ export interface ICanaryConfigUpdateResponse {
2
+ canaryConfigId: string;
3
+ }
@@ -0,0 +1,72 @@
1
+ import { ICanaryConfig } from './ICanaryConfig';
2
+ import { ICanaryJudgeResult } from './ICanaryJudgeResult';
3
+ import { ICanaryScoreThresholds } from './ICanaryScoreThresholds';
4
+
5
+ export const CANARY_EXECUTION_NO_PIPELINE_STATUS = 'no-parent-pipeline-execution';
6
+
7
+ export interface ICanaryExecutionStatusResult {
8
+ id: string; // Added by Deck on load.
9
+ canaryConfigId: string;
10
+ complete: boolean;
11
+ status: string;
12
+ result: ICanaryResult;
13
+ exception?: ICanaryExecutionException;
14
+ stageStatus: { [key: string]: string };
15
+ startTimeIso: string;
16
+ endTimeIso: string;
17
+ application: string;
18
+ pipelineId: string;
19
+ parentPipelineExecutionId: string;
20
+ metricSetPairListId: string;
21
+ metricsAccountName: string;
22
+ config: ICanaryConfig;
23
+ canaryExecutionRequest: ICanaryExecutionRequest;
24
+ storageAccountName: string;
25
+ }
26
+
27
+ export interface ICanaryExecutionException {
28
+ exceptionType: string;
29
+ operation: string;
30
+ details: { [key: string]: any };
31
+ timestamp: string;
32
+ }
33
+
34
+ export interface ICanaryResult {
35
+ judgeResult: ICanaryJudgeResult;
36
+ }
37
+
38
+ export interface ICanaryExecutionRequest {
39
+ thresholds: ICanaryScoreThresholds;
40
+ scopes: {
41
+ [scopeName: string]: ICanaryScopePair;
42
+ };
43
+ }
44
+
45
+ export interface ICanaryExecutionRequestParams {
46
+ application?: string;
47
+ parentPipelineExecutionId?: string;
48
+ metricsAccountName?: string;
49
+ storageAccountName?: string;
50
+ configurationAccountName?: string;
51
+ }
52
+
53
+ export interface ICanaryExecutionResponse {
54
+ canaryExecutionId: string;
55
+ }
56
+
57
+ export type ICanaryScopesByName = ICanaryExecutionRequest['scopes'];
58
+
59
+ export interface ICanaryScopePair {
60
+ controlScope: ICanaryScope;
61
+ experimentScope: ICanaryScope;
62
+ }
63
+
64
+ export interface ICanaryScope {
65
+ scope: string;
66
+ location: string;
67
+ start: string;
68
+ end: string;
69
+ step: number;
70
+ extendedScopeParams?: { [param: string]: string };
71
+ resourceType?: string;
72
+ }
@@ -0,0 +1,51 @@
1
+ import { MetricClassificationLabel } from './MetricClassificationLabel';
2
+ import { ScoreClassificationLabel } from './ScoreClassificationLabel';
3
+
4
+ export interface ICanaryJudgeResult {
5
+ judgeName: string;
6
+ results: ICanaryAnalysisResult[];
7
+ groupScores: ICanaryJudgeGroupScore[];
8
+ score: ICanaryJudgeScore;
9
+ }
10
+
11
+ export interface ICanaryAnalysisResult {
12
+ name: string;
13
+ id: string;
14
+ tags: { [key: string]: string };
15
+ classification: MetricClassificationLabel;
16
+ classificationReason: string;
17
+ groups: string[];
18
+ // TODO: is every judge going to implement the stats interface for
19
+ // experiment and control metadata (which would make the reporting UI much easier),
20
+ // or just the Netflix judge?
21
+ experimentMetadata: {
22
+ [key: string]: any;
23
+ stats: ICanaryAnalysisResultsStats;
24
+ };
25
+ controlMetadata: {
26
+ [key: string]: any;
27
+ stats: ICanaryAnalysisResultsStats;
28
+ };
29
+ resultMetadata?: { [key: string]: any };
30
+ }
31
+
32
+ export interface ICanaryAnalysisResultsStats {
33
+ count: number;
34
+ mean: number;
35
+ min: number;
36
+ max: number;
37
+ median: number;
38
+ }
39
+
40
+ export interface ICanaryJudgeGroupScore {
41
+ name: string;
42
+ score: number;
43
+ classification: string;
44
+ classificationReason: string;
45
+ }
46
+
47
+ export interface ICanaryJudgeScore {
48
+ score: number;
49
+ classification: ScoreClassificationLabel;
50
+ classificationReason: string;
51
+ }
@@ -0,0 +1,5 @@
1
+ export interface ICanaryJudgeResultSummary {
2
+ updatedTimestampIso: string;
3
+ name: string;
4
+ updatedTimestamp: number;
5
+ }
@@ -0,0 +1,4 @@
1
+ export interface ICanaryScoreThresholds {
2
+ pass: number;
3
+ marginal: number;
4
+ }
@@ -0,0 +1,5 @@
1
+ export interface IJudge {
2
+ // Easy to imagine that this could be extended soon, e.g., with a description or other metadata.
3
+ name: string;
4
+ visible: boolean;
5
+ }
@@ -0,0 +1,14 @@
1
+ export interface IKayentaAccount {
2
+ name: string;
3
+ type: string;
4
+ supportedTypes: KayentaAccountType[];
5
+ metricsStoreType?: string;
6
+ locations?: string[];
7
+ recommendedLocations?: string[];
8
+ }
9
+
10
+ export enum KayentaAccountType {
11
+ MetricsStore = 'METRICS_STORE',
12
+ ObjectStore = 'OBJECT_STORE',
13
+ ConfigurationStore = 'CONFIGURATION_STORE',
14
+ }