@rancher/shell 0.3.0 → 0.3.2

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 (342) 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/_gauges.scss +1 -1
  4. package/assets/styles/global/_layout.scss +5 -2
  5. package/assets/styles/global/_select.scss +1 -4
  6. package/assets/styles/themes/_dark.scss +5 -4
  7. package/assets/styles/themes/_light.scss +4 -3
  8. package/assets/styles/themes/_suse.scss +1 -1
  9. package/assets/styles/vendor/vue-select.scss +4 -3
  10. package/assets/translations/en-us.yaml +673 -73
  11. package/assets/translations/zh-hans.yaml +720 -207
  12. package/chart/monitoring/steps/uninstall-v1.vue +2 -2
  13. package/cloud-credential/azure.vue +23 -0
  14. package/cloud-credential/harvester.vue +25 -62
  15. package/cloud-credential/pnap.vue +80 -0
  16. package/components/.DS_Store +0 -0
  17. package/components/ActionMenu.vue +28 -7
  18. package/components/AdvancedSection.vue +9 -2
  19. package/components/Alert.vue +2 -2
  20. package/components/ButtonDropdown.vue +0 -2
  21. package/components/ButtonGroup.vue +1 -0
  22. package/components/CollapsibleCard.vue +0 -1
  23. package/components/CruResource.vue +41 -4
  24. package/components/DetailTop.vue +72 -4
  25. package/components/DisableAuthProviderModal.vue +106 -0
  26. package/{rancher-components/components/Utils/DraggableZone → components}/DraggableZone.vue +0 -0
  27. package/components/ExplorerMembers.vue +253 -30
  28. package/components/ExplorerProjectsNamespaces.vue +77 -33
  29. package/components/ExtensionPanel.vue +42 -0
  30. package/components/GrowlManager.vue +3 -3
  31. package/components/IconOrSvg.vue +178 -0
  32. package/components/LogItem.vue +69 -0
  33. package/components/PodSecurityAdmission.vue +302 -0
  34. package/components/PromptModal.vue +1 -0
  35. package/components/ResourceDetail/Masthead.vue +69 -4
  36. package/components/ResourceDetail/index.vue +12 -5
  37. package/components/ResourceList/Masthead.vue +11 -1
  38. package/components/ResourceList/ResourceLoadingIndicator.vue +12 -2
  39. package/components/ResourceList/index.vue +66 -12
  40. package/components/ResourceList/resource-list.config.js +7 -0
  41. package/components/ResourceTable.vue +33 -6
  42. package/components/SimpleBox.vue +1 -1
  43. package/components/SortableTable/THead.vue +21 -14
  44. package/components/SortableTable/filtering.js +1 -1
  45. package/components/SortableTable/index.vue +21 -10
  46. package/components/SortableTable/selection.js +15 -3
  47. package/components/Tabbed/Tab.vue +1 -1
  48. package/components/Tabbed/index.vue +20 -15
  49. package/components/__tests__/.DS_Store +0 -0
  50. package/components/__tests__/AsyncButton.test.ts +140 -0
  51. package/components/__tests__/BackLink.test.ts +33 -0
  52. package/components/__tests__/ButtonGroup.test.ts +124 -0
  53. package/components/__tests__/ClusterBadge.test.ts +32 -0
  54. package/components/__tests__/CollapsibleCard.test.ts +64 -0
  55. package/components/__tests__/ConsumptionGauge.test.ts +88 -0
  56. package/components/__tests__/CruResource.test.ts +3 -2
  57. package/components/__tests__/FixedBanner.test.ts +129 -0
  58. package/components/__tests__/GrowlManager.test.ts +147 -0
  59. package/components/__tests__/NamespaceFilter.test.ts +33 -25
  60. package/components/__tests__/PercentageBar.test.ts +32 -0
  61. package/components/__tests__/PodSecurityAdmission.test.ts +398 -0
  62. package/components/auth/AuthBanner.vue +20 -10
  63. package/components/auth/RoleDetailEdit.vue +26 -17
  64. package/components/auth/SelectPrincipal.vue +36 -5
  65. package/components/form/ArrayList.vue +3 -35
  66. package/components/form/ArrayListGrouped.vue +13 -4
  67. package/components/form/ArrayListSelect.vue +5 -5
  68. package/components/form/Error.vue +8 -0
  69. package/components/form/KeyValue.vue +39 -7
  70. package/components/form/LabeledSelect.vue +5 -2
  71. package/components/form/Labels.vue +46 -16
  72. package/components/form/Members/ClusterPermissionsEditor.vue +17 -17
  73. package/components/form/Members/MembershipEditor.vue +12 -12
  74. package/components/form/NameNsDescription.vue +1 -1
  75. package/components/form/NodeScheduling.vue +1 -1
  76. package/components/form/Probe.vue +3 -3
  77. package/components/form/ResourceQuota/Project.vue +6 -6
  78. package/components/form/ResourceTabs/index.vue +24 -6
  79. package/components/form/Security.vue +7 -6
  80. package/components/form/Select.vue +3 -2
  81. package/components/form/SelectOrCreateAuthSecret.vue +22 -29
  82. package/components/form/ServicePorts.vue +8 -0
  83. package/components/form/WorkloadPorts.vue +7 -1
  84. package/components/form/__tests__/ArrayList.test.ts +74 -0
  85. package/components/form/__tests__/ArrayListGrouped.test.ts +6 -4
  86. package/components/formatter/Checked.vue +1 -1
  87. package/components/formatter/ClusterLink.vue +5 -0
  88. package/components/formatter/IconIsDefault.vue +2 -2
  89. package/components/formatter/InternalExternalIP.vue +11 -8
  90. package/components/formatter/LiveDuration.vue +78 -0
  91. package/components/formatter/WorkloadHealthScale.vue +5 -3
  92. package/components/nav/Header.vue +74 -7
  93. package/components/nav/NamespaceFilter.vue +146 -63
  94. package/components/nav/TopLevelMenu.vue +22 -19
  95. package/components/nav/WindowManager/ContainerLogs.vue +83 -126
  96. package/components/nav/WindowManager/ContainerShell.vue +9 -7
  97. package/components/nav/WindowManager/Window.vue +2 -0
  98. package/components/nav/WindowManager/index.vue +10 -0
  99. package/config/elemental-types.js +9 -0
  100. package/config/features.js +2 -0
  101. package/config/home-links.js +4 -1
  102. package/config/pod-security-admission.ts +82 -0
  103. package/config/product/apps.js +1 -1
  104. package/config/product/auth.js +6 -5
  105. package/config/product/backup.js +1 -1
  106. package/config/product/explorer.js +6 -6
  107. package/config/product/fleet.js +1 -1
  108. package/config/product/manager.js +6 -2
  109. package/config/query-params.js +1 -0
  110. package/config/secret.js +0 -1
  111. package/config/settings.ts +26 -9
  112. package/config/table-headers.js +22 -11
  113. package/config/types.js +4 -1
  114. package/config/uiplugins.js +3 -3
  115. package/content/docs/zh-hans/getting-started.md +113 -137
  116. package/content/docs/zh-hans/whats-new.md +8 -46
  117. package/core/plugin-helpers.js +171 -0
  118. package/core/plugin.ts +61 -1
  119. package/core/plugins.js +33 -0
  120. package/core/types.ts +128 -2
  121. package/creators/pkg/package-lock.json +37 -0
  122. package/creators/pkg/package.json +1 -1
  123. package/detail/catalog.cattle.io.app.vue +1 -1
  124. package/detail/pod.vue +1 -1
  125. package/detail/provisioning.cattle.io.cluster.vue +35 -9
  126. package/detail/service.vue +2 -9
  127. package/detail/workload/index.vue +0 -1
  128. package/dialog/AddClusterMemberDialog.vue +22 -28
  129. package/dialog/AddProjectMemberDialog.vue +53 -9
  130. package/dialog/DiagnosticTimingsDialog.vue +8 -7
  131. package/dialog/DrainNode.vue +44 -48
  132. package/dialog/ForceMachineRemoveDialog.vue +5 -7
  133. package/dialog/GenericPrompt.vue +15 -20
  134. package/dialog/RollbackWorkloadDialog.vue +15 -46
  135. package/dialog/RotateCertificatesDialog.vue +5 -7
  136. package/dialog/RotateEncryptionKeyDialog.vue +5 -9
  137. package/dialog/SaveAsRKETemplateDialog.vue +5 -13
  138. package/dialog/ScaleMachineDownDialog.vue +1 -1
  139. package/dialog/ScalePoolDownDialog.vue +121 -0
  140. package/edit/__tests__/management.cattle.io.setting.test.ts +3 -3
  141. package/edit/auth/azuread.vue +16 -16
  142. package/edit/auth/github.vue +8 -0
  143. package/edit/auth/googleoauth.vue +10 -1
  144. package/edit/auth/ldap/index.vue +10 -0
  145. package/edit/auth/oidc.vue +10 -0
  146. package/edit/auth/saml.vue +10 -0
  147. package/edit/autoscaling.horizontalpodautoscaler/index.vue +1 -1
  148. package/edit/catalog.cattle.io.clusterrepo.vue +3 -0
  149. package/edit/cloudcredential.vue +3 -7
  150. package/edit/logging-flow/Match.vue +39 -8
  151. package/edit/logging-flow/index.vue +27 -4
  152. package/edit/management.cattle.io.podsecurityadmissionconfigurationtemplate.vue +107 -0
  153. package/edit/management.cattle.io.project.vue +8 -1
  154. package/edit/management.cattle.io.setting.vue +5 -2
  155. package/edit/management.cattle.io.user.vue +7 -1
  156. package/edit/monitoring.coreos.com.alertmanagerconfig/receiverConfig.vue +36 -8
  157. package/edit/monitoring.coreos.com.alertmanagerconfig/types/email.vue +2 -2
  158. package/edit/monitoring.coreos.com.prometheusrule/GroupRules.vue +14 -6
  159. package/edit/namespace.vue +18 -4
  160. package/edit/networking.k8s.io.ingress/Certificate.vue +1 -0
  161. package/edit/networking.k8s.io.ingress/IngressClass.vue +8 -6
  162. package/edit/networking.k8s.io.ingress/RulePath.vue +12 -6
  163. package/edit/networking.k8s.io.ingress/index.vue +8 -6
  164. package/edit/persistentvolume/index.vue +30 -27
  165. package/edit/persistentvolume/plugins/cephfs.vue +29 -29
  166. package/edit/persistentvolume/plugins/csi.vue +102 -62
  167. package/edit/persistentvolume/plugins/fc.vue +19 -19
  168. package/edit/persistentvolume/plugins/iscsi.vue +45 -45
  169. package/edit/persistentvolume/plugins/rbd.vue +39 -39
  170. package/edit/persistentvolumeclaim.vue +78 -75
  171. package/edit/provisioning.cattle.io.cluster/MachinePool.vue +11 -7
  172. package/edit/provisioning.cattle.io.cluster/RegistryConfigs.vue +10 -1
  173. package/edit/provisioning.cattle.io.cluster/RegistryMirrors.vue +87 -27
  174. package/edit/provisioning.cattle.io.cluster/SelectCredential.vue +3 -6
  175. package/edit/provisioning.cattle.io.cluster/__tests__/rke2.test.ts +96 -0
  176. package/edit/provisioning.cattle.io.cluster/import.vue +1 -1
  177. package/edit/provisioning.cattle.io.cluster/index.vue +29 -6
  178. package/edit/provisioning.cattle.io.cluster/rke2.vue +445 -154
  179. package/edit/secret/index.vue +3 -7
  180. package/edit/service.vue +3 -1
  181. package/edit/storage.k8s.io.storageclass/index.vue +100 -16
  182. package/edit/storage.k8s.io.storageclass/provisioners/driver.harvesterhci.io.vue +114 -0
  183. package/edit/workload/__tests__/index.test.ts +98 -0
  184. package/edit/workload/index.vue +58 -8
  185. package/edit/workload/mixins/workload.js +107 -70
  186. package/edit/workload/storage/ContainerMountPaths.vue +0 -10
  187. package/edit/workload/storage/emptyDir.vue +88 -0
  188. package/edit/workload/storage/ephemeralVolume/index.vue +1 -1
  189. package/edit/workload/storage/index.vue +8 -0
  190. package/edit/workload/storage/persistentVolumeClaim/index.vue +1 -1
  191. package/layouts/default.vue +57 -44
  192. package/list/__tests__/workload.test.ts +5 -2
  193. package/list/catalog.cattle.io.app.vue +1 -0
  194. package/list/cis.cattle.io.clusterscan.vue +1 -0
  195. package/list/fleet.cattle.io.bundle.vue +5 -6
  196. package/list/fleet.cattle.io.cluster.vue +6 -3
  197. package/list/fleet.cattle.io.clusterregistrationtoken.vue +5 -6
  198. package/list/fleet.cattle.io.gitrepo.vue +4 -9
  199. package/list/helm.cattle.io.projecthelmchart.vue +1 -5
  200. package/list/logging.banzaicloud.io.clusterflow.vue +4 -1
  201. package/list/logging.banzaicloud.io.flow.vue +6 -5
  202. package/list/management.cattle.io.cluster.vue +1 -0
  203. package/list/management.cattle.io.feature.vue +3 -4
  204. package/list/management.cattle.io.podsecurityadmissionconfigurationtemplate.vue +47 -0
  205. package/list/management.cattle.io.setting.vue +2 -2
  206. package/list/management.cattle.io.user.vue +4 -10
  207. package/list/monitoring.coreos.com.alertmanagerconfig.vue +2 -7
  208. package/list/node.vue +8 -5
  209. package/list/persistentvolume.vue +3 -3
  210. package/list/persistentvolumeclaim.vue +3 -4
  211. package/list/provisioning.cattle.io.cluster.vue +18 -19
  212. package/list/service.vue +6 -14
  213. package/list/workload.vue +43 -38
  214. package/machine-config/azure.vue +429 -60
  215. package/machine-config/pnap.vue +288 -0
  216. package/mixins/auth-config.js +1 -3
  217. package/mixins/browser-tab-visibility.js +8 -14
  218. package/mixins/chart.js +1 -1
  219. package/mixins/create-edit-view/impl.js +4 -0
  220. package/mixins/create-edit-view/index.js +4 -2
  221. package/mixins/resource-fetch-namespaced.js +98 -0
  222. package/mixins/resource-fetch.js +79 -45
  223. package/mixins/resource-manager.js +1 -23
  224. package/models/apps.controllerrevision.js +7 -0
  225. package/models/apps.daemonset.js +18 -0
  226. package/models/apps.deployment.js +44 -0
  227. package/models/apps.replicaset.js +7 -0
  228. package/models/apps.statefulset.js +18 -0
  229. package/models/batch.job.js +7 -14
  230. package/models/cluster/node.js +10 -2
  231. package/models/cluster.x-k8s.io.machine.js +26 -4
  232. package/models/cluster.x-k8s.io.machinedeployment.js +12 -2
  233. package/models/event.js +7 -0
  234. package/models/logging.banzaicloud.io.flow.js +4 -0
  235. package/models/management.cattle.io.cluster.js +1 -1
  236. package/models/management.cattle.io.clusterroletemplatebinding.js +1 -1
  237. package/models/management.cattle.io.globalrole.js +2 -2
  238. package/models/management.cattle.io.node.js +37 -2
  239. package/models/management.cattle.io.podsecurityadmissionconfigurationtemplate.ts +4 -0
  240. package/models/management.cattle.io.project.js +30 -11
  241. package/models/management.cattle.io.setting.js +1 -1
  242. package/models/management.cattle.io.user.js +37 -1
  243. package/models/namespace.js +42 -5
  244. package/models/persistentvolume.js +14 -2
  245. package/models/pod.js +15 -0
  246. package/models/projectroletemplatebinding.js +7 -0
  247. package/models/provisioning.cattle.io.cluster.js +61 -10
  248. package/models/rke-machine.cattle.io.pnapmachinetemplate.js +15 -0
  249. package/models/service.js +14 -13
  250. package/models/storage.k8s.io.storageclass.js +33 -18
  251. package/models/workload.js +38 -7
  252. package/nuxt.config.js +27 -17
  253. package/package.json +7 -7
  254. package/pages/about.vue +14 -2
  255. package/pages/c/_cluster/apps/charts/index.vue +21 -3
  256. package/pages/c/_cluster/apps/charts/install.vue +59 -22
  257. package/pages/c/_cluster/auth/config/_id.vue +6 -0
  258. package/pages/c/_cluster/auth/config/index.vue +8 -6
  259. package/pages/c/_cluster/auth/group.principal/assign-edit.vue +1 -1
  260. package/pages/c/_cluster/auth/roles/index.vue +1 -1
  261. package/pages/c/_cluster/explorer/index.vue +51 -6
  262. package/pages/c/_cluster/longhorn/index.vue +1 -1
  263. package/pages/c/_cluster/monitoring/alertmanagerconfig/_alertmanagerconfigid/receiver.vue +15 -4
  264. package/pages/c/_cluster/monitoring/index.vue +1 -1
  265. package/pages/c/_cluster/neuvector/index.vue +1 -1
  266. package/pages/c/_cluster/settings/performance.vue +48 -2
  267. package/pages/c/_cluster/uiplugins/DeveloperInstallDialog.vue +2 -0
  268. package/pages/c/_cluster/uiplugins/InstallDialog.vue +3 -0
  269. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +42 -2
  270. package/pages/c/_cluster/uiplugins/RemoveUIPlugins.vue +2 -0
  271. package/pages/c/_cluster/uiplugins/SetupUIPlugins.vue +1 -0
  272. package/pages/c/_cluster/uiplugins/UninstallDialog.vue +2 -0
  273. package/pages/c/_cluster/uiplugins/index.vue +42 -3
  274. package/pages/diagnostic.vue +5 -4
  275. package/pages/home.vue +105 -30
  276. package/pages/prefs.vue +23 -12
  277. package/pages/rio/mesh.vue +1 -1
  278. package/pkg/dynamic-importer.lib.js +8 -0
  279. package/pkg/vue.config.js +4 -0
  280. package/plugins/dashboard-store/__tests__/mutations.spec.js +406 -0
  281. package/plugins/dashboard-store/actions.js +32 -25
  282. package/plugins/dashboard-store/getters.js +50 -33
  283. package/plugins/dashboard-store/mutations.js +134 -28
  284. package/plugins/dashboard-store/resource-class.js +37 -42
  285. package/plugins/steve/actions.js +30 -0
  286. package/plugins/steve/caches/resourceCache.js +60 -0
  287. package/plugins/steve/getters.js +44 -1
  288. package/plugins/steve/mutations.js +97 -36
  289. package/plugins/steve/resourceWatcher.js +277 -0
  290. package/plugins/steve/schema.utils.js +25 -0
  291. package/plugins/steve/subscribe.js +288 -115
  292. package/plugins/steve/worker/index.js +17 -0
  293. package/plugins/steve/worker/web-worker.advanced.js +302 -0
  294. package/plugins/steve/{web-worker.steve-sub-worker.js → worker/web-worker.basic.js} +3 -44
  295. package/rancher-components/Card/Card.vue +3 -3
  296. package/rancher-components/Form/TextArea/TextAreaAutoGrow.vue +1 -0
  297. package/rancher-components/StringList/StringList.test.ts +45 -420
  298. package/rancher-components/StringList/StringList.vue +1 -10
  299. package/rancher-components/components/Banner/Banner.test.ts +44 -0
  300. package/rancher-components/components/Banner/Banner.vue +130 -61
  301. package/rancher-components/components/Form/Checkbox/Checkbox.test.ts +13 -22
  302. package/rancher-components/components/Form/Checkbox/Checkbox.vue +8 -6
  303. package/rancher-components/components/Form/ToggleSwitch/ToggleSwitch.test.ts +9 -9
  304. package/rancher-components/components/LabeledTooltip/LabeledTooltip.vue +0 -1
  305. package/rancher-components/components/StringList/StringList.test.ts +7 -7
  306. package/rancher-components/components/StringList/StringList.vue +21 -15
  307. package/scripts/test-plugins-build.sh +8 -0
  308. package/static/loading-indicator.html +1 -1
  309. package/store/action-menu.js +4 -3
  310. package/store/index.js +54 -3
  311. package/store/plugins.js +0 -17
  312. package/store/pnap.js +128 -0
  313. package/store/prefs.js +4 -2
  314. package/store/type-map.js +81 -13
  315. package/types/pod-security-admission.ts +36 -0
  316. package/types/shell/index.d.ts +497 -396
  317. package/utils/__tests__/object.test.ts +17 -1
  318. package/utils/__tests__/pod-security-admission.test.ts +61 -0
  319. package/utils/async.ts +36 -0
  320. package/utils/color.js +45 -0
  321. package/utils/crypto/browserHashUtils.js +18 -0
  322. package/utils/dynamic-importer.js +8 -0
  323. package/utils/install-redirect.js +1 -1
  324. package/utils/object.js +24 -0
  325. package/utils/pod-security-admission.ts +39 -0
  326. package/utils/socket.js +61 -24
  327. package/utils/string.js +2 -0
  328. package/utils/svg-filter.js +301 -0
  329. package/utils/time.js +49 -0
  330. package/utils/validators/cidr.js +4 -0
  331. package/utils/validators/formRules/__tests__/index.test.ts +23 -3
  332. package/utils/validators/formRules/index.ts +14 -0
  333. package/config/product/harvester-manager.js +0 -162
  334. package/edit/harvesterhci.io.management.cluster.vue +0 -153
  335. package/list/harvesterhci.io.management.cluster.vue +0 -241
  336. package/machine-config/harvester.vue +0 -693
  337. package/models/harvesterhci.io.management.cluster.js +0 -228
  338. package/pages/c/_cluster/harvesterManager/index.vue +0 -24
  339. package/rancher-components/Card/Card.test.ts +0 -39
  340. package/rancher-components/Utils/DraggableZone/DraggableZone.vue +0 -181
  341. package/rancher-components/Utils/DraggableZone/index.ts +0 -1
  342. package/rancher-components/components/Utils/DraggableZone/index.ts +0 -1
@@ -0,0 +1,106 @@
1
+ <script>
2
+ import { Card } from '@components/Card';
3
+ export default {
4
+ name: 'PromptRemove',
5
+ components: { Card },
6
+ props: {
7
+ /**
8
+ * Inherited global identifier prefix for tests
9
+ * Define a term based on the parent component to avoid conflicts on multiple components
10
+ */
11
+ componentTestid: {
12
+ type: String,
13
+ default: 'disable-auth-provider'
14
+ }
15
+ },
16
+ methods: {
17
+ show() {
18
+ this.$modal.show('disableAuthProviderModal');
19
+ },
20
+ close() {
21
+ this.$modal.hide('disableAuthProviderModal');
22
+ },
23
+ disable() {
24
+ this.$modal.hide('disableAuthProviderModal');
25
+ this.$emit('disable');
26
+ },
27
+ }
28
+ };
29
+ </script>
30
+
31
+ <template>
32
+ <modal
33
+ class="remove-modal"
34
+ name="disableAuthProviderModal"
35
+ :width="400"
36
+ height="auto"
37
+ styles="max-height: 100vh;"
38
+ @closed="close"
39
+ >
40
+ <Card
41
+ class="disable-auth-provider"
42
+ :show-highlight-border="false"
43
+ >
44
+ <h4
45
+ slot="title"
46
+ class="text-default-text"
47
+ >
48
+ {{ t('promptRemove.title') }}
49
+ </h4>
50
+ <div slot="body">
51
+ <div class="mb-10">
52
+ <p v-html="t('promptRemove.attemptingToRemoveAuthConfig', null, true)" />
53
+ </div>
54
+ </div>
55
+ <template #actions>
56
+ <button
57
+ class="btn role-secondary"
58
+ @click="close"
59
+ >
60
+ {{ t('generic.cancel') }}
61
+ </button>
62
+ <div class="spacer" />
63
+ <button
64
+ class="btn role-primary bg-error ml-10"
65
+ :data-testid="componentTestid + '-confirm-button'"
66
+ @click="disable"
67
+ >
68
+ {{ t('generic.disable') }}
69
+ </button>
70
+ </template>
71
+ </Card>
72
+ </modal>
73
+ </template>
74
+
75
+ <style lang='scss'>
76
+ .disable-auth-provider {
77
+ &.card-container {
78
+ box-shadow: none;
79
+ }
80
+ #confirm {
81
+ width: 90%;
82
+ margin-left: 3px;
83
+ }
84
+
85
+ .remove-modal {
86
+ border-radius: var(--border-radius);
87
+ overflow: scroll;
88
+ max-height: 100vh;
89
+ & ::-webkit-scrollbar-corner {
90
+ background: rgba(0,0,0,0);
91
+ }
92
+ }
93
+
94
+ .actions {
95
+ text-align: right;
96
+ }
97
+
98
+ .card-actions {
99
+ display: flex;
100
+
101
+ .spacer {
102
+ flex: 1;
103
+ }
104
+ }
105
+ }
106
+ </style>
@@ -5,7 +5,12 @@ import Masthead from '@shell/components/ResourceList/Masthead';
5
5
  import { AGE, ROLE, STATE, PRINCIPAL } from '@shell/config/table-headers';
6
6
  import { canViewClusterPermissionsEditor } from '@shell/components/form/Members/ClusterPermissionsEditor.vue';
7
7
  import Banner from '@components/Banner/Banner.vue';
8
+ import Tabbed from '@shell/components/Tabbed/index.vue';
9
+ import Tab from '@shell/components/Tabbed/Tab.vue';
10
+ import SortableTable from '@shell/components/SortableTable';
8
11
  import { mapGetters } from 'vuex';
12
+ import { canViewProjectMembershipEditor } from '@shell/components/form/Members/ProjectMembershipEditor.vue';
13
+ import { allHash } from '@shell/utils/promise';
9
14
 
10
15
  /**
11
16
  * Explorer members page.
@@ -17,7 +22,10 @@ export default {
17
22
  components: {
18
23
  Banner,
19
24
  Masthead,
20
- ResourceTable
25
+ ResourceTable,
26
+ Tabbed,
27
+ Tab,
28
+ SortableTable
21
29
  },
22
30
 
23
31
  props: {
@@ -38,19 +46,39 @@ export default {
38
46
  `rancher/schemaFor`
39
47
  ](NORMAN.CLUSTER_ROLE_TEMPLATE_BINDING);
40
48
 
41
- const hydration = [
42
- clusterRoleTemplateBindingSchema ? this.$store.dispatch(
43
- `rancher/findAll`,
44
- { type: NORMAN.CLUSTER_ROLE_TEMPLATE_BINDING },
45
- { root: true }
46
- ) : [],
47
- clusterRoleTemplateBindingSchema ? this.$store.dispatch(`management/findAll`, { type: MANAGEMENT.CLUSTER_ROLE_TEMPLATE_BINDING }) : [],
48
- this.$store.dispatch('rancher/findAll', { type: NORMAN.PRINCIPAL }),
49
- this.$store.dispatch(`management/findAll`, { type: MANAGEMENT.USER }),
50
- this.$store.dispatch(`management/findAll`, { type: MANAGEMENT.ROLE_TEMPLATE })
51
- ];
52
-
53
- await Promise.all(hydration);
49
+ const projectRoleTemplateBindingSchema = this.$store.getters['rancher/schemaFor'](NORMAN.PROJECT_ROLE_TEMPLATE_BINDING);
50
+
51
+ this.$set(this, 'normanClusterRTBSchema', clusterRoleTemplateBindingSchema);
52
+ this.$set(this, 'normanProjectRTBSchema', projectRoleTemplateBindingSchema);
53
+
54
+ if (clusterRoleTemplateBindingSchema) {
55
+ Promise.all([
56
+ this.$store.dispatch(`rancher/findAll`, { type: NORMAN.CLUSTER_ROLE_TEMPLATE_BINDING }, { root: true }),
57
+ this.$store.dispatch(`management/findAll`, { type: MANAGEMENT.CLUSTER_ROLE_TEMPLATE_BINDING })
58
+ ]).then(([normanBindings]) => {
59
+ this.$set(this, 'normanClusterRoleTemplateBindings', normanBindings);
60
+ this.loadingClusterBindings = false;
61
+ });
62
+ }
63
+
64
+ if (projectRoleTemplateBindingSchema) {
65
+ this.$store.dispatch('rancher/findAll', { type: NORMAN.PROJECT_ROLE_TEMPLATE_BINDING }, { root: true })
66
+ .then((bindings) => {
67
+ this.$set(this, 'projectRoleTemplateBindings', bindings);
68
+ this.loadingProjectBindings = false;
69
+ });
70
+ }
71
+
72
+ this.$store.dispatch('management/findAll', { type: MANAGEMENT.PROJECT })
73
+ .then(projects => this.$set(this, 'projects', projects));
74
+
75
+ const hydration = {
76
+ normanPrincipals: this.$store.dispatch('rancher/findAll', { type: NORMAN.PRINCIPAL }),
77
+ mgmt: this.$store.dispatch(`management/findAll`, { type: MANAGEMENT.USER }),
78
+ mgmtRoleTemplates: this.$store.dispatch(`management/findAll`, { type: MANAGEMENT.ROLE_TEMPLATE }),
79
+ };
80
+
81
+ await allHash(hydration);
54
82
  },
55
83
 
56
84
  data() {
@@ -66,54 +94,249 @@ export default {
66
94
  cluster: this.$store.getters['currentCluster'].id
67
95
  }
68
96
  },
69
- resource: MANAGEMENT.CLUSTER_ROLE_TEMPLATE_BINDING,
70
- VIRTUAL_TYPES
97
+ resource: MANAGEMENT.CLUSTER_ROLE_TEMPLATE_BINDING,
98
+ normanClusterRTBSchema: null,
99
+ normanProjectRTBSchema: null,
100
+ normanClusterRoleTemplateBindings: [],
101
+ projectRoleTemplateBindings: [],
102
+ projects: [],
103
+ VIRTUAL_TYPES,
104
+ projectRoleTemplateColumns: [
105
+ STATE,
106
+ {
107
+ name: 'member',
108
+ labeKey: 'generic.name',
109
+ value: 'principalId',
110
+ formatter: 'Principal'
111
+ },
112
+ {
113
+ name: 'role',
114
+ labelKey: 'tableHeaders.role',
115
+ value: 'roleTemplate.nameDisplay'
116
+ },
117
+ { ...AGE, value: 'createdTS' }
118
+ ],
119
+ loadingProjectBindings: true,
120
+ loadingClusterBindings: true
71
121
  };
72
122
  },
73
123
 
74
124
  computed: {
125
+ ...mapGetters(['currentCluster']),
75
126
  clusterRoleTemplateBindings() {
76
- return this.$store.getters[`rancher/all`](NORMAN.CLUSTER_ROLE_TEMPLATE_BINDING).map(b => b.clusterroletemplatebinding) ;
127
+ return this.normanClusterRoleTemplateBindings.map(b => b.clusterroletemplatebinding) ;
77
128
  },
78
- ...mapGetters(['currentCluster']),
79
129
  filteredClusterRoleTemplateBindings() {
80
130
  return this.clusterRoleTemplateBindings.filter(
81
131
  b => b?.clusterName === this.$store.getters['currentCluster'].id
82
132
  );
83
133
  },
134
+ filteredProjects() {
135
+ return this.projects.reduce((all, p) => {
136
+ if (p?.spec?.clusterName === this.currentCluster.id) {
137
+ all[p.id] = p;
138
+ }
139
+
140
+ return all;
141
+ }, {});
142
+ },
143
+ filteredProjectRoleTemplateBindings() {
144
+ const out = this.projectRoleTemplateBindings.filter((rb) => {
145
+ const projectId = rb.projectId.replace(':', '/');
146
+
147
+ return !!this.filteredProjects[projectId];
148
+ });
149
+
150
+ return out;
151
+ },
152
+ projectsWithoutRoles() {
153
+ const inUse = this.filteredProjectRoleTemplateBindings.reduce((projects, binding) => {
154
+ const thisProjectId = (binding.projectId || '').replace(':', '/');
155
+
156
+ if (!projects.includes(thisProjectId)) {
157
+ projects.push(thisProjectId);
158
+ }
159
+
160
+ return projects;
161
+ }, []);
162
+
163
+ return Object.keys(this.filteredProjects).reduce((all, projectId) => {
164
+ const project = this.filteredProjects[projectId];
165
+
166
+ if ( !inUse.includes(projectId)) {
167
+ all.push(project);
168
+ }
169
+
170
+ return all;
171
+ }, []);
172
+ },
173
+
174
+ // We're using this because we need to show projects as groups even if the project doesn't have any role bindings
175
+ rowsWithFakeProjects() {
176
+ const fakeRows = this.projectsWithoutRoles.map((project) => {
177
+ return {
178
+ groupByLabel: `${ ('resourceTable.groupLabel.notInAProject') }-${ project.id }`,
179
+ isFake: true,
180
+ mainRowKey: project.id,
181
+ nameDisplay: project.spec?.displayName, // Enable filtering by the project name
182
+ project,
183
+ availableActions: [],
184
+ projectId: project.id
185
+ };
186
+ });
187
+
188
+ return [...fakeRows, ...this.filteredProjectRoleTemplateBindings];
189
+ },
84
190
  canManageMembers() {
85
191
  return canViewClusterPermissionsEditor(this.$store);
86
192
  },
193
+ canManageProjectMembers() {
194
+ return canViewProjectMembershipEditor(this.$store);
195
+ },
87
196
  isLocal() {
88
197
  return this.$store.getters['currentCluster'].isLocal;
89
- }
198
+ },
199
+ canEditProjectMembers() {
200
+ return this.normanProjectRTBSchema?.collectionMethods.find(x => x.toLowerCase() === 'post');
201
+ },
202
+ canEditClusterMembers() {
203
+ return this.normanClusterRTBSchema?.collectionMethods.find(x => x.toLowerCase() === 'post');
204
+ },
205
+ },
206
+ methods: {
207
+ getMgmtProjectId(group) {
208
+ return group.group.key.replace(':', '/');
209
+ },
210
+ getMgmtProject(group) {
211
+ return this.$store.getters['management/byId'](MANAGEMENT.PROJECT, this.getMgmtProjectId(group));
212
+ },
213
+ getProjectLabel(group) {
214
+ return this.getMgmtProject(group)?.spec?.displayName;
215
+ },
216
+ addProjectMember(group) {
217
+ this.$store.dispatch('cluster/promptModal', {
218
+ component: 'AddProjectMemberDialog',
219
+ componentProps: {
220
+ projectId: group.group.key.replace('/', ':'),
221
+ saveInModal: true
222
+ },
223
+ modalSticky: true
224
+ });
225
+ },
226
+ slotName(project) {
227
+ return `main-row:${ project.id }`;
228
+ },
90
229
  }
91
230
  };
92
231
  </script>
93
232
 
94
233
  <template>
95
- <div>
234
+ <div class="project-members">
96
235
  <Masthead
97
236
  :schema="schema"
98
237
  :resource="resource"
99
238
  :favorite-resource="VIRTUAL_TYPES.CLUSTER_MEMBERS"
100
239
  :create-location="createLocation"
101
240
  :create-button-label="t('members.createActionLabel')"
241
+ :is-creatable="false"
242
+ :type-display="t('members.clusterAndProject')"
102
243
  />
103
244
  <Banner
104
245
  v-if="isLocal"
105
246
  color="error"
106
247
  :label="t('members.localClusterWarning')"
107
248
  />
108
- <ResourceTable
109
- :schema="schema"
110
- :headers="headers"
111
- :rows="$fetchState.pending ? [] : filteredClusterRoleTemplateBindings"
112
- :groupable="false"
113
- :namespaced="false"
114
- :loading="$fetchState.pending || !currentCluster"
115
- sub-search="subSearch"
116
- :sub-fields="['nameDisplay']"
117
- />
249
+ <Tabbed>
250
+ <Tab
251
+ name="cluster-membership"
252
+ label="Cluster Membership"
253
+ >
254
+ <div
255
+ v-if="canEditClusterMembers"
256
+ class="row mb-10 cluster-add"
257
+ >
258
+ <n-link
259
+ :to="createLocation"
260
+ class="btn role-primary pull-right"
261
+ >
262
+ {{ t('members.createActionLabel') }}
263
+ </n-link>
264
+ </div>
265
+ <ResourceTable
266
+ :schema="schema"
267
+ :headers="headers"
268
+ :rows="filteredClusterRoleTemplateBindings"
269
+ :groupable="false"
270
+ :namespaced="false"
271
+ :loading="$fetchState.pending || !currentCluster || loadingClusterBindings"
272
+ sub-search="subSearch"
273
+ :sub-fields="['nameDisplay']"
274
+ />
275
+ </Tab>
276
+ <Tab
277
+ v-if="canManageProjectMembers"
278
+ name="project-membership"
279
+ label="Project Membership"
280
+ >
281
+ <SortableTable
282
+ group-by="projectId"
283
+ :loading="$fetchState.pending || !currentCluster || loadingProjectBindings"
284
+ :rows="rowsWithFakeProjects"
285
+ :headers="projectRoleTemplateColumns"
286
+ >
287
+ <template #group-by="group">
288
+ <div class="group-bar">
289
+ <div
290
+ v-trim-whitespace
291
+ class="group-tab"
292
+ >
293
+ <div
294
+ class="project-name"
295
+ v-html="getProjectLabel(group)"
296
+ />
297
+ </div>
298
+ <div class="right">
299
+ <button
300
+ v-if="canEditProjectMembers"
301
+ type="button"
302
+ class="create-namespace btn btn-sm role-secondary mr-5 right"
303
+ @click="addProjectMember(group)"
304
+ >
305
+ {{ t('members.createActionLabel') }}
306
+ </button>
307
+ </div>
308
+ </div>
309
+ </template>
310
+ <template
311
+ v-for="project in projectsWithoutRoles"
312
+ v-slot:[slotName(project)]
313
+ >
314
+ <tr
315
+ :key="project.id"
316
+ class="main-row"
317
+ >
318
+ <td
319
+ class="empty text-center"
320
+ colspan="100%"
321
+ >
322
+ {{ t('members.noRolesAssigned') }}
323
+ </td>
324
+ </tr>
325
+ </template>
326
+ </SortableTable>
327
+ </Tab>
328
+ </Tabbed>
118
329
  </div>
119
330
  </template>
331
+
332
+ <style lang='scss' scoped>
333
+ .project-members {
334
+ & ::v-deep .group-bar{
335
+ display: flex;
336
+ justify-content: space-between;
337
+ }
338
+ }
339
+ .cluster-add {
340
+ justify-content: flex-end;
341
+ }
342
+ </style>
@@ -86,7 +86,7 @@ export default {
86
86
  headers() {
87
87
  const project = {
88
88
  name: 'project',
89
- label: 'Project',
89
+ label: this.t('tableHeaders.project'),
90
90
  value: 'project.nameDisplay',
91
91
  sort: ['projectNameSort', 'nameSort'],
92
92
  };
@@ -235,6 +235,19 @@ export default {
235
235
  }
236
236
  },
237
237
  methods: {
238
+ /**
239
+ * Get PSA HTML to be displayed in the tooltips
240
+ */
241
+ getPsaTooltip(row) {
242
+ const dictionary = row.psaTooltipsDescription;
243
+ const list = Object.values(dictionary)
244
+ .sort()
245
+ .map(text => `<li>${ text }</li>`).join('');
246
+ const title = `<p>${ this.t('podSecurityAdmission.name') }: </p>`;
247
+
248
+ return `${ title }<ul class="psa-tooltip">${ list }</ul>`;
249
+ },
250
+
238
251
  userIsFilteringForSpecificNamespaceOrProject() {
239
252
  const activeFilters = this.$store.getters['namespaceFilters'];
240
253
 
@@ -342,21 +355,21 @@ export default {
342
355
  v-on="$listeners"
343
356
  >
344
357
  <template #group-by="group">
345
- <div
346
- class="project-bar"
347
- :class="{'has-description': projectDescription(group.group)}"
358
+ <div
359
+ class="project-bar"
360
+ :class="{'has-description': projectDescription(group.group)}"
348
361
  >
349
- <div
350
- v-trim-whitespace
351
- class="group-tab"
362
+ <div
363
+ v-trim-whitespace
364
+ class="group-tab"
352
365
  >
353
- <div
354
- class="project-name"
355
- v-html="projectLabel(group.group)"
366
+ <div
367
+ class="project-name"
368
+ v-html="projectLabel(group.group)"
356
369
  />
357
- <div
358
- v-if="projectDescription(group.group)"
359
- class="description text-muted text-small"
370
+ <div
371
+ v-if="projectDescription(group.group)"
372
+ class="description text-muted text-small"
360
373
  >
361
374
  {{ projectDescription(group.group) }}
362
375
  </div>
@@ -369,11 +382,11 @@ export default {
369
382
  >
370
383
  {{ t('projectNamespaces.createNamespace') }}
371
384
  </n-link>
372
- <button
373
- type="button"
374
- class="project-action btn btn-sm role-multi-action actions mr-10"
375
- :class="{invisible: !showProjectActionButton(group.group)}"
376
- @click="showProjectAction($event, group.group)"
385
+ <button
386
+ type="button"
387
+ class="project-action btn btn-sm role-multi-action actions mr-10"
388
+ :class="{invisible: !showProjectActionButton(group.group)}"
389
+ @click="showProjectAction($event, group.group)"
377
390
  >
378
391
  <i class="icon icon-actions" />
379
392
  </button>
@@ -382,22 +395,40 @@ export default {
382
395
  </template>
383
396
  <template #cell:project="{row}">
384
397
  <span v-if="row.project">{{ row.project.nameDisplay }}</span>
385
- <span
386
- v-else
387
- class="text-muted"
398
+ <span
399
+ v-else
400
+ class="text-muted"
388
401
  >&ndash;</span>
389
402
  </template>
390
- <template
391
- v-for="project in projectsWithoutNamespaces"
392
- v-slot:[slotName(project)]
403
+ <template #cell:name="{row}">
404
+ <div class="namespace-name">
405
+ <n-link
406
+ v-if="row.detailLocation"
407
+ :to="row.detailLocation"
408
+ >
409
+ {{ row.name }}
410
+ </n-link>
411
+ <span v-else>
412
+ {{ row.name }}
413
+ </span>
414
+ <i
415
+ v-if="row.hasSystemLabels"
416
+ v-tooltip="getPsaTooltip(row)"
417
+ class="icon icon-lock ml-5"
418
+ />
419
+ </div>
420
+ </template>
421
+ <template
422
+ v-for="project in projectsWithoutNamespaces"
423
+ v-slot:[slotName(project)]
393
424
  >
394
- <tr
395
- :key="project.id"
396
- class="main-row"
425
+ <tr
426
+ :key="project.id"
427
+ class="main-row"
397
428
  >
398
- <td
399
- class="empty text-center"
400
- colspan="5"
429
+ <td
430
+ class="empty text-center"
431
+ colspan="5"
401
432
  >
402
433
  {{ t('projectNamespaces.noNamespaces') }}
403
434
  </td>
@@ -405,9 +436,9 @@ export default {
405
436
  </template>
406
437
  <template #main-row:fake-empty>
407
438
  <tr class="main-row">
408
- <td
409
- class="empty text-center"
410
- colspan="5"
439
+ <td
440
+ class="empty text-center"
441
+ colspan="5"
411
442
  >
412
443
  {{ t('projectNamespaces.noProjectNoNamespaces') }}
413
444
  </td>
@@ -448,6 +479,19 @@ export default {
448
479
  }
449
480
  }
450
481
  }
482
+
483
+ .namespace-name {
484
+ display: flex;
485
+ align-items: center;
486
+ }
451
487
  }
452
488
  }
453
489
  </style>
490
+ <style lang="scss">
491
+ .psa-tooltip {
492
+ // These could pop up a lot as the mouse moves around, keep them as small and unintrusive as possible
493
+ // (easier to test with v-tooltip="{ content: getPSA(row), autoHide: false, show: true }")
494
+ margin: 3px 0;
495
+ padding: 0 8px 0 22px;
496
+ }
497
+ </style>
@@ -0,0 +1,42 @@
1
+ <script>
2
+ import { getApplicableExtensionEnhancements } from '@shell/core/plugin-helpers';
3
+
4
+ export default {
5
+ name: 'ExtensionPanel',
6
+ props: {
7
+ resource: {
8
+ type: Object,
9
+ default: () => {
10
+ return {};
11
+ }
12
+ },
13
+ type: {
14
+ type: String,
15
+ default: ''
16
+ },
17
+ location: {
18
+ type: String,
19
+ default: ''
20
+ },
21
+ },
22
+ data() {
23
+ return { extensionData: getApplicableExtensionEnhancements(this, this.type, this.location, this.$route) };
24
+ },
25
+ };
26
+ </script>
27
+
28
+ <template>
29
+ <div
30
+ v-if="extensionData.length"
31
+ >
32
+ <div
33
+ v-for="item, i in extensionData"
34
+ :key="`extensionData${location}${i}`"
35
+ >
36
+ <component
37
+ :is="item.component"
38
+ :resource="resource"
39
+ />
40
+ </div>
41
+ </div>
42
+ </template>
@@ -86,8 +86,9 @@ export default {
86
86
  >
87
87
  <div class="growl-list">
88
88
  <div
89
- v-for="growl in stack"
89
+ v-for="(growl, idx) in stack"
90
90
  :key="growl.id"
91
+ :data-testid="`growl-list-item-${idx}`"
91
92
  :class="{'growl': true, ['bg-'+growl.color]: true}"
92
93
  >
93
94
  <div
@@ -117,7 +118,7 @@ export default {
117
118
  <button
118
119
  type="button"
119
120
  class="btn btn-sm role-primary"
120
- @click="closeAll"
121
+ @click="closeAll()"
121
122
  >
122
123
  {{ t('growl.clearAll') }}
123
124
  </button>
@@ -151,7 +152,6 @@ export default {
151
152
 
152
153
  .close {
153
154
  padding: 5px;
154
- font-size: 24px;
155
155
  }
156
156
 
157
157
  .icon-container {