@rancher/shell 0.3.0 → 0.3.1

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 (322) hide show
  1. package/assets/styles/global/_button.scss +5 -1
  2. package/assets/styles/global/_columns.scss +4 -0
  3. package/assets/styles/global/_layout.scss +1 -2
  4. package/assets/styles/global/_select.scss +1 -4
  5. package/assets/styles/themes/_dark.scss +4 -4
  6. package/assets/styles/themes/_light.scss +4 -3
  7. package/assets/styles/themes/_suse.scss +1 -1
  8. package/assets/styles/vendor/vue-select.scss +4 -3
  9. package/assets/translations/en-us.yaml +669 -73
  10. package/assets/translations/zh-hans.yaml +547 -165
  11. package/chart/monitoring/steps/uninstall-v1.vue +2 -2
  12. package/cloud-credential/azure.vue +23 -0
  13. package/cloud-credential/harvester.vue +25 -62
  14. package/cloud-credential/pnap.vue +80 -0
  15. package/components/.DS_Store +0 -0
  16. package/components/AdvancedSection.vue +9 -2
  17. package/components/Alert.vue +2 -2
  18. package/components/ButtonDropdown.vue +0 -2
  19. package/components/ButtonGroup.vue +1 -0
  20. package/components/CollapsibleCard.vue +0 -1
  21. package/components/CruResource.vue +41 -4
  22. package/components/DetailTop.vue +58 -3
  23. package/components/DisableAuthProviderModal.vue +106 -0
  24. package/{rancher-components/components/Utils/DraggableZone → components}/DraggableZone.vue +0 -0
  25. package/components/ExplorerMembers.vue +253 -30
  26. package/components/ExplorerProjectsNamespaces.vue +77 -33
  27. package/components/GrowlManager.vue +3 -3
  28. package/components/IconOrSvg.vue +149 -0
  29. package/components/LogItem.vue +69 -0
  30. package/components/PodSecurityAdmission.vue +302 -0
  31. package/components/PromptModal.vue +1 -0
  32. package/components/ResourceDetail/Masthead.vue +54 -2
  33. package/components/ResourceDetail/index.vue +12 -5
  34. package/components/ResourceList/Masthead.vue +11 -1
  35. package/components/ResourceList/ResourceLoadingIndicator.vue +12 -2
  36. package/components/ResourceList/index.vue +53 -12
  37. package/components/ResourceList/resource-list.config.js +7 -0
  38. package/components/ResourceTable.vue +31 -6
  39. package/components/SimpleBox.vue +1 -1
  40. package/components/SortableTable/THead.vue +15 -5
  41. package/components/SortableTable/index.vue +21 -10
  42. package/components/Tabbed/index.vue +20 -15
  43. package/components/__tests__/.DS_Store +0 -0
  44. package/components/__tests__/AsyncButton.test.ts +140 -0
  45. package/components/__tests__/BackLink.test.ts +33 -0
  46. package/components/__tests__/ButtonGroup.test.ts +124 -0
  47. package/components/__tests__/ClusterBadge.test.ts +32 -0
  48. package/components/__tests__/CollapsibleCard.test.ts +64 -0
  49. package/components/__tests__/ConsumptionGauge.test.ts +88 -0
  50. package/components/__tests__/CruResource.test.ts +3 -2
  51. package/components/__tests__/FixedBanner.test.ts +129 -0
  52. package/components/__tests__/GrowlManager.test.ts +147 -0
  53. package/components/__tests__/NamespaceFilter.test.ts +33 -25
  54. package/components/__tests__/PercentageBar.test.ts +32 -0
  55. package/components/__tests__/PodSecurityAdmission.test.ts +398 -0
  56. package/components/auth/AuthBanner.vue +20 -10
  57. package/components/auth/RoleDetailEdit.vue +26 -17
  58. package/components/auth/SelectPrincipal.vue +36 -5
  59. package/components/form/ArrayList.vue +3 -35
  60. package/components/form/ArrayListGrouped.vue +13 -4
  61. package/components/form/ArrayListSelect.vue +5 -5
  62. package/components/form/Error.vue +8 -0
  63. package/components/form/KeyValue.vue +39 -7
  64. package/components/form/LabeledSelect.vue +5 -2
  65. package/components/form/Labels.vue +46 -16
  66. package/components/form/Members/ClusterPermissionsEditor.vue +17 -17
  67. package/components/form/Members/MembershipEditor.vue +12 -12
  68. package/components/form/NameNsDescription.vue +1 -1
  69. package/components/form/NodeScheduling.vue +1 -1
  70. package/components/form/Probe.vue +3 -3
  71. package/components/form/ResourceQuota/Project.vue +6 -6
  72. package/components/form/ResourceTabs/index.vue +1 -6
  73. package/components/form/Security.vue +7 -6
  74. package/components/form/Select.vue +3 -2
  75. package/components/form/SelectOrCreateAuthSecret.vue +22 -29
  76. package/components/form/ServicePorts.vue +8 -0
  77. package/components/form/WorkloadPorts.vue +7 -1
  78. package/components/form/__tests__/ArrayList.test.ts +74 -0
  79. package/components/form/__tests__/ArrayListGrouped.test.ts +6 -4
  80. package/components/formatter/Checked.vue +1 -1
  81. package/components/formatter/ClusterLink.vue +5 -0
  82. package/components/formatter/IconIsDefault.vue +2 -2
  83. package/components/formatter/InternalExternalIP.vue +11 -8
  84. package/components/formatter/LiveDuration.vue +78 -0
  85. package/components/formatter/WorkloadHealthScale.vue +5 -3
  86. package/components/nav/Header.vue +6 -3
  87. package/components/nav/NamespaceFilter.vue +146 -63
  88. package/components/nav/TopLevelMenu.vue +22 -19
  89. package/components/nav/WindowManager/ContainerLogs.vue +83 -126
  90. package/components/nav/WindowManager/ContainerShell.vue +9 -7
  91. package/components/nav/WindowManager/Window.vue +2 -0
  92. package/components/nav/WindowManager/index.vue +10 -0
  93. package/config/elemental-types.js +9 -0
  94. package/config/features.js +2 -0
  95. package/config/home-links.js +4 -1
  96. package/config/pod-security-admission.ts +82 -0
  97. package/config/product/apps.js +1 -1
  98. package/config/product/auth.js +6 -5
  99. package/config/product/explorer.js +6 -6
  100. package/config/product/fleet.js +1 -1
  101. package/config/product/manager.js +6 -2
  102. package/config/secret.js +0 -1
  103. package/config/settings.ts +26 -9
  104. package/config/table-headers.js +22 -11
  105. package/config/types.js +4 -1
  106. package/content/docs/zh-hans/getting-started.md +113 -137
  107. package/content/docs/zh-hans/whats-new.md +8 -46
  108. package/creators/pkg/package-lock.json +37 -0
  109. package/creators/pkg/package.json +1 -1
  110. package/detail/catalog.cattle.io.app.vue +1 -1
  111. package/detail/pod.vue +1 -1
  112. package/detail/provisioning.cattle.io.cluster.vue +35 -9
  113. package/detail/service.vue +2 -9
  114. package/detail/workload/index.vue +0 -1
  115. package/dialog/AddClusterMemberDialog.vue +22 -28
  116. package/dialog/AddProjectMemberDialog.vue +53 -9
  117. package/dialog/DiagnosticTimingsDialog.vue +8 -7
  118. package/dialog/DrainNode.vue +44 -48
  119. package/dialog/ForceMachineRemoveDialog.vue +5 -7
  120. package/dialog/GenericPrompt.vue +15 -20
  121. package/dialog/RollbackWorkloadDialog.vue +15 -46
  122. package/dialog/RotateCertificatesDialog.vue +5 -7
  123. package/dialog/RotateEncryptionKeyDialog.vue +5 -9
  124. package/dialog/SaveAsRKETemplateDialog.vue +5 -13
  125. package/dialog/ScaleMachineDownDialog.vue +1 -1
  126. package/dialog/ScalePoolDownDialog.vue +121 -0
  127. package/edit/__tests__/management.cattle.io.setting.test.ts +3 -3
  128. package/edit/auth/azuread.vue +16 -16
  129. package/edit/auth/github.vue +8 -0
  130. package/edit/auth/googleoauth.vue +10 -1
  131. package/edit/auth/ldap/index.vue +10 -0
  132. package/edit/auth/oidc.vue +10 -0
  133. package/edit/auth/saml.vue +10 -0
  134. package/edit/autoscaling.horizontalpodautoscaler/index.vue +1 -1
  135. package/edit/cloudcredential.vue +3 -7
  136. package/edit/logging-flow/Match.vue +39 -8
  137. package/edit/logging-flow/index.vue +27 -4
  138. package/edit/management.cattle.io.podsecurityadmissionconfigurationtemplate.vue +107 -0
  139. package/edit/management.cattle.io.project.vue +8 -1
  140. package/edit/management.cattle.io.setting.vue +5 -2
  141. package/edit/management.cattle.io.user.vue +7 -1
  142. package/edit/monitoring.coreos.com.alertmanagerconfig/receiverConfig.vue +23 -7
  143. package/edit/monitoring.coreos.com.alertmanagerconfig/types/email.vue +2 -2
  144. package/edit/monitoring.coreos.com.prometheusrule/GroupRules.vue +14 -6
  145. package/edit/namespace.vue +18 -4
  146. package/edit/networking.k8s.io.ingress/Certificate.vue +1 -0
  147. package/edit/networking.k8s.io.ingress/IngressClass.vue +8 -6
  148. package/edit/networking.k8s.io.ingress/RulePath.vue +12 -6
  149. package/edit/networking.k8s.io.ingress/index.vue +8 -6
  150. package/edit/persistentvolume/index.vue +30 -27
  151. package/edit/persistentvolume/plugins/cephfs.vue +29 -29
  152. package/edit/persistentvolume/plugins/csi.vue +102 -62
  153. package/edit/persistentvolume/plugins/fc.vue +19 -19
  154. package/edit/persistentvolume/plugins/iscsi.vue +45 -45
  155. package/edit/persistentvolume/plugins/rbd.vue +39 -39
  156. package/edit/persistentvolumeclaim.vue +78 -75
  157. package/edit/provisioning.cattle.io.cluster/MachinePool.vue +11 -7
  158. package/edit/provisioning.cattle.io.cluster/RegistryConfigs.vue +10 -1
  159. package/edit/provisioning.cattle.io.cluster/RegistryMirrors.vue +87 -27
  160. package/edit/provisioning.cattle.io.cluster/SelectCredential.vue +3 -6
  161. package/edit/provisioning.cattle.io.cluster/__tests__/rke2.test.ts +93 -0
  162. package/edit/provisioning.cattle.io.cluster/import.vue +1 -1
  163. package/edit/provisioning.cattle.io.cluster/index.vue +29 -6
  164. package/edit/provisioning.cattle.io.cluster/rke2.vue +440 -152
  165. package/edit/secret/index.vue +3 -7
  166. package/edit/service.vue +3 -1
  167. package/edit/storage.k8s.io.storageclass/index.vue +100 -16
  168. package/edit/storage.k8s.io.storageclass/provisioners/driver.harvesterhci.io.vue +114 -0
  169. package/edit/workload/__tests__/index.test.ts +98 -0
  170. package/edit/workload/index.vue +58 -8
  171. package/edit/workload/mixins/workload.js +107 -70
  172. package/edit/workload/storage/ContainerMountPaths.vue +0 -10
  173. package/edit/workload/storage/emptyDir.vue +88 -0
  174. package/edit/workload/storage/ephemeralVolume/index.vue +1 -1
  175. package/edit/workload/storage/index.vue +8 -0
  176. package/edit/workload/storage/persistentVolumeClaim/index.vue +1 -1
  177. package/layouts/default.vue +57 -44
  178. package/list/__tests__/workload.test.ts +5 -2
  179. package/list/catalog.cattle.io.app.vue +1 -0
  180. package/list/cis.cattle.io.clusterscan.vue +1 -0
  181. package/list/fleet.cattle.io.bundle.vue +5 -6
  182. package/list/fleet.cattle.io.cluster.vue +6 -3
  183. package/list/fleet.cattle.io.clusterregistrationtoken.vue +5 -6
  184. package/list/fleet.cattle.io.gitrepo.vue +4 -9
  185. package/list/helm.cattle.io.projecthelmchart.vue +1 -5
  186. package/list/logging.banzaicloud.io.clusterflow.vue +4 -1
  187. package/list/logging.banzaicloud.io.flow.vue +6 -5
  188. package/list/management.cattle.io.cluster.vue +1 -0
  189. package/list/management.cattle.io.feature.vue +3 -4
  190. package/list/management.cattle.io.podsecurityadmissionconfigurationtemplate.vue +47 -0
  191. package/list/management.cattle.io.setting.vue +2 -2
  192. package/list/management.cattle.io.user.vue +4 -10
  193. package/list/monitoring.coreos.com.alertmanagerconfig.vue +2 -7
  194. package/list/node.vue +8 -5
  195. package/list/persistentvolume.vue +3 -3
  196. package/list/persistentvolumeclaim.vue +3 -4
  197. package/list/provisioning.cattle.io.cluster.vue +18 -19
  198. package/list/service.vue +6 -14
  199. package/list/workload.vue +43 -38
  200. package/machine-config/azure.vue +429 -60
  201. package/machine-config/pnap.vue +288 -0
  202. package/mixins/auth-config.js +1 -3
  203. package/mixins/browser-tab-visibility.js +8 -14
  204. package/mixins/chart.js +1 -1
  205. package/mixins/create-edit-view/impl.js +4 -0
  206. package/mixins/create-edit-view/index.js +4 -2
  207. package/mixins/resource-fetch-namespaced.js +98 -0
  208. package/mixins/resource-fetch.js +79 -45
  209. package/mixins/resource-manager.js +1 -23
  210. package/models/apps.controllerrevision.js +7 -0
  211. package/models/apps.daemonset.js +18 -0
  212. package/models/apps.deployment.js +44 -0
  213. package/models/apps.replicaset.js +7 -0
  214. package/models/apps.statefulset.js +18 -0
  215. package/models/batch.job.js +7 -14
  216. package/models/cluster/node.js +10 -2
  217. package/models/cluster.x-k8s.io.machine.js +26 -4
  218. package/models/cluster.x-k8s.io.machinedeployment.js +12 -2
  219. package/models/event.js +7 -0
  220. package/models/logging.banzaicloud.io.flow.js +4 -0
  221. package/models/management.cattle.io.cluster.js +1 -1
  222. package/models/management.cattle.io.clusterroletemplatebinding.js +1 -1
  223. package/models/management.cattle.io.globalrole.js +2 -2
  224. package/models/management.cattle.io.node.js +37 -2
  225. package/models/management.cattle.io.podsecurityadmissionconfigurationtemplate.ts +4 -0
  226. package/models/management.cattle.io.project.js +30 -11
  227. package/models/management.cattle.io.setting.js +1 -1
  228. package/models/management.cattle.io.user.js +37 -1
  229. package/models/namespace.js +42 -5
  230. package/models/persistentvolume.js +14 -2
  231. package/models/pod.js +15 -0
  232. package/models/projectroletemplatebinding.js +7 -0
  233. package/models/provisioning.cattle.io.cluster.js +61 -10
  234. package/models/rke-machine.cattle.io.pnapmachinetemplate.js +15 -0
  235. package/models/service.js +14 -13
  236. package/models/storage.k8s.io.storageclass.js +33 -18
  237. package/models/workload.js +38 -7
  238. package/nuxt.config.js +27 -17
  239. package/package.json +7 -7
  240. package/pages/about.vue +14 -2
  241. package/pages/c/_cluster/apps/charts/index.vue +4 -3
  242. package/pages/c/_cluster/apps/charts/install.vue +59 -22
  243. package/pages/c/_cluster/auth/config/_id.vue +6 -0
  244. package/pages/c/_cluster/auth/config/index.vue +8 -6
  245. package/pages/c/_cluster/auth/group.principal/assign-edit.vue +1 -1
  246. package/pages/c/_cluster/auth/roles/index.vue +1 -1
  247. package/pages/c/_cluster/explorer/index.vue +12 -6
  248. package/pages/c/_cluster/longhorn/index.vue +1 -1
  249. package/pages/c/_cluster/monitoring/alertmanagerconfig/_alertmanagerconfigid/receiver.vue +15 -4
  250. package/pages/c/_cluster/monitoring/index.vue +1 -1
  251. package/pages/c/_cluster/neuvector/index.vue +1 -1
  252. package/pages/c/_cluster/settings/performance.vue +48 -2
  253. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +34 -1
  254. package/pages/c/_cluster/uiplugins/index.vue +28 -2
  255. package/pages/diagnostic.vue +5 -4
  256. package/pages/home.vue +105 -30
  257. package/pages/prefs.vue +23 -12
  258. package/pages/rio/mesh.vue +1 -1
  259. package/pkg/dynamic-importer.lib.js +8 -0
  260. package/pkg/vue.config.js +4 -0
  261. package/plugins/dashboard-store/__tests__/mutations.spec.js +406 -0
  262. package/plugins/dashboard-store/actions.js +32 -25
  263. package/plugins/dashboard-store/getters.js +50 -33
  264. package/plugins/dashboard-store/mutations.js +134 -28
  265. package/plugins/dashboard-store/resource-class.js +21 -41
  266. package/plugins/steve/actions.js +30 -0
  267. package/plugins/steve/caches/resourceCache.js +60 -0
  268. package/plugins/steve/getters.js +44 -1
  269. package/plugins/steve/mutations.js +97 -36
  270. package/plugins/steve/resourceWatcher.js +277 -0
  271. package/plugins/steve/schema.utils.js +25 -0
  272. package/plugins/steve/subscribe.js +288 -115
  273. package/plugins/steve/worker/index.js +17 -0
  274. package/plugins/steve/worker/web-worker.advanced.js +302 -0
  275. package/plugins/steve/{web-worker.steve-sub-worker.js → worker/web-worker.basic.js} +3 -44
  276. package/rancher-components/Card/Card.vue +3 -3
  277. package/rancher-components/Form/TextArea/TextAreaAutoGrow.vue +1 -0
  278. package/rancher-components/StringList/StringList.test.ts +45 -420
  279. package/rancher-components/StringList/StringList.vue +1 -10
  280. package/rancher-components/components/Banner/Banner.test.ts +44 -0
  281. package/rancher-components/components/Banner/Banner.vue +129 -61
  282. package/rancher-components/components/Form/Checkbox/Checkbox.test.ts +13 -22
  283. package/rancher-components/components/Form/Checkbox/Checkbox.vue +8 -6
  284. package/rancher-components/components/Form/ToggleSwitch/ToggleSwitch.test.ts +9 -9
  285. package/rancher-components/components/LabeledTooltip/LabeledTooltip.vue +0 -1
  286. package/rancher-components/components/StringList/StringList.test.ts +7 -7
  287. package/rancher-components/components/StringList/StringList.vue +21 -15
  288. package/scripts/test-plugins-build.sh +8 -0
  289. package/static/loading-indicator.html +1 -1
  290. package/store/index.js +54 -3
  291. package/store/plugins.js +0 -17
  292. package/store/pnap.js +128 -0
  293. package/store/prefs.js +4 -2
  294. package/store/type-map.js +55 -13
  295. package/types/pod-security-admission.ts +36 -0
  296. package/types/shell/index.d.ts +496 -396
  297. package/utils/__tests__/object.test.ts +17 -1
  298. package/utils/__tests__/pod-security-admission.test.ts +61 -0
  299. package/utils/async.ts +36 -0
  300. package/utils/color.js +45 -0
  301. package/utils/crypto/browserHashUtils.js +18 -0
  302. package/utils/dynamic-importer.js +8 -0
  303. package/utils/install-redirect.js +1 -1
  304. package/utils/object.js +24 -0
  305. package/utils/pod-security-admission.ts +39 -0
  306. package/utils/socket.js +61 -24
  307. package/utils/string.js +2 -0
  308. package/utils/svg-filter.js +301 -0
  309. package/utils/time.js +49 -0
  310. package/utils/validators/cidr.js +4 -0
  311. package/utils/validators/formRules/__tests__/index.test.ts +23 -3
  312. package/utils/validators/formRules/index.ts +14 -0
  313. package/config/product/harvester-manager.js +0 -162
  314. package/edit/harvesterhci.io.management.cluster.vue +0 -153
  315. package/list/harvesterhci.io.management.cluster.vue +0 -241
  316. package/machine-config/harvester.vue +0 -693
  317. package/models/harvesterhci.io.management.cluster.js +0 -228
  318. package/pages/c/_cluster/harvesterManager/index.vue +0 -24
  319. package/rancher-components/Card/Card.test.ts +0 -39
  320. package/rancher-components/Utils/DraggableZone/DraggableZone.vue +0 -181
  321. package/rancher-components/Utils/DraggableZone/index.ts +0 -1
  322. package/rancher-components/components/Utils/DraggableZone/index.ts +0 -1
@@ -0,0 +1,301 @@
1
+ 'use strict';
2
+
3
+ // Utilities to generate a css filter to give a required color
4
+
5
+ class Color {
6
+ constructor(r, g, b) {
7
+ this.set(r, g, b);
8
+ }
9
+
10
+ toString() {
11
+ return `rgb(${ Math.round(this.r) }, ${ Math.round(this.g) }, ${ Math.round(this.b) })`;
12
+ }
13
+
14
+ set(r, g, b) {
15
+ this.r = this.clamp(r);
16
+ this.g = this.clamp(g);
17
+ this.b = this.clamp(b);
18
+ }
19
+
20
+ hueRotate(angle = 0) {
21
+ angle = angle / 180 * Math.PI;
22
+ const sin = Math.sin(angle);
23
+ const cos = Math.cos(angle);
24
+
25
+ this.multiply([
26
+ 0.213 + cos * 0.787 - sin * 0.213,
27
+ 0.715 - cos * 0.715 - sin * 0.715,
28
+ 0.072 - cos * 0.072 + sin * 0.928,
29
+ 0.213 - cos * 0.213 + sin * 0.143,
30
+ 0.715 + cos * 0.285 + sin * 0.140,
31
+ 0.072 - cos * 0.072 - sin * 0.283,
32
+ 0.213 - cos * 0.213 - sin * 0.787,
33
+ 0.715 - cos * 0.715 + sin * 0.715,
34
+ 0.072 + cos * 0.928 + sin * 0.072,
35
+ ]);
36
+ }
37
+
38
+ grayscale(value = 1) {
39
+ this.multiply([
40
+ 0.2126 + 0.7874 * (1 - value),
41
+ 0.7152 - 0.7152 * (1 - value),
42
+ 0.0722 - 0.0722 * (1 - value),
43
+ 0.2126 - 0.2126 * (1 - value),
44
+ 0.7152 + 0.2848 * (1 - value),
45
+ 0.0722 - 0.0722 * (1 - value),
46
+ 0.2126 - 0.2126 * (1 - value),
47
+ 0.7152 - 0.7152 * (1 - value),
48
+ 0.0722 + 0.9278 * (1 - value),
49
+ ]);
50
+ }
51
+
52
+ sepia(value = 1) {
53
+ this.multiply([
54
+ 0.393 + 0.607 * (1 - value),
55
+ 0.769 - 0.769 * (1 - value),
56
+ 0.189 - 0.189 * (1 - value),
57
+ 0.349 - 0.349 * (1 - value),
58
+ 0.686 + 0.314 * (1 - value),
59
+ 0.168 - 0.168 * (1 - value),
60
+ 0.272 - 0.272 * (1 - value),
61
+ 0.534 - 0.534 * (1 - value),
62
+ 0.131 + 0.869 * (1 - value),
63
+ ]);
64
+ }
65
+
66
+ saturate(value = 1) {
67
+ this.multiply([
68
+ 0.213 + 0.787 * value,
69
+ 0.715 - 0.715 * value,
70
+ 0.072 - 0.072 * value,
71
+ 0.213 - 0.213 * value,
72
+ 0.715 + 0.285 * value,
73
+ 0.072 - 0.072 * value,
74
+ 0.213 - 0.213 * value,
75
+ 0.715 - 0.715 * value,
76
+ 0.072 + 0.928 * value,
77
+ ]);
78
+ }
79
+
80
+ multiply(matrix) {
81
+ const newR = this.clamp(this.r * matrix[0] + this.g * matrix[1] + this.b * matrix[2]);
82
+ const newG = this.clamp(this.r * matrix[3] + this.g * matrix[4] + this.b * matrix[5]);
83
+ const newB = this.clamp(this.r * matrix[6] + this.g * matrix[7] + this.b * matrix[8]);
84
+
85
+ this.r = newR;
86
+ this.g = newG;
87
+ this.b = newB;
88
+ }
89
+
90
+ brightness(value = 1) {
91
+ this.linear(value);
92
+ }
93
+
94
+ contrast(value = 1) {
95
+ this.linear(value, -(0.5 * value) + 0.5);
96
+ }
97
+
98
+ linear(slope = 1, intercept = 0) {
99
+ this.r = this.clamp(this.r * slope + intercept * 255);
100
+ this.g = this.clamp(this.g * slope + intercept * 255);
101
+ this.b = this.clamp(this.b * slope + intercept * 255);
102
+ }
103
+
104
+ invert(value = 1) {
105
+ this.r = this.clamp((value + this.r / 255 * (1 - 2 * value)) * 255);
106
+ this.g = this.clamp((value + this.g / 255 * (1 - 2 * value)) * 255);
107
+ this.b = this.clamp((value + this.b / 255 * (1 - 2 * value)) * 255);
108
+ }
109
+
110
+ hsl() {
111
+ // Code taken from https://stackoverflow.com/a/9493060/2688027, licensed under CC BY-SA.
112
+ const r = this.r / 255;
113
+ const g = this.g / 255;
114
+ const b = this.b / 255;
115
+ const max = Math.max(r, g, b);
116
+ const min = Math.min(r, g, b);
117
+ const l = (max + min) / 2;
118
+ let h = l;
119
+ let s = l;
120
+
121
+ if (max === min) {
122
+ h = s = 0;
123
+ } else {
124
+ const d = max - min;
125
+
126
+ s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
127
+ switch (max) {
128
+ case r:
129
+ h = (g - b) / d + (g < b ? 6 : 0);
130
+ break;
131
+
132
+ case g:
133
+ h = (b - r) / d + 2;
134
+ break;
135
+
136
+ case b:
137
+ h = (r - g) / d + 4;
138
+ break;
139
+ }
140
+ h /= 6;
141
+ }
142
+
143
+ return {
144
+ h: h * 100,
145
+ s: s * 100,
146
+ l: l * 100,
147
+ };
148
+ }
149
+
150
+ clamp(value) {
151
+ if (value > 255) {
152
+ value = 255;
153
+ } else if (value < 0) {
154
+ value = 0;
155
+ }
156
+
157
+ return value;
158
+ }
159
+ }
160
+
161
+ export class Solver {
162
+ constructor(rgb) {
163
+ this.target = new Color(rgb.r, rgb.g, rgb.b);
164
+ this.targetHSL = this.target.hsl();
165
+ this.reusedColor = new Color(0, 0, 0);
166
+ }
167
+
168
+ solve() {
169
+ const result = this.solveNarrow(this.solveWide());
170
+
171
+ return {
172
+ values: result.values,
173
+ loss: result.loss,
174
+ filter: this.css(result.values),
175
+ };
176
+ }
177
+
178
+ solveWide() {
179
+ const A = 5;
180
+ const c = 15;
181
+ const a = [60, 180, 18000, 600, 1.2, 1.2];
182
+
183
+ let best = { loss: Infinity };
184
+
185
+ for (let i = 0; best.loss > 25 && i < 3; i++) {
186
+ const initial = [50, 20, 3750, 50, 100, 100];
187
+ const result = this.spsa(A, a, c, initial, 1000);
188
+
189
+ if (result.loss < best.loss) {
190
+ best = result;
191
+ }
192
+ }
193
+
194
+ return best;
195
+ }
196
+
197
+ solveNarrow(wide) {
198
+ const A = wide.loss;
199
+ const c = 2;
200
+ const A1 = A + 1;
201
+ const a = [0.25 * A1, 0.25 * A1, A1, 0.25 * A1, 0.2 * A1, 0.2 * A1];
202
+
203
+ return this.spsa(A, a, c, wide.values, 500);
204
+ }
205
+
206
+ spsa(A, a, c, values, iters) {
207
+ const alpha = 1;
208
+ const gamma = 0.16666666666666666;
209
+
210
+ let best = null;
211
+ let bestLoss = Infinity;
212
+ const deltas = new Array(6);
213
+ const highArgs = new Array(6);
214
+ const lowArgs = new Array(6);
215
+
216
+ for (let k = 0; k < iters; k++) {
217
+ const ck = c / Math.pow(k + 1, gamma);
218
+
219
+ for (let i = 0; i < 6; i++) {
220
+ deltas[i] = Math.random() > 0.5 ? 1 : -1;
221
+ highArgs[i] = values[i] + ck * deltas[i];
222
+ lowArgs[i] = values[i] - ck * deltas[i];
223
+ }
224
+
225
+ const lossDiff = this.loss(highArgs) - this.loss(lowArgs);
226
+
227
+ for (let i = 0; i < 6; i++) {
228
+ const g = lossDiff / (2 * ck) * deltas[i];
229
+ const ak = a[i] / Math.pow(A + k + 1, alpha);
230
+
231
+ values[i] = fix(values[i] - ak * g, i);
232
+ }
233
+
234
+ const loss = this.loss(values);
235
+
236
+ if (loss < bestLoss) {
237
+ best = values.slice(0);
238
+ bestLoss = loss;
239
+ }
240
+ }
241
+
242
+ return { values: best, loss: bestLoss };
243
+
244
+ function fix(value, idx) {
245
+ let max = 100;
246
+
247
+ if (idx === 2 /* saturate */) {
248
+ max = 7500;
249
+ } else if (idx === 4 /* brightness */ || idx === 5 /* contrast */) {
250
+ max = 200;
251
+ }
252
+
253
+ if (idx === 3 /* hue-rotate */) {
254
+ if (value > max) {
255
+ value %= max;
256
+ } else if (value < 0) {
257
+ value = max + value % max;
258
+ }
259
+ } else if (value < 0) {
260
+ value = 0;
261
+ } else if (value > max) {
262
+ value = max;
263
+ }
264
+
265
+ return value;
266
+ }
267
+ }
268
+
269
+ loss(filters) {
270
+ // Argument is array of percentages.
271
+ const color = this.reusedColor;
272
+
273
+ color.set(0, 0, 0);
274
+
275
+ color.invert(filters[0] / 100);
276
+ color.sepia(filters[1] / 100);
277
+ color.saturate(filters[2] / 100);
278
+ color.hueRotate(filters[3] * 3.6);
279
+ color.brightness(filters[4] / 100);
280
+ color.contrast(filters[5] / 100);
281
+
282
+ const colorHSL = color.hsl();
283
+
284
+ return (
285
+ Math.abs(color.r - this.target.r) +
286
+ Math.abs(color.g - this.target.g) +
287
+ Math.abs(color.b - this.target.b) +
288
+ Math.abs(colorHSL.h - this.targetHSL.h) +
289
+ Math.abs(colorHSL.s - this.targetHSL.s) +
290
+ Math.abs(colorHSL.l - this.targetHSL.l)
291
+ );
292
+ }
293
+
294
+ css(filters) {
295
+ function fmt(idx, multiplier = 1) {
296
+ return Math.round(filters[idx] * multiplier);
297
+ }
298
+
299
+ return `filter: invert(${ fmt(0) }%) sepia(${ fmt(1) }%) saturate(${ fmt(2) }%) hue-rotate(${ fmt(3, 3.6) }deg) brightness(${ fmt(4) }%) contrast(${ fmt(5) }%);`;
300
+ }
301
+ }
package/utils/time.js CHANGED
@@ -60,3 +60,52 @@ export function getSecondsDiff(startDate, endDate) {
60
60
  Math.abs(Date.parse(endDate) - Date.parse(startDate)) / 1000
61
61
  );
62
62
  }
63
+
64
+ /**
65
+ * return { diff: number; label: string }
66
+ *
67
+ * diff: update frequency in seconds
68
+ * label: content of the cell's column
69
+ */
70
+ export function elapsedTime(seconds) {
71
+ if (!seconds) {
72
+ return {};
73
+ }
74
+
75
+ if (seconds < 120) {
76
+ return {
77
+ diff: 1,
78
+ label: `${ seconds }s`
79
+ };
80
+ }
81
+
82
+ const minutes = Math.floor(seconds / 60);
83
+
84
+ if (minutes < 10) {
85
+ return {
86
+ diff: 1,
87
+ label: `${ minutes }m${ seconds - (minutes * 60) }s`
88
+ };
89
+ }
90
+
91
+ const hours = Math.floor(seconds / 3600);
92
+
93
+ if (hours < 3) {
94
+ return {
95
+ diff: 60,
96
+ label: `${ minutes }m`,
97
+ };
98
+ }
99
+
100
+ if (hours > 7) {
101
+ return {
102
+ diff: 60,
103
+ label: `${ hours }h`,
104
+ };
105
+ }
106
+
107
+ return {
108
+ diff: 60,
109
+ label: `${ hours }h${ minutes - (hours * 60) }m`,
110
+ };
111
+ }
@@ -3,3 +3,7 @@ const validCIDRregex = /^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(25[0-5]|2
3
3
  export function isValidCIDR(cidr) {
4
4
  return !!cidr.match(validCIDRregex);
5
5
  }
6
+
7
+ export function isValidMac(value) {
8
+ return /^[A-Fa-f0-9]{2}(-[A-Fa-f0-9]{2}){5}$|^[A-Fa-f0-9]{2}(:[A-Fa-f0-9]{2}){5}$/.test(value);
9
+ }
@@ -1041,17 +1041,37 @@ describe('formRules', () => {
1041
1041
  ['minLength', 2, ['test'], ['x']],
1042
1042
  ['maxLength', 10, ['x'], ['wrong value']],
1043
1043
  ['betweenLengths', [2, 10], ['test'], ['x', 'wrong value']],
1044
- ])('%p with parameter %p should', (rule, argument, correctValues, wrongValues) => {
1045
- it.each(wrongValues as [])('return error for value %p', (wrong) => {
1044
+ ])('given factory validator %p with parameter %p', (rule, argument, correctValues, wrongValues) => {
1045
+ it.each(wrongValues as [])('should return error for value %p', (wrong) => {
1046
1046
  const formRuleResult = (formRules as any)[rule](argument)(wrong);
1047
1047
 
1048
1048
  expect(formRuleResult).not.toBeUndefined();
1049
1049
  });
1050
1050
 
1051
- it.each(correctValues as [])('return valid for value %p', (correct) => {
1051
+ it.each(correctValues as [])('should return valid for value %p', (correct) => {
1052
1052
  const formRuleResult = (formRules as any)[rule](argument)(correct);
1053
1053
 
1054
1054
  expect(formRuleResult).toBeUndefined();
1055
1055
  });
1056
1056
  });
1057
+
1058
+ describe.each([
1059
+ ['requiredInt', [2, 2.2], ['e']],
1060
+ ['isInteger', ['2', 2, 0], [2.2, 'e', '1.0']],
1061
+ ['isPositive', ['0', 1], [-1]],
1062
+ ['isOctal', ['0', 0, 10], ['01']],
1063
+
1064
+ ])('given validator %p', (rule, correctValues, wrongValues) => {
1065
+ it.each(wrongValues as [])('should return error for value %p', (wrong) => {
1066
+ const formRuleResult = (formRules as any)[rule](wrong);
1067
+
1068
+ expect(formRuleResult).not.toBeUndefined();
1069
+ });
1070
+
1071
+ it.each(correctValues as [])('should return valid for value %p', (correct) => {
1072
+ const formRuleResult = (formRules as any)[rule](correct);
1073
+
1074
+ expect(formRuleResult).toBeUndefined();
1075
+ });
1076
+ });
1057
1077
  });
@@ -83,6 +83,17 @@ export default function(t: Translation, { key = 'Value' }: ValidationOptions) {
83
83
 
84
84
  const requiredInt: Validator = (val: string) => isNaN(parseInt(val, 10)) ? t('validation.number.requiredInt', { key }) : undefined;
85
85
 
86
+ const isInteger: Validator = (val: string | number) => !Number.isInteger(+val) || `${ val }`.match(/\.+/g) ? t('validation.number.requiredInt', { key }) : undefined;
87
+
88
+ const isPositive: Validator = (val: string | number) => +val < 0 ? t('validation.number.isPositive', { key }) : undefined;
89
+
90
+ const isOctal: Validator = (val: string | number) => {
91
+ const valueString = `${ val }`;
92
+ const isValid = valueString.match(/(^0+)(.+)/);
93
+
94
+ return isValid ? t('validation.number.isOctal', { key }) : undefined;
95
+ };
96
+
86
97
  const portNumber: Validator = (val: string) => parseInt(val, 10) < 1 || parseInt(val, 10) > 65535 ? t('validation.number.between', {
87
98
  key, min: '1', max: '65535'
88
99
  }) : undefined;
@@ -470,6 +481,9 @@ export default function(t: Translation, { key = 'Value' }: ValidationOptions) {
470
481
  portNumber,
471
482
  required,
472
483
  requiredInt,
484
+ isInteger,
485
+ isPositive,
486
+ isOctal,
473
487
  roleTemplateRules,
474
488
  ruleGroups,
475
489
  servicePort,
@@ -1,162 +0,0 @@
1
- import { HCI, MANAGEMENT, CAPI } from '@shell/config/types';
2
- import { HARVESTER, MULTI_CLUSTER } from '@shell/store/features';
3
- import { DSL } from '@shell/store/type-map';
4
- import { STATE, NAME as NAME_COL, AGE, VERSION } from '@shell/config/table-headers';
5
- import { allHash } from '@shell/utils/promise';
6
- import dynamicPluginLoader from '@shell/pkg/dynamic-plugin-loader';
7
- import { BLANK_CLUSTER } from '@shell/store';
8
-
9
- dynamicPluginLoader.register({
10
- load: async({ route, store }) => {
11
- // Check that we've either got here either
12
- // - directly (page refresh/load -> have path but no name)
13
- // - via router name (have name but no path)
14
- let clusterId;
15
- const pathParts = route.path.split('/');
16
-
17
- if (pathParts?.[1] === HARVESTER_NAME && pathParts?.[3] ) {
18
- clusterId = pathParts?.[3];
19
- } else {
20
- const nameParts = route.name?.split('-');
21
-
22
- if (nameParts?.[0] === HARVESTER_NAME) {
23
- clusterId = route.params?.cluster;
24
- }
25
- }
26
-
27
- // If we have a cluster id, try to load the plugin via the harvester cluster's `loadClusterPlugin`
28
- if (clusterId) {
29
- const provClusters = await store.dispatch('management/findAll', { type: CAPI.RANCHER_CLUSTER });
30
- const provCluster = provClusters.find(p => p.mgmt.id === clusterId);
31
-
32
- if (provCluster) {
33
- const harvCluster = await store.dispatch('management/create', {
34
- ...provCluster,
35
- type: HCI.CLUSTER
36
- });
37
-
38
- if (harvCluster) {
39
- try {
40
- await harvCluster.loadClusterPlugin();
41
-
42
- return route;
43
- } catch (err) {
44
- // If we've failed to load the harvester plugin nav to the harvester cluster list (probably got here from a bookmarked
45
- // harvester instance that hasn't been updated to serve a plugin)
46
- console.error('Failed to load harvester package: ', typeof error === 'object' ? JSON.stringify(err) : err); // eslint-disable-line no-console
47
-
48
- return harvesterClustersLocation;
49
- }
50
- }
51
- }
52
- }
53
- }
54
- });
55
-
56
- export const NAME = 'harvesterManager';
57
-
58
- export const HARVESTER_NAME = 'harvester';
59
-
60
- const MACHINE_POOLS = {
61
- name: 'summary',
62
- labelKey: 'tableHeaders.machines',
63
- sort: false,
64
- search: false,
65
- value: 'nodes.length',
66
- align: 'center',
67
- width: 100,
68
- };
69
-
70
- const harvesterClustersLocation = {
71
- name: 'c-cluster-product-resource',
72
- params: {
73
- cluster: BLANK_CLUSTER,
74
- product: NAME,
75
- resource: HCI.CLUSTER
76
- }
77
- };
78
-
79
- export function init(store) {
80
- const {
81
- product,
82
- basicType,
83
- headers,
84
- spoofedType,
85
- configureType
86
- } = DSL(store, NAME);
87
-
88
- product({
89
- ifHaveType: CAPI.RANCHER_CLUSTER,
90
- ifFeature: [MULTI_CLUSTER, HARVESTER],
91
- inStore: 'management',
92
- icon: 'harvester',
93
- removable: false,
94
- showClusterSwitcher: false,
95
- weight: 100,
96
- to: harvesterClustersLocation,
97
- });
98
-
99
- configureType(HCI.CLUSTER, { showListMasthead: false });
100
- headers(HCI.CLUSTER, [
101
- STATE,
102
- NAME_COL,
103
- {
104
- ...VERSION,
105
- value: 'kubernetesVersion',
106
- getValue: row => row.kubernetesVersion
107
- },
108
- MACHINE_POOLS,
109
- AGE,
110
- {
111
- name: 'harvester',
112
- label: ' ',
113
- align: 'right',
114
- width: 65,
115
- },
116
- ]);
117
- basicType([HCI.CLUSTER]);
118
- spoofedType({
119
- label: store.getters['i18n/t']('harvesterManager.cluster.label'),
120
- name: HCI.CLUSTER,
121
- type: HCI.CLUSTER,
122
- namespaced: false,
123
- weight: -1,
124
- route: {
125
- name: 'c-cluster-product-resource',
126
- params: {
127
- product: NAME,
128
- resource: HCI.CLUSTER,
129
- }
130
- },
131
- exact: false,
132
- schemas: [
133
- {
134
- id: HCI.CLUSTER,
135
- type: 'schema',
136
- collectionMethods: [],
137
- resourceFields: {},
138
- attributes: { namespaced: true },
139
- },
140
- ],
141
- group: 'Root',
142
- getInstances: async() => {
143
- const hash = {
144
- rancherClusters: store.dispatch('management/findAll', { type: CAPI.RANCHER_CLUSTER }),
145
- clusters: store.dispatch('management/findAll', { type: MANAGEMENT.CLUSTER }),
146
- };
147
-
148
- if (store.getters['management/schemaFor'](MANAGEMENT.NODE)) {
149
- hash.nodes = store.dispatch('management/findAll', { type: MANAGEMENT.NODE });
150
- }
151
-
152
- const res = await allHash(hash);
153
-
154
- return res.rancherClusters.map((c) => {
155
- return {
156
- ...c,
157
- type: HCI.CLUSTER,
158
- };
159
- });
160
- },
161
- });
162
- }