diff --git a/react/src/components/FolderCreateModal.tsx b/react/src/components/FolderCreateModal.tsx index 1e8ed6aa3..619146b3c 100644 --- a/react/src/components/FolderCreateModal.tsx +++ b/react/src/components/FolderCreateModal.tsx @@ -1,6 +1,7 @@ import { useBaiSignedRequestWithPromise } from '../helper'; -import { useCurrentDomainValue } from '../hooks'; -import { useTanMutation } from '../hooks/reactQueryAlias'; +import { useCurrentDomainValue, useSuspendedBackendaiClient } from '../hooks'; +import { useCurrentUserRole } from '../hooks/backendai'; +import { useTanMutation, useTanQuery } from '../hooks/reactQueryAlias'; import { useCurrentProjectValue } from '../hooks/useCurrentProject'; import BAIModal, { BAIModalProps } from './BAIModal'; import Flex from './Flex'; @@ -9,6 +10,7 @@ import StorageSelect from './StorageSelect'; import { App, Button, Divider, Form, Input, Radio, Switch, theme } from 'antd'; import { createStyles } from 'antd-style'; import { FormInstance } from 'antd/lib'; +import _ from 'lodash'; import { Suspense, useRef } from 'react'; import { useTranslation } from 'react-i18next'; @@ -76,11 +78,21 @@ const FolderCreateModal: React.FC = ({ const { message } = App.useApp(); const formRef = useRef(null); + const baiClient = useSuspendedBackendaiClient(); + const userRole = useCurrentUserRole(); const currentDomain = useCurrentDomainValue(); const currentProject = useCurrentProjectValue(); const baiRequestWithPromise = useBaiSignedRequestWithPromise(); + const { data: allowedTypes, isFetching: isFetchingAllowedTypes } = + useTanQuery({ + queryKey: ['allowedTypes', modalProps.open], + enabled: modalProps.open, + queryFn: () => + modalProps.open ? baiClient.vfolder.list_allowed_types() : undefined, + }); + const mutationToCreateFolder = useTanMutation< FolderCreationResponse, { message?: string }, @@ -135,6 +147,7 @@ const FolderCreateModal: React.FC = ({ return ( = ({ } width={650} + okButtonProps={{ loading: mutationToCreateFolder.isPending }} onCancel={() => { onRequestClose(); }} @@ -224,8 +238,19 @@ const FolderCreateModal: React.FC = ({ style={{ flex: 1, marginBottom: 0 }} > - User - Project + {/* Both checks are required: + * - role check (admin/superadmin): Controls permission to create project folders + * - allowedTypes check: Ensures the 'group' type is registered in ETCD + * allowedTypes comes from ETCD and contains all registered types regardless of permissions, + * so we need both checks for proper access control + */} + {_.includes(allowedTypes, 'user') ? ( + User + ) : null} + {(userRole === 'admin' || userRole === 'superadmin') && + _.includes(allowedTypes, 'group') ? ( + Project + ) : null} diff --git a/react/src/components/StorageStatusPanel.tsx b/react/src/components/StorageStatusPanel.tsx index 6d3c65ddc..be05efa25 100644 --- a/react/src/components/StorageStatusPanel.tsx +++ b/react/src/components/StorageStatusPanel.tsx @@ -1,5 +1,6 @@ import { addQuotaScopeTypePrefix, usageIndicatorColor } from '../helper'; import { useCurrentDomainValue, useSuspendedBackendaiClient } from '../hooks'; +import { useCurrentUserRole } from '../hooks/backendai'; import { useSuspenseTanQuery } from '../hooks/reactQueryAlias'; import { useCurrentProjectValue } from '../hooks/useCurrentProject'; import Flex from './Flex'; @@ -35,6 +36,7 @@ const StorageStatusPanel: React.FC<{ const { token } = theme.useToken(); const baiClient = useSuspendedBackendaiClient(); const currentProject = useCurrentProjectValue(); + const currentUserRole = useCurrentUserRole(); const [selectedVolumeInfo, setSelectedVolumeInfo] = useState(); const deferredSelectedVolumeInfo = useDeferredValue(selectedVolumeInfo); @@ -116,6 +118,7 @@ const StorageStatusPanel: React.FC<{ const { user_resource_policy, + project_resource_policy, keypair_resource_policy, project_quota_scope, user_quota_scope, @@ -123,7 +126,7 @@ const StorageStatusPanel: React.FC<{ graphql` query StorageStatusPanelQuery( $user_RP_name: String - # $project_RP_name: String! + $project_RP_name: String! $keypair_resource_policy_name: String $project_quota_scope_id: String! $user_quota_scope_id: String! @@ -133,9 +136,10 @@ const StorageStatusPanel: React.FC<{ user_resource_policy(name: $user_RP_name) @since(version: "23.09.6") { max_vfolder_count } - # project_resource_policy(name: $project_RP_name) @since(version: "23.09.1") { - # max_vfolder_count - # } + project_resource_policy(name: $project_RP_name) + @since(version: "23.09.1") { + max_vfolder_count + } keypair_resource_policy(name: $keypair_resource_policy_name) # use max_vfolder_count in keypair_resource_policy before adding max_vfolder_count in user_resource_policy @deprecatedSince(version: "23.09.4") { @@ -157,7 +161,7 @@ const StorageStatusPanel: React.FC<{ `, { user_RP_name: user?.resource_policy, - // project_RP_name: currentProjectDetail?.resource_policy || "", + project_RP_name: currentProject?.name ?? '', keypair_resource_policy_name: keypair?.resource_policy, project_quota_scope_id: addQuotaScopeTypePrefix( 'project', @@ -171,7 +175,6 @@ const StorageStatusPanel: React.FC<{ !deferredSelectedVolumeInfo?.id, }, ); - // Support version: // keypair resource policy < 23.09.4 // user resource policy, project resource policy >= 23.09.6 @@ -228,6 +231,19 @@ const StorageStatusPanel: React.FC<{ {t('data.ProjectFolder')}: {projectFolderCount} + {(currentUserRole === 'admin' || + currentUserRole === 'superadmin') && + (project_resource_policy?.max_vfolder_count ?? -1) >= 0 ? ( + <> + {' / '} + + {t('data.Limit')}: + + {project_resource_policy?.max_vfolder_count === 0 + ? '∞' + : project_resource_policy?.max_vfolder_count} + + ) : null} diff --git a/react/src/pages/VFolderListPage.tsx b/react/src/pages/VFolderListPage.tsx index f3d8f1006..eafc1ecb3 100644 --- a/react/src/pages/VFolderListPage.tsx +++ b/react/src/pages/VFolderListPage.tsx @@ -40,12 +40,11 @@ const tabParam = withDefault(StringParam, 'general'); const VFolderListPage: React.FC = (props) => { const { t } = useTranslation(); + const { token } = theme.useToken(); const [curTabKey, setCurTabKey] = useQueryParam('tab', tabParam, { updateType: 'replace', }); - const baiClient = useSuspendedBackendaiClient(); const [fetchKey, updateFetchKey] = useUpdatableState('first'); - const dataViewRef = useRef(null); const [isOpenCreateModal, { toggle: openCreateModal }] = useToggle(false); const [inviteFolderId, setInviteFolderId] = useState(null); const [ @@ -53,7 +52,8 @@ const VFolderListPage: React.FC = (props) => { { toggle: toggleImportFromHuggingFaceModal }, ] = useToggle(false); - const { token } = theme.useToken(); + const dataViewRef = useRef(null); + const baiClient = useSuspendedBackendaiClient(); const enableImportFromHuggingFace = baiClient._config.enableImportFromHuggingFace;