@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,80 @@
1
+ import * as Creators from 'kayenta/actions/creators';
2
+ import { CanarySettings } from 'kayenta/canary.settings';
3
+ import { ICanaryConfig } from 'kayenta/domain/ICanaryConfig';
4
+ import { DISABLE_EDIT_CONFIG, DisableableInput } from 'kayenta/layout/disableable';
5
+ import { ICanaryState } from 'kayenta/reducers';
6
+ import { mapStateToConfig } from 'kayenta/service/canaryConfig.service';
7
+ import { get, isNumber } from 'lodash';
8
+ import * as React from 'react';
9
+ import { connect } from 'react-redux';
10
+ import { Action } from 'redux';
11
+
12
+ import FormRow from '../layout/formRow';
13
+
14
+ export interface IGroupWeightOwnProps {
15
+ group: string;
16
+ }
17
+
18
+ interface IGroupWeightStateProps {
19
+ config: ICanaryConfig;
20
+ }
21
+
22
+ interface IGroupWeightDispatchProps {
23
+ handleInputChange: (event: React.ChangeEvent<HTMLInputElement>) => void;
24
+ }
25
+
26
+ /*
27
+ * Component for configuring a group weight.
28
+ * */
29
+ function GroupWeight({
30
+ group,
31
+ config,
32
+ handleInputChange,
33
+ }: IGroupWeightOwnProps & IGroupWeightStateProps & IGroupWeightDispatchProps) {
34
+ const groupWeight = getGroupWeights(config)[group];
35
+ return (
36
+ <FormRow label={group} inputOnly={true}>
37
+ <DisableableInput
38
+ type="number"
39
+ value={isNumber(groupWeight) ? groupWeight : ''}
40
+ onChange={handleInputChange}
41
+ min={0}
42
+ max={100}
43
+ disabled={CanarySettings.disableConfigEdit}
44
+ disabledStateKeys={[DISABLE_EDIT_CONFIG]}
45
+ />
46
+ </FormRow>
47
+ );
48
+ }
49
+
50
+ function getGroupWeights(config: ICanaryConfig): { [key: string]: number } {
51
+ return get(config, 'classifier.groupWeights', {});
52
+ }
53
+
54
+ function mapStateToProps(
55
+ state: ICanaryState,
56
+ ownProps: IGroupWeightOwnProps,
57
+ ): IGroupWeightOwnProps & IGroupWeightStateProps {
58
+ return {
59
+ ...ownProps,
60
+ config: mapStateToConfig(state),
61
+ };
62
+ }
63
+
64
+ function mapDispatchToProps(
65
+ dispatch: (action: Action & any) => void,
66
+ { group }: IGroupWeightOwnProps,
67
+ ): IGroupWeightDispatchProps {
68
+ return {
69
+ handleInputChange: (event: React.ChangeEvent<HTMLInputElement>) => {
70
+ dispatch(
71
+ Creators.updateGroupWeight({
72
+ group,
73
+ weight: event.target.value ? parseInt(event.target.value, 10) : null,
74
+ }),
75
+ );
76
+ },
77
+ };
78
+ }
79
+
80
+ export default connect(mapStateToProps, mapDispatchToProps)(GroupWeight);
@@ -0,0 +1,38 @@
1
+ import { flatMap, uniq } from 'lodash';
2
+ import * as React from 'react';
3
+ import { connect } from 'react-redux';
4
+
5
+ import GroupWeight from './groupWeight';
6
+ import { ICanaryState } from '../reducers/index';
7
+ import { mapStateToConfig } from '../service/canaryConfig.service';
8
+
9
+ interface IGroupWeightsStateProps {
10
+ groups: string[];
11
+ }
12
+
13
+ /*
14
+ * Component for rendering list of group weight configurers.
15
+ */
16
+ function GroupWeights({ groups }: IGroupWeightsStateProps) {
17
+ const hasGroups = groups.length > 0;
18
+ return (
19
+ <section>
20
+ {hasGroups ? (
21
+ groups.map((group) => <GroupWeight key={group} group={group} />)
22
+ ) : (
23
+ <p key="no-groups">You have not configured any grouped metrics.</p>
24
+ )}
25
+ </section>
26
+ );
27
+ }
28
+
29
+ function mapStateToProps(state: ICanaryState): IGroupWeightsStateProps {
30
+ const config = mapStateToConfig(state);
31
+ const metrics = config ? config.metrics : [];
32
+ const groups = uniq(flatMap(metrics, (metric) => metric.groups || []));
33
+ return {
34
+ groups,
35
+ };
36
+ }
37
+
38
+ export default connect(mapStateToProps)(GroupWeights);
@@ -0,0 +1,42 @@
1
+ import { mount } from 'enzyme';
2
+ import { identity } from 'lodash';
3
+ import * as React from 'react';
4
+ import { Provider } from 'react-redux';
5
+ import { createMockStore } from 'redux-test-utils';
6
+
7
+ import { noop, ValidationMessage } from '@spinnaker/core';
8
+
9
+ import { IInlineTemplateEditorProps, InlineTemplateEditor } from './inlineTemplateEditor';
10
+ import { DisableableTextarea } from '../layout/disableable';
11
+
12
+ describe('<InlineTemplateEditor />', () => {
13
+ const buildComponent = (props: IInlineTemplateEditorProps) =>
14
+ mount(
15
+ <Provider store={createMockStore()}>
16
+ <InlineTemplateEditor {...props} />
17
+ </Provider>,
18
+ ).find(InlineTemplateEditor);
19
+
20
+ it('renders a textarea with template value', () => {
21
+ const component = buildComponent({
22
+ templateValue: 'metadata.user_labels."app"="${scope}"',
23
+ transformValueForSave: identity,
24
+ editTemplateValue: noop,
25
+ });
26
+ expect(component.find(DisableableTextarea).first().props().value).toEqual('metadata.user_labels."app"="${scope}"');
27
+ });
28
+ it('renders an error for empty input', () => {
29
+ let component = buildComponent({
30
+ templateValue: 'metadata.user_labels."app"="${scope}"',
31
+ transformValueForSave: identity,
32
+ editTemplateValue: noop,
33
+ });
34
+ expect(component.find(ValidationMessage).length).toEqual(0);
35
+ component = buildComponent({
36
+ templateValue: '',
37
+ transformValueForSave: identity,
38
+ editTemplateValue: noop,
39
+ });
40
+ expect(component.find(ValidationMessage).props().message).toEqual('Template is required');
41
+ });
42
+ });
@@ -0,0 +1,61 @@
1
+ import * as Creators from 'kayenta/actions/creators';
2
+ import { CanarySettings } from 'kayenta/canary.settings';
3
+ import { DISABLE_EDIT_CONFIG, DisableableTextarea } from 'kayenta/layout/disableable';
4
+ import FormRow from 'kayenta/layout/formRow';
5
+ import { ICanaryState } from 'kayenta/reducers';
6
+ import {
7
+ inlineTemplateValueSelector,
8
+ transformInlineTemplateForSave,
9
+ } from 'kayenta/selectors/filterTemplatesSelectors';
10
+ import { isEmpty } from 'lodash';
11
+ import * as React from 'react';
12
+ import { connect } from 'react-redux';
13
+
14
+ interface IInlineTemplateEditorStateProps {
15
+ templateValue: string;
16
+ transformValueForSave: (value: string) => string;
17
+ }
18
+
19
+ interface IInlineTemplateEditorDispatchProps {
20
+ editTemplateValue: (value: string) => void;
21
+ }
22
+
23
+ export type IInlineTemplateEditorProps = IInlineTemplateEditorStateProps & IInlineTemplateEditorDispatchProps;
24
+
25
+ export function InlineTemplateEditor({
26
+ editTemplateValue,
27
+ templateValue,
28
+ transformValueForSave,
29
+ }: IInlineTemplateEditorProps) {
30
+ return (
31
+ <FormRow
32
+ label="Template"
33
+ inputOnly={true}
34
+ helpId="canary.config.filterTemplate"
35
+ error={isEmpty(templateValue) && 'Template is required'}
36
+ >
37
+ <DisableableTextarea
38
+ className="template-editor-textarea"
39
+ disabledStateKeys={[DISABLE_EDIT_CONFIG]}
40
+ disabled={CanarySettings.disableConfigEdit}
41
+ onChange={(e: any) => editTemplateValue(transformValueForSave(e.target.value))}
42
+ value={templateValue}
43
+ />
44
+ </FormRow>
45
+ );
46
+ }
47
+
48
+ function mapStateToProps(state: ICanaryState): IInlineTemplateEditorStateProps {
49
+ return {
50
+ templateValue: inlineTemplateValueSelector(state),
51
+ transformValueForSave: transformInlineTemplateForSave(state),
52
+ };
53
+ }
54
+
55
+ function mapDispatchToProps(dispatch: any): IInlineTemplateEditorDispatchProps {
56
+ return {
57
+ editTemplateValue: (value: string) => dispatch(Creators.editInlineTemplate({ value })),
58
+ };
59
+ }
60
+
61
+ export default connect(mapStateToProps, mapDispatchToProps)(InlineTemplateEditor);
@@ -0,0 +1,82 @@
1
+ import * as Creators from 'kayenta/actions/creators';
2
+ import { CanarySettings } from 'kayenta/canary.settings';
3
+ import { DISABLE_EDIT_CONFIG, DisableableInput, DisableableReactSelect } from 'kayenta/layout/disableable';
4
+ import FormRow from 'kayenta/layout/formRow';
5
+ import { ICanaryState } from 'kayenta/reducers';
6
+ import * as React from 'react';
7
+ import { connect } from 'react-redux';
8
+ import { Option } from 'react-select';
9
+ import { Action } from 'redux';
10
+
11
+ interface IJudgeSelectStateProps {
12
+ judgeOptions: Option[];
13
+ selectedJudge: string;
14
+ renderState: JudgeSelectRenderState;
15
+ }
16
+
17
+ interface IJudgeSelectDispatchProps {
18
+ handleJudgeSelect: (option: Option) => void;
19
+ }
20
+
21
+ export enum JudgeSelectRenderState {
22
+ Multiple,
23
+ Single,
24
+ None,
25
+ }
26
+
27
+ /*
28
+ * Select field for picking canary judge.
29
+ */
30
+ function JudgeSelect({
31
+ judgeOptions,
32
+ selectedJudge,
33
+ handleJudgeSelect,
34
+ renderState,
35
+ }: IJudgeSelectStateProps & IJudgeSelectDispatchProps) {
36
+ switch (renderState) {
37
+ case JudgeSelectRenderState.Multiple:
38
+ return (
39
+ <FormRow>
40
+ <DisableableReactSelect
41
+ value={selectedJudge}
42
+ options={judgeOptions}
43
+ clearable={false}
44
+ onChange={handleJudgeSelect}
45
+ disabled={CanarySettings.disableConfigEdit}
46
+ disabledStateKeys={[DISABLE_EDIT_CONFIG]}
47
+ />
48
+ </FormRow>
49
+ );
50
+ case JudgeSelectRenderState.Single:
51
+ return (
52
+ <FormRow>
53
+ <DisableableInput
54
+ type="text"
55
+ value={selectedJudge}
56
+ disabled={true}
57
+ disabledStateKeys={[DISABLE_EDIT_CONFIG]}
58
+ />
59
+ </FormRow>
60
+ );
61
+ case JudgeSelectRenderState.None:
62
+ return null;
63
+ }
64
+ }
65
+
66
+ function mapStateToProps(state: ICanaryState): IJudgeSelectStateProps {
67
+ return {
68
+ judgeOptions: (state.data.judges || []).map((judge) => ({ value: judge.name, label: judge.name })),
69
+ selectedJudge: state.selectedConfig.judge.judgeConfig.name,
70
+ renderState: state.selectedConfig.judge.renderState,
71
+ };
72
+ }
73
+
74
+ function mapDispatchToProps(dispatch: (action: Action & any) => void): IJudgeSelectDispatchProps {
75
+ return {
76
+ handleJudgeSelect: (option: Option) => {
77
+ dispatch(Creators.selectJudgeName({ judge: { name: option.value as string } }));
78
+ },
79
+ };
80
+ }
81
+
82
+ export default connect(mapStateToProps, mapDispatchToProps)(JudgeSelect);
@@ -0,0 +1,30 @@
1
+ import { ICanaryMetricConfig } from 'kayenta/domain/ICanaryConfig';
2
+ import metricStoreConfigService from 'kayenta/metricStore/metricStoreConfig.service';
3
+ import { ICanaryState } from 'kayenta/reducers';
4
+ import * as React from 'react';
5
+ import { connect } from 'react-redux';
6
+
7
+ interface IMetricConfigurerDelegatorStateProps {
8
+ editingMetric: ICanaryMetricConfig;
9
+ }
10
+
11
+ /*
12
+ * Should find and render the appropriate metric configurer for a given metric store.
13
+ * */
14
+ function MetricConfigurerDelegator({ editingMetric }: IMetricConfigurerDelegatorStateProps) {
15
+ const config = metricStoreConfigService.getDelegate(editingMetric.query.type);
16
+ if (config && config.metricConfigurer) {
17
+ const MetricConfigurer = config.metricConfigurer;
18
+ return <MetricConfigurer />;
19
+ } else {
20
+ return <p>Metric configuration has not been implemented for {editingMetric.query.type}.</p>;
21
+ }
22
+ }
23
+
24
+ function mapStateToProps(state: ICanaryState): IMetricConfigurerDelegatorStateProps {
25
+ return {
26
+ editingMetric: state.selectedConfig.editingMetric,
27
+ };
28
+ }
29
+
30
+ export default connect(mapStateToProps)(MetricConfigurerDelegator);
@@ -0,0 +1,21 @@
1
+ .metrics-action-buttons {
2
+ .link {
3
+ padding: 0 12px;
4
+ }
5
+ }
6
+
7
+ .metric-filter-options {
8
+ label {
9
+ font-weight: 600;
10
+ margin: 6px;
11
+ padding: 6px 12px;
12
+ border: 1px solid var(--color-divider);
13
+ }
14
+ }
15
+
16
+ .metric-fail-on-icon {
17
+ color: var(--color-warning);
18
+ &.critical {
19
+ color: var(--color-danger);
20
+ }
21
+ }
@@ -0,0 +1,215 @@
1
+ import classNames from 'classnames';
2
+ import * as Creators from 'kayenta/actions/creators';
3
+ import { CanarySettings } from 'kayenta/canary.settings';
4
+ import { ICanaryMetricConfig } from 'kayenta/domain';
5
+ import { DISABLE_EDIT_CONFIG, DisableableButton } from 'kayenta/layout/disableable';
6
+ import { ITableColumn, NativeTable } from 'kayenta/layout/table';
7
+ import { ICanaryState } from 'kayenta/reducers';
8
+ import { cloneDeep } from 'lodash';
9
+ import * as React from 'react';
10
+ import { connect } from 'react-redux';
11
+ import { Action } from 'redux';
12
+
13
+ import { Tooltip } from '@spinnaker/core';
14
+
15
+ import ChangeMetricGroupModal from './changeMetricGroupModal';
16
+
17
+ import './metricList.less';
18
+
19
+ interface IMetricListStateProps {
20
+ selectedGroup: string;
21
+ metrics: ICanaryMetricConfig[];
22
+ showGroups: boolean;
23
+ changingGroupMetric: ICanaryMetricConfig;
24
+ groupList: string[];
25
+ metricStore: string;
26
+ disableEdit: boolean;
27
+ }
28
+
29
+ interface IMetricListDispatchProps {
30
+ addMetric: (event: any) => void;
31
+ editMetric: (event: any) => void;
32
+ copyMetric: (metric: ICanaryMetricConfig) => void;
33
+ removeMetric: (event: any) => void;
34
+ openChangeMetricGroupModal: (event: any) => void;
35
+ }
36
+
37
+ function FailOn({ metric }: { metric: ICanaryMetricConfig }) {
38
+ const direction = metric.analysisConfigurations?.canary?.direction;
39
+ const isCritical = metric.analysisConfigurations?.canary?.critical;
40
+ const tooltipSuffix = isCritical ? '(critical — if this metric fails, the entire canary will fail)' : '';
41
+ const classes = classNames('metric-fail-on-icon', 'fas', 'sp-margin-xs-right', { critical: isCritical });
42
+ if (direction === 'decrease') {
43
+ return (
44
+ <Tooltip value={`decrease ${tooltipSuffix}`}>
45
+ <i className={`fa-caret-square-down ${classes}`} />
46
+ </Tooltip>
47
+ );
48
+ }
49
+ if (direction === 'increase') {
50
+ return (
51
+ <Tooltip value={`increase ${tooltipSuffix}`}>
52
+ <i className={`fa-caret-square-up ${classes}`} />
53
+ </Tooltip>
54
+ );
55
+ }
56
+ return (
57
+ <Tooltip value={`increase OR decrease ${tooltipSuffix}`}>
58
+ <span>
59
+ <i className={`fa-caret-square-down ${classes}`} />
60
+ <i className={`fa-caret-square-up ${classes}`} />
61
+ </span>
62
+ </Tooltip>
63
+ );
64
+ }
65
+
66
+ /*
67
+ * Configures an entire list of metrics.
68
+ */
69
+ function MetricList({
70
+ metrics,
71
+ groupList,
72
+ selectedGroup,
73
+ showGroups,
74
+ addMetric,
75
+ editMetric,
76
+ copyMetric,
77
+ removeMetric,
78
+ changingGroupMetric,
79
+ openChangeMetricGroupModal,
80
+ metricStore,
81
+ disableEdit,
82
+ }: IMetricListStateProps & IMetricListDispatchProps) {
83
+ const columns: Array<ITableColumn<ICanaryMetricConfig>> = [
84
+ {
85
+ label: 'Metric Name',
86
+ getContent: (metric) => <span>{metric.name || '(new)'}</span>,
87
+ },
88
+ {
89
+ label: 'Fail On',
90
+ getContent: (metric) => <FailOn metric={metric} />,
91
+ },
92
+ {
93
+ label: 'Groups',
94
+ getContent: (metric) => <span>{metric.groups.join(', ')}</span>,
95
+ hide: () => !showGroups,
96
+ },
97
+ {
98
+ getContent: (metric) => (
99
+ <div className="horizontal pull-right metrics-action-buttons">
100
+ <button className="link" data-id={metric.id} onClick={editMetric}>
101
+ {disableEdit || CanarySettings.disableConfigEdit ? 'View' : 'Edit'}
102
+ </button>
103
+ <button
104
+ className="link"
105
+ data-id={metric.id}
106
+ disabled={disableEdit || CanarySettings.disableConfigEdit}
107
+ onClick={openChangeMetricGroupModal}
108
+ >
109
+ Move Group
110
+ </button>
111
+ <button
112
+ className="link"
113
+ data-id={metric.id}
114
+ disabled={disableEdit || CanarySettings.disableConfigEdit}
115
+ onClick={() => copyMetric(metric)}
116
+ >
117
+ Copy
118
+ </button>
119
+ <button
120
+ className="link"
121
+ data-id={metric.id}
122
+ disabled={disableEdit || CanarySettings.disableConfigEdit}
123
+ onClick={removeMetric}
124
+ >
125
+ Delete
126
+ </button>
127
+ </div>
128
+ ),
129
+ },
130
+ ];
131
+
132
+ return (
133
+ <>
134
+ <NativeTable columns={columns} rows={metrics} rowKey={(metric) => metric.id} className="header-white" />
135
+ {!metrics.length && selectedGroup ? (
136
+ <p>
137
+ This group is empty! The group will be not be present the next time the config is loaded unless it is saved
138
+ with at least one metric in it.
139
+ </p>
140
+ ) : null}
141
+ {changingGroupMetric && <ChangeMetricGroupModal metric={changingGroupMetric} />}
142
+ <DisableableButton
143
+ className="passive self-left"
144
+ data-group={selectedGroup}
145
+ data-default={groupList[0]}
146
+ data-metric-store={metricStore}
147
+ onClick={addMetric}
148
+ disabled={CanarySettings.disableConfigEdit}
149
+ disabledStateKeys={[DISABLE_EDIT_CONFIG]}
150
+ >
151
+ Add Metric
152
+ </DisableableButton>
153
+ </>
154
+ );
155
+ }
156
+
157
+ function mapStateToProps(state: ICanaryState): IMetricListStateProps {
158
+ const selectedGroup = state.selectedConfig.group.selected;
159
+ const metricList = state.selectedConfig.metricList;
160
+
161
+ const filter = selectedGroup ? (metric: ICanaryMetricConfig) => metric.groups.includes(selectedGroup) : () => true;
162
+ return {
163
+ selectedGroup,
164
+ groupList: state.selectedConfig.group.list,
165
+ metrics: metricList.filter(filter),
166
+ showGroups: !selectedGroup || metricList.filter(filter).some((metric) => metric.groups.length > 1),
167
+ changingGroupMetric: state.selectedConfig.metricList.find(
168
+ (m) => m.id === state.selectedConfig.changeMetricGroup.metric,
169
+ ),
170
+ metricStore: state.selectedConfig.selectedStore,
171
+ disableEdit: state.app.disableConfigEdit || CanarySettings.disableConfigEdit,
172
+ };
173
+ }
174
+
175
+ function mapDispatchToProps(dispatch: (action: Action & any) => void): IMetricListDispatchProps {
176
+ return {
177
+ addMetric: (event: any) => {
178
+ const group = event.target.dataset.group || event.target.dataset.default;
179
+ dispatch(
180
+ Creators.addMetric({
181
+ metric: {
182
+ // TODO: need to block saving an invalid name
183
+ // TODO: for Atlas metrics, attempt to gather name when query changes
184
+ id: '[new]',
185
+ analysisConfigurations: {},
186
+ name: '',
187
+ query: {
188
+ type: event.target.dataset.metricStore,
189
+ serviceType: event.target.dataset.metricStore,
190
+ },
191
+ groups: group ? [group] : [],
192
+ scopeName: 'default', // scopeName always defaults to `default` and is not configurable from the UI
193
+ isNew: true,
194
+ },
195
+ }),
196
+ );
197
+ },
198
+ copyMetric: (metric: ICanaryMetricConfig) => {
199
+ const metricCopy: ICanaryMetricConfig = cloneDeep(metric);
200
+ metricCopy.id = '[new]';
201
+ metricCopy.isNew = true;
202
+ metricCopy.name = '';
203
+ dispatch(Creators.addMetric({ metric: metricCopy }));
204
+ },
205
+ editMetric: (event: any) => {
206
+ dispatch(Creators.editMetricBegin({ id: event.target.dataset.id }));
207
+ },
208
+ removeMetric: (event: any) => {
209
+ dispatch(Creators.removeMetric({ id: event.target.dataset.id }));
210
+ },
211
+ openChangeMetricGroupModal: (event: any) => dispatch(Creators.changeMetricGroup({ id: event.target.dataset.id })),
212
+ };
213
+ }
214
+
215
+ export default connect(mapStateToProps, mapDispatchToProps)(MetricList);
@@ -0,0 +1,66 @@
1
+ import * as Creators from 'kayenta/actions/creators';
2
+ import { CanarySettings } from 'kayenta/canary.settings';
3
+ import { KayentaAccountType } from 'kayenta/domain';
4
+ import { DISABLE_EDIT_CONFIG, DisableableSelect } from 'kayenta/layout/disableable';
5
+ import FormRow from 'kayenta/layout/formRow';
6
+ import { ICanaryState } from 'kayenta/reducers';
7
+ import { chain } from 'lodash';
8
+ import * as React from 'react';
9
+ import { connect, Dispatch } from 'react-redux';
10
+
11
+ interface IMetricStoreSelectorStateProps {
12
+ stores: string[];
13
+ selectedStore: string;
14
+ }
15
+
16
+ interface IMetricStoreSelectorDispatchProps {
17
+ select: (event: any) => void;
18
+ }
19
+
20
+ const MetricStoreSelector = ({
21
+ stores,
22
+ selectedStore,
23
+ select,
24
+ }: IMetricStoreSelectorDispatchProps & IMetricStoreSelectorStateProps) => {
25
+ if (stores.length < 2) {
26
+ return null;
27
+ }
28
+
29
+ return (
30
+ <FormRow label="Metric Store" inputOnly={true}>
31
+ <DisableableSelect
32
+ value={selectedStore || ''}
33
+ onChange={select}
34
+ className="form-control input-sm"
35
+ disabled={CanarySettings.disableConfigEdit}
36
+ disabledStateKeys={[DISABLE_EDIT_CONFIG]}
37
+ >
38
+ {stores.map((s) => (
39
+ <option key={s} value={s}>
40
+ {s}
41
+ </option>
42
+ ))}
43
+ </DisableableSelect>
44
+ </FormRow>
45
+ );
46
+ };
47
+
48
+ const mapStateToProps = (state: ICanaryState): IMetricStoreSelectorStateProps => {
49
+ return {
50
+ stores: chain(state.data.kayentaAccounts.data)
51
+ .filter((account) => account.supportedTypes.includes(KayentaAccountType.MetricsStore))
52
+ .map((account) => account.metricsStoreType || account.type)
53
+ .uniq()
54
+ .sort()
55
+ .valueOf(),
56
+ selectedStore: state.selectedConfig.selectedStore,
57
+ };
58
+ };
59
+
60
+ const mapDispatchToProps = (dispatch: Dispatch<ICanaryState>): IMetricStoreSelectorDispatchProps => {
61
+ return {
62
+ select: (event: any) => dispatch(Creators.selectMetricStore({ store: event.target.value })),
63
+ };
64
+ };
65
+
66
+ export default connect(mapStateToProps, mapDispatchToProps)(MetricStoreSelector);