mirror of
https://github.com/langgenius/dify.git
synced 2026-05-12 15:58:19 +08:00
feat: refactor access rule management to use updated role and account structures, enhance loading states, and improve role assignment functionality
This commit is contained in:
parent
a3b00a2f83
commit
9609f003c6
@ -26,7 +26,7 @@ const AccessRuleRow = ({
|
||||
onEdit,
|
||||
onAddRole,
|
||||
}: AccessRuleRowProps) => {
|
||||
const { policy, role_ids, account_ids } = rule
|
||||
const { policy, roles, accounts } = rule
|
||||
const { id: policyId, resource_type } = policy
|
||||
|
||||
const handleEdit = useCallback(() => onEdit?.(rule), [onEdit, rule])
|
||||
@ -38,8 +38,8 @@ const AccessRuleRow = ({
|
||||
const handleRemoveRole = useCallback((id: string, type: BindingType) => {
|
||||
const payload = {
|
||||
id: policyId,
|
||||
role_ids: role_ids.map(role => role.id),
|
||||
account_ids: account_ids.map(account => account.id),
|
||||
role_ids: roles.map(role => role.role_id),
|
||||
account_ids: accounts.map(account => account.account_id),
|
||||
}
|
||||
if (type === 'role') {
|
||||
payload.role_ids = payload.role_ids.filter(roleId => roleId !== id)
|
||||
@ -61,7 +61,7 @@ const AccessRuleRow = ({
|
||||
},
|
||||
})
|
||||
}
|
||||
}, [account_ids, policyId, resource_type, role_ids, updateAppAccessRuleBindings, updateDatasetAccessRuleBindings])
|
||||
}, [accounts, policyId, resource_type, roles, updateAppAccessRuleBindings, updateDatasetAccessRuleBindings])
|
||||
|
||||
return (
|
||||
<div className={cn('flex items-start gap-2 py-3.5', className)}>
|
||||
@ -73,20 +73,20 @@ const AccessRuleRow = ({
|
||||
{policy.description}
|
||||
</p>
|
||||
<div className="mt-2 flex flex-wrap items-center gap-1.5">
|
||||
{role_ids.map(role => (
|
||||
{roles.map(role => (
|
||||
<RoleTag
|
||||
key={role.id}
|
||||
id={role.id}
|
||||
label={role.name}
|
||||
key={role.role_id}
|
||||
id={role.role_id}
|
||||
label={role.role_name}
|
||||
type="role"
|
||||
onRemove={handleRemoveRole}
|
||||
/>
|
||||
))}
|
||||
{account_ids.map(account => (
|
||||
{accounts.map(account => (
|
||||
<RoleTag
|
||||
key={account.id}
|
||||
id={account.id}
|
||||
label={account.name}
|
||||
key={account.account_id}
|
||||
id={account.account_id}
|
||||
label={account.account_name}
|
||||
type="account"
|
||||
onRemove={handleRemoveRole}
|
||||
/>
|
||||
|
||||
@ -9,6 +9,7 @@ import AccessRuleRow from './access-rule-row'
|
||||
type AccessRuleSectionProps = {
|
||||
title: string
|
||||
rules: AccessPolicyWithBindings[]
|
||||
isLoadingRules: boolean
|
||||
createButtonLabel: string
|
||||
onCreate?: () => void
|
||||
onEditRule?: (rule: AccessPolicyWithBindings) => void
|
||||
@ -19,6 +20,7 @@ type AccessRuleSectionProps = {
|
||||
const AccessRuleSection = ({
|
||||
title,
|
||||
rules,
|
||||
isLoadingRules,
|
||||
createButtonLabel,
|
||||
onCreate,
|
||||
onEditRule,
|
||||
@ -31,7 +33,12 @@ const AccessRuleSection = ({
|
||||
<h3 className="pr-3 system-xs-medium-uppercase tracking-wide text-text-tertiary">
|
||||
{title}
|
||||
</h3>
|
||||
<Button variant="secondary" size="medium" onClick={onCreate}>
|
||||
<Button
|
||||
variant="secondary"
|
||||
size="medium"
|
||||
onClick={onCreate}
|
||||
disabled={isLoadingRules}
|
||||
>
|
||||
{createButtonLabel}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@ -15,7 +15,7 @@ const AppAccessRuleSection = ({
|
||||
onEditRule,
|
||||
onAddRole,
|
||||
}: AppAccessRuleSectionProps) => {
|
||||
const { data: appAccessRulesResponse } = useWorkspaceAppAccessRules({
|
||||
const { data: appAccessRulesResponse, isLoading } = useWorkspaceAppAccessRules({
|
||||
page: 1,
|
||||
limit: 20,
|
||||
})
|
||||
@ -26,6 +26,7 @@ const AppAccessRuleSection = ({
|
||||
<AccessRuleSection
|
||||
title="App Access Rules"
|
||||
rules={appAccessRules}
|
||||
isLoadingRules={isLoading}
|
||||
createButtonLabel="Create App permission set"
|
||||
onCreate={onCreate}
|
||||
onEditRule={onEditRule}
|
||||
|
||||
@ -15,7 +15,7 @@ const DatasetAccessRuleSection = ({
|
||||
onEditRule,
|
||||
onAddRole,
|
||||
}: DatasetAccessRuleSectionProps) => {
|
||||
const { data: datasetAccessRulesResponse } = useWorkspaceDatasetAccessRules({
|
||||
const { data: datasetAccessRulesResponse, isLoading } = useWorkspaceDatasetAccessRules({
|
||||
page: 1,
|
||||
limit: 20,
|
||||
})
|
||||
@ -26,6 +26,7 @@ const DatasetAccessRuleSection = ({
|
||||
<AccessRuleSection
|
||||
title="Knowledge Base Access Rules"
|
||||
rules={datasetAccessRules}
|
||||
isLoadingRules={isLoading}
|
||||
createButtonLabel="Create KB permission set"
|
||||
onCreate={onCreate}
|
||||
onEditRule={onEditRule}
|
||||
|
||||
@ -4,7 +4,7 @@ import type { PermissionSetFormValues, PermissionSetModalMode } from './permissi
|
||||
import type { AccessPolicyResourceType, AccessPolicyWithBindings } from '@/models/access-control'
|
||||
import { toast } from '@langgenius/dify-ui/toast'
|
||||
import { useCallback, useState } from 'react'
|
||||
import { useCreateAccessRule, useUpdateAppAccessRuleBindings, useUpdateDatasetAccessRuleBindings } from '@/service/access-control/use-workspace-access-rules'
|
||||
import { useCreateAccessRule, useUpdateAccessRule, useUpdateAppAccessRuleBindings, useUpdateDatasetAccessRuleBindings } from '@/service/access-control/use-workspace-access-rules'
|
||||
import AddRuleTargetsModal from './add-rule-targets-modal'
|
||||
import AppAccessRuleSection from './app-access-rule-section'
|
||||
import DatasetAccessRuleSection from './dataset-access-rule-section'
|
||||
@ -13,6 +13,7 @@ import PermissionSetModal from './permission-set-modal'
|
||||
type PermissionSetModalState = {
|
||||
mode: PermissionSetModalMode
|
||||
resourceType: AccessPolicyResourceType
|
||||
ruleId?: string
|
||||
initialValues?: PermissionSetFormValues
|
||||
}
|
||||
|
||||
@ -74,6 +75,7 @@ const AccessRulesPage = () => {
|
||||
setPermissionSetModalState({
|
||||
mode: 'edit',
|
||||
resourceType,
|
||||
ruleId: policy.id,
|
||||
initialValues: {
|
||||
name: policy.name,
|
||||
description: policy.description,
|
||||
@ -84,23 +86,43 @@ const AccessRulesPage = () => {
|
||||
[],
|
||||
)
|
||||
|
||||
const { mutateAsync: createAccessRule } = useCreateAccessRule(permissionSetModalState?.resourceType)
|
||||
const { mutateAsync: createAccessRule } = useCreateAccessRule()
|
||||
const { mutateAsync: updateAccessRule } = useUpdateAccessRule()
|
||||
|
||||
const handlePermissionSetSubmit = useCallback(
|
||||
(values: PermissionSetFormValues) => {
|
||||
const mode = permissionSetModalState?.mode || ''
|
||||
const id = permissionSetModalState?.ruleId || ''
|
||||
const { name, description, permissionKeys } = values
|
||||
createAccessRule({
|
||||
name,
|
||||
description,
|
||||
permission_keys: permissionKeys,
|
||||
}, {
|
||||
onSuccess: () => {
|
||||
toast.success('Access rule created successfully')
|
||||
closePermissionSetModal()
|
||||
},
|
||||
})
|
||||
if (mode === 'create') {
|
||||
createAccessRule({
|
||||
name,
|
||||
description,
|
||||
permission_keys: permissionKeys,
|
||||
resourceType: permissionSetModalState!.resourceType,
|
||||
}, {
|
||||
onSuccess: () => {
|
||||
toast.success('Access rule created successfully')
|
||||
closePermissionSetModal()
|
||||
},
|
||||
})
|
||||
}
|
||||
else if (mode === 'edit') {
|
||||
updateAccessRule({
|
||||
id: id!,
|
||||
name,
|
||||
description,
|
||||
permission_keys: permissionKeys,
|
||||
resourceType: permissionSetModalState!.resourceType,
|
||||
}, {
|
||||
onSuccess: () => {
|
||||
toast.success('Access rule updated successfully')
|
||||
closePermissionSetModal()
|
||||
},
|
||||
})
|
||||
}
|
||||
},
|
||||
[closePermissionSetModal, createAccessRule],
|
||||
[closePermissionSetModal, createAccessRule, updateAccessRule, permissionSetModalState],
|
||||
)
|
||||
|
||||
const createApp = useCallback(() => handleCreate('app'), [handleCreate])
|
||||
@ -131,8 +153,8 @@ const AccessRulesPage = () => {
|
||||
{addingRule && (
|
||||
<AddRuleTargetsModal
|
||||
ruleName={addingRule.policy.name}
|
||||
initialRoleIds={addingRule.role_ids.map(role => role.id)}
|
||||
initialMemberIds={addingRule.account_ids.map(account => account.id)}
|
||||
initialRoleIds={addingRule.roles.map(role => role.role_id)}
|
||||
initialMemberIds={addingRule.accounts.map(account => account.account_id)}
|
||||
onClose={closeAddModal}
|
||||
onSubmit={handleAddSubmit}
|
||||
/>
|
||||
|
||||
@ -15,6 +15,7 @@ import { useMemo, useState } from 'react'
|
||||
import { useTranslation } from 'react-i18next'
|
||||
import Checkbox from '@/app/components/base/checkbox'
|
||||
import Input from '@/app/components/base/input'
|
||||
import { useWorkspaceRoleList } from '@/service/access-control/use-workspace-roles'
|
||||
|
||||
export type AssignableRole = {
|
||||
id: string
|
||||
@ -23,38 +24,31 @@ export type AssignableRole = {
|
||||
}
|
||||
|
||||
export type AssignRolesModalProps = {
|
||||
open: boolean
|
||||
member: Member
|
||||
onClose: () => void
|
||||
onSubmit: (roleIds: string[]) => void
|
||||
}
|
||||
|
||||
type AssignRolesModalBodyProps = {
|
||||
roles: AssignableRole[]
|
||||
} & Omit<AssignRolesModalProps, 'open'>
|
||||
|
||||
// TODO: replace with roles fetched from the permissions API once available.
|
||||
const MOCK_ASSIGNABLE_ROLES: AssignableRole[] = [
|
||||
{ id: 'admin', name: 'Admin', description: 'Full access to workspace management and settings' },
|
||||
{ id: 'editor', name: 'Editor', description: 'Create and edit resources without settings access' },
|
||||
{ id: 'member', name: 'Member', description: 'Basic workspace access' },
|
||||
{ id: 'auditor', name: 'Auditor', description: 'View application logs and audit trails' },
|
||||
{ id: 'tester', name: 'Tester', description: 'Test applications in sandbox environments' },
|
||||
]
|
||||
type AssignRolesModalBodyProps = AssignRolesModalProps
|
||||
|
||||
const AssignRolesModalBody = ({
|
||||
roles,
|
||||
member,
|
||||
onClose,
|
||||
onSubmit,
|
||||
}: AssignRolesModalBodyProps) => {
|
||||
const { t } = useTranslation()
|
||||
const [selected, setSelected] = useState<string[]>(() => {
|
||||
const match = MOCK_ASSIGNABLE_ROLES.find(r => r.id === member.role)
|
||||
return match ? [match.id] : []
|
||||
return member.roles?.map(role => role.id) || []
|
||||
})
|
||||
const [keyword, setKeyword] = useState('')
|
||||
|
||||
const { data: rolesData, isLoading: rolesLoading } = useWorkspaceRoleList({
|
||||
page: 1,
|
||||
limit: 20,
|
||||
})
|
||||
|
||||
const roles = useMemo(() => rolesData?.data ?? [], [rolesData])
|
||||
|
||||
const filteredRoles = useMemo(() => {
|
||||
const trimmed = keyword.trim().toLowerCase()
|
||||
if (!trimmed)
|
||||
@ -116,58 +110,62 @@ const AssignRolesModalBody = ({
|
||||
className="mt-2 min-h-0 flex-1"
|
||||
slotClassNames={{ viewport: 'px-3 overscroll-contain' }}
|
||||
>
|
||||
{filteredRoles.length === 0
|
||||
{rolesLoading
|
||||
? (
|
||||
<div className="px-3 py-6 text-center system-sm-regular text-text-tertiary">
|
||||
{t('members.assignRolesModal.empty', {
|
||||
ns: 'common',
|
||||
defaultValue: 'No matching roles',
|
||||
})}
|
||||
Loading roles...
|
||||
</div>
|
||||
)
|
||||
: (
|
||||
<ul className="flex flex-col gap-0.5">
|
||||
{filteredRoles.map((role) => {
|
||||
const checked = selected.includes(role.id)
|
||||
const handleToggle = () => toggle(role.id)
|
||||
return (
|
||||
<li key={role.id}>
|
||||
<div
|
||||
role="checkbox"
|
||||
aria-checked={checked}
|
||||
tabIndex={0}
|
||||
className={cn(
|
||||
'flex cursor-pointer items-start gap-3 rounded-lg px-3 py-2.5 hover:bg-state-base-hover focus-visible:outline-2 focus-visible:-outline-offset-2 focus-visible:outline-components-input-border-active',
|
||||
checked && 'bg-state-accent-hover hover:bg-state-accent-hover',
|
||||
)}
|
||||
onClick={handleToggle}
|
||||
onKeyDown={(e) => {
|
||||
if (e.key === ' ' || e.key === 'Enter') {
|
||||
e.preventDefault()
|
||||
handleToggle()
|
||||
}
|
||||
}}
|
||||
>
|
||||
<Checkbox
|
||||
checked={checked}
|
||||
className="pointer-events-none mt-0.5"
|
||||
/>
|
||||
<div className="min-w-0 flex-1">
|
||||
<div className="system-sm-semibold text-text-secondary">
|
||||
{role.name}
|
||||
</div>
|
||||
{role.description && (
|
||||
<div className="mt-0.5 system-xs-regular text-text-tertiary">
|
||||
{role.description}
|
||||
</div>
|
||||
: filteredRoles.length === 0
|
||||
? (
|
||||
<div className="px-3 py-6 text-center system-sm-regular text-text-tertiary">
|
||||
{t('members.assignRolesModal.empty', {
|
||||
ns: 'common',
|
||||
defaultValue: 'No matching roles',
|
||||
})}
|
||||
</div>
|
||||
)
|
||||
: (
|
||||
<ul className="flex flex-col gap-0.5">
|
||||
{filteredRoles.map((role) => {
|
||||
const checked = selected.includes(role.id)
|
||||
const handleToggle = () => toggle(role.id)
|
||||
return (
|
||||
<li key={role.id}>
|
||||
<div
|
||||
role="checkbox"
|
||||
aria-checked={checked}
|
||||
tabIndex={0}
|
||||
className={cn(
|
||||
'flex cursor-pointer items-start gap-3 rounded-lg px-3 py-2.5 hover:bg-state-base-hover focus-visible:outline-2 focus-visible:-outline-offset-2 focus-visible:outline-components-input-border-active',
|
||||
checked && 'bg-state-accent-hover hover:bg-state-accent-hover',
|
||||
)}
|
||||
onClick={handleToggle}
|
||||
onKeyDown={(e) => {
|
||||
if (e.key === ' ' || e.key === 'Enter') {
|
||||
e.preventDefault()
|
||||
handleToggle()
|
||||
}
|
||||
}}
|
||||
>
|
||||
<Checkbox
|
||||
checked={checked}
|
||||
className="pointer-events-none mt-0.5"
|
||||
/>
|
||||
<div className="min-w-0 flex-1">
|
||||
<div className="system-sm-semibold text-text-secondary">
|
||||
{role.name}
|
||||
</div>
|
||||
<div className="mt-0.5 system-xs-regular text-text-tertiary">
|
||||
{role.description || 'No description'}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</li>
|
||||
)
|
||||
})}
|
||||
</ul>
|
||||
)}
|
||||
</li>
|
||||
)
|
||||
})}
|
||||
</ul>
|
||||
)}
|
||||
</ScrollArea>
|
||||
|
||||
<div className="flex shrink-0 items-center justify-between gap-3 border-t border-divider-subtle px-6 py-4">
|
||||
@ -175,7 +173,6 @@ const AssignRolesModalBody = ({
|
||||
{t('members.assignRolesModal.selectedCount', {
|
||||
ns: 'common',
|
||||
count: selected.length,
|
||||
defaultValue: '{{count}} selected',
|
||||
})}
|
||||
</div>
|
||||
<div className="flex items-center gap-2">
|
||||
@ -192,21 +189,19 @@ const AssignRolesModalBody = ({
|
||||
}
|
||||
|
||||
const AssignRolesModal = ({
|
||||
open,
|
||||
member,
|
||||
onClose,
|
||||
onSubmit,
|
||||
}: AssignRolesModalProps) => {
|
||||
return (
|
||||
<Dialog
|
||||
open={open}
|
||||
open
|
||||
onOpenChange={(nextOpen) => {
|
||||
if (!nextOpen)
|
||||
onClose()
|
||||
}}
|
||||
>
|
||||
<AssignRolesModalBody
|
||||
roles={MOCK_ASSIGNABLE_ROLES}
|
||||
member={member}
|
||||
onClose={onClose}
|
||||
onSubmit={onSubmit}
|
||||
|
||||
@ -24,13 +24,6 @@ import TransferOwnershipModal from './transfer-ownership-modal'
|
||||
|
||||
const MembersPage = () => {
|
||||
const { t } = useTranslation()
|
||||
const RoleMap = {
|
||||
owner: t('members.owner', { ns: 'common' }),
|
||||
admin: t('members.admin', { ns: 'common' }),
|
||||
editor: t('members.editor', { ns: 'common' }),
|
||||
dataset_operator: t('members.datasetOperator', { ns: 'common' }),
|
||||
normal: t('members.normal', { ns: 'common' }),
|
||||
}
|
||||
const locale = useLocale()
|
||||
|
||||
const { userProfile, currentWorkspace, isCurrentWorkspaceOwner, isCurrentWorkspaceManager } = useAppContext()
|
||||
@ -141,7 +134,7 @@ const MembersPage = () => {
|
||||
<MemberRow
|
||||
key={account.id}
|
||||
member={account}
|
||||
roleLabel={RoleMap[account.role] || RoleMap.normal}
|
||||
roles={account.roles}
|
||||
isCurrentUser={userProfile.email === account.email}
|
||||
canManage={isCurrentWorkspaceManager}
|
||||
operatorRole={currentWorkspace.role}
|
||||
@ -192,7 +185,6 @@ const MembersPage = () => {
|
||||
<MemberDetailsModal
|
||||
open={!!detailsMember}
|
||||
member={detailsMember}
|
||||
roleLabel={RoleMap[detailsMember.role] || RoleMap.normal}
|
||||
canAssignRoles={
|
||||
isCurrentWorkspaceManager
|
||||
&& detailsMember.role !== 'owner'
|
||||
|
||||
@ -9,15 +9,15 @@ import {
|
||||
DialogContent,
|
||||
DialogTitle,
|
||||
} from '@langgenius/dify-ui/dialog'
|
||||
import { memo, useState } from 'react'
|
||||
import { memo, useCallback, useState } from 'react'
|
||||
import { useTranslation } from 'react-i18next'
|
||||
import { useRolesOfMember } from '@/service/access-control/use-member-roles'
|
||||
import AssignRolesModal from '../assign-roles-modal'
|
||||
import PermissionRoleChip from './permission-role-chip'
|
||||
|
||||
export type MemberDetailsModalProps = {
|
||||
open: boolean
|
||||
member: Member
|
||||
roleLabel: string
|
||||
canAssignRoles?: boolean
|
||||
onClose: () => void
|
||||
onAssignSubmit?: (roleIds: string[]) => void
|
||||
@ -26,7 +26,6 @@ export type MemberDetailsModalProps = {
|
||||
const MemberDetailsModal = ({
|
||||
open,
|
||||
member,
|
||||
roleLabel,
|
||||
canAssignRoles = false,
|
||||
onClose,
|
||||
onAssignSubmit,
|
||||
@ -34,12 +33,21 @@ const MemberDetailsModal = ({
|
||||
const { t } = useTranslation()
|
||||
const [assignOpen, setAssignOpen] = useState(false)
|
||||
|
||||
const assignedRoles = [{ key: member.role, label: roleLabel }]
|
||||
const { data: rolesOfMember } = useRolesOfMember(member.id)
|
||||
|
||||
const handleAssignSubmit = (ids: string[]) => {
|
||||
const roles = rolesOfMember?.roles || []
|
||||
|
||||
const builtinRoles = roles.filter(role => role.is_builtin)
|
||||
const customRoles = roles.filter(role => !role.is_builtin)
|
||||
|
||||
const handleClose = useCallback(() => {
|
||||
setAssignOpen(false)
|
||||
}, [])
|
||||
|
||||
const handleAssignSubmit = useCallback((ids: string[]) => {
|
||||
onAssignSubmit?.(ids)
|
||||
setAssignOpen(false)
|
||||
}
|
||||
}, [onAssignSubmit])
|
||||
|
||||
return (
|
||||
<>
|
||||
@ -87,7 +95,7 @@ const MemberDetailsModal = ({
|
||||
})}
|
||||
</span>
|
||||
<span className="system-xs-medium text-text-tertiary">
|
||||
{assignedRoles.length}
|
||||
{roles.length}
|
||||
</span>
|
||||
</div>
|
||||
{canAssignRoles && (
|
||||
@ -108,33 +116,50 @@ const MemberDetailsModal = ({
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="mt-4">
|
||||
<div className="mb-2 system-2xs-medium-uppercase text-text-tertiary">
|
||||
{t('members.memberDetails.generalGroup', {
|
||||
ns: 'common',
|
||||
defaultValue: 'GENERAL',
|
||||
})}
|
||||
{builtinRoles.length > 0 && (
|
||||
<div className="mt-4">
|
||||
<div className="mb-2 system-2xs-medium-uppercase text-text-tertiary">
|
||||
{t('members.memberDetails.generalGroup', {
|
||||
ns: 'common',
|
||||
})}
|
||||
</div>
|
||||
<div className="flex flex-wrap gap-1.5">
|
||||
{builtinRoles.map(role => (
|
||||
<PermissionRoleChip
|
||||
key={role.id}
|
||||
roleKey={role.id}
|
||||
label={role.name}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex flex-wrap gap-1.5">
|
||||
{assignedRoles.map(role => (
|
||||
<PermissionRoleChip
|
||||
key={role.key}
|
||||
roleKey={role.key}
|
||||
label={role.label}
|
||||
highlighted
|
||||
/>
|
||||
))}
|
||||
)}
|
||||
{customRoles.length > 0 && (
|
||||
<div className="mt-4">
|
||||
<div className="mb-2 system-2xs-medium-uppercase text-text-tertiary">
|
||||
{t('members.memberDetails.customGroup', {
|
||||
ns: 'common',
|
||||
})}
|
||||
</div>
|
||||
<div className="flex flex-wrap gap-1.5">
|
||||
{customRoles.map(role => (
|
||||
<PermissionRoleChip
|
||||
key={role.id}
|
||||
roleKey={role.id}
|
||||
label={role.name}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
|
||||
{assignOpen && (
|
||||
<AssignRolesModal
|
||||
open={assignOpen}
|
||||
member={member}
|
||||
onClose={() => setAssignOpen(false)}
|
||||
onClose={handleClose}
|
||||
onSubmit={handleAssignSubmit}
|
||||
/>
|
||||
)}
|
||||
|
||||
@ -13,7 +13,6 @@ import { getRolePermissionKeys } from './role-permissions'
|
||||
export type PermissionRoleChipProps = {
|
||||
roleKey: string
|
||||
label: string
|
||||
highlighted?: boolean
|
||||
onRemove?: () => void
|
||||
className?: string
|
||||
}
|
||||
@ -21,7 +20,6 @@ export type PermissionRoleChipProps = {
|
||||
const PermissionRoleChip = ({
|
||||
roleKey,
|
||||
label,
|
||||
highlighted = false,
|
||||
onRemove,
|
||||
className,
|
||||
}: PermissionRoleChipProps) => {
|
||||
@ -32,10 +30,8 @@ const PermissionRoleChip = ({
|
||||
const chip = (
|
||||
<span
|
||||
className={cn(
|
||||
'inline-flex h-6 max-w-full cursor-default items-center gap-1 rounded-md px-1.5 system-xs-medium shadow-xs',
|
||||
highlighted
|
||||
? 'bg-state-accent-hover text-text-accent'
|
||||
: 'bg-background-body text-text-secondary',
|
||||
'group inline-flex h-6 max-w-full cursor-default items-center gap-1 rounded-md px-1.5 system-xs-medium shadow-xs',
|
||||
'bg-background-body text-text-secondary group-hover:bg-state-accent-hover group-hover:text-text-accent',
|
||||
className,
|
||||
)}
|
||||
data-testid="permission-role-chip"
|
||||
@ -56,7 +52,7 @@ const PermissionRoleChip = ({
|
||||
}}
|
||||
className={cn(
|
||||
'flex h-4 w-4 items-center justify-center rounded hover:bg-black/5',
|
||||
highlighted ? 'text-text-accent' : 'text-text-tertiary',
|
||||
'text-text-tertiary',
|
||||
)}
|
||||
>
|
||||
<span aria-hidden className="i-ri-close-line h-3 w-3" />
|
||||
|
||||
@ -10,7 +10,10 @@ import RoleBadges from './role-badges'
|
||||
|
||||
type MemberRowProps = {
|
||||
member: Member
|
||||
roleLabel: string
|
||||
roles: Array<{
|
||||
id: string
|
||||
name: string
|
||||
}>
|
||||
isCurrentUser: boolean
|
||||
canManage: boolean
|
||||
operatorRole: string
|
||||
@ -22,7 +25,7 @@ type MemberRowProps = {
|
||||
|
||||
const MemberRow = ({
|
||||
member,
|
||||
roleLabel,
|
||||
roles,
|
||||
isCurrentUser,
|
||||
canManage,
|
||||
operatorRole,
|
||||
@ -34,6 +37,8 @@ const MemberRow = ({
|
||||
const { t } = useTranslation()
|
||||
const { formatTimeFromNow } = useFormatTimeFromNow()
|
||||
|
||||
const roleNames = roles.map(role => role.name)
|
||||
|
||||
const openDetails = useCallback(() => {
|
||||
onOpenDetails(member)
|
||||
}, [member, onOpenDetails])
|
||||
@ -98,7 +103,7 @@ const MemberRow = ({
|
||||
>
|
||||
<RoleBadges
|
||||
className="grow"
|
||||
roles={[roleLabel]}
|
||||
roleNames={roleNames}
|
||||
/>
|
||||
{canManage && (
|
||||
<MemberMenu
|
||||
|
||||
@ -22,17 +22,17 @@ const RoleBadge = ({ label, className }: RoleBadgeProps) => {
|
||||
}
|
||||
|
||||
export type RoleBadgesProps = {
|
||||
roles: string[]
|
||||
roleNames: string[]
|
||||
max?: number
|
||||
className?: string
|
||||
}
|
||||
|
||||
const RoleBadges = ({ roles, max = 2, className }: RoleBadgesProps) => {
|
||||
if (!roles.length)
|
||||
const RoleBadges = ({ roleNames, max = 2, className }: RoleBadgesProps) => {
|
||||
if (!roleNames.length)
|
||||
return null
|
||||
|
||||
const visible = roles.slice(0, max)
|
||||
const overflow = roles.slice(max)
|
||||
const visible = roleNames.slice(0, max)
|
||||
const overflow = roleNames.slice(max)
|
||||
|
||||
return (
|
||||
<div className={cn('flex min-w-0 flex-wrap items-center gap-1', className)}>
|
||||
|
||||
@ -246,6 +246,7 @@
|
||||
"members.memberActions": "Member actions",
|
||||
"members.memberDetails.assign": "Assign",
|
||||
"members.memberDetails.assignedRoles": "Assigned Roles",
|
||||
"members.memberDetails.customGroup": "CUSTOMIZED",
|
||||
"members.memberDetails.generalGroup": "GENERAL",
|
||||
"members.memberDetails.openAria": "Open member details for {{name}}",
|
||||
"members.memberDetails.permissions.assignRoles": "Assign roles",
|
||||
|
||||
@ -246,7 +246,8 @@
|
||||
"members.memberActions": "成员操作",
|
||||
"members.memberDetails.assign": "分配",
|
||||
"members.memberDetails.assignedRoles": "已分配角色",
|
||||
"members.memberDetails.generalGroup": "通用角色",
|
||||
"members.memberDetails.customGroup": "自定义",
|
||||
"members.memberDetails.generalGroup": "通用",
|
||||
"members.memberDetails.openAria": "打开 {{name}} 的成员详情",
|
||||
"members.memberDetails.permissions.assignRoles": "分配角色",
|
||||
"members.memberDetails.permissions.createApps": "创建应用",
|
||||
|
||||
@ -131,13 +131,13 @@ export type UpdateAccessPolicyRequest = {
|
||||
export type BindingType = 'role' | 'account'
|
||||
|
||||
export type Bindings = {
|
||||
role_ids: Array<{
|
||||
id: string
|
||||
name: string
|
||||
roles: Array<{
|
||||
role_id: string
|
||||
role_name: string
|
||||
}>
|
||||
account_ids: Array<{
|
||||
id: string
|
||||
name: string
|
||||
accounts: Array<{
|
||||
account_id: string
|
||||
account_name: string
|
||||
}>
|
||||
}
|
||||
|
||||
@ -170,12 +170,12 @@ export type GetDatasetAccessPoliciesResponse = {
|
||||
pagination: Pagination
|
||||
}
|
||||
|
||||
export type RolesOfMemberResponse = {
|
||||
account_id: string
|
||||
roles: Role[]
|
||||
}
|
||||
|
||||
export type UpdateRolesOfMemberRequest = {
|
||||
member_id: string
|
||||
role_ids: string[]
|
||||
}
|
||||
|
||||
export type UpdateRolesOfMemberResponse = {
|
||||
account_id: string
|
||||
roles: Role[]
|
||||
}
|
||||
|
||||
@ -54,6 +54,10 @@ export type Member = Pick<UserProfileResponse, 'id' | 'name' | 'email' | 'last_l
|
||||
avatar: string
|
||||
status: 'pending' | 'active' | 'banned' | 'closed'
|
||||
role: 'owner' | 'admin' | 'editor' | 'normal' | 'dataset_operator'
|
||||
roles: Array<{
|
||||
id: string
|
||||
name: string
|
||||
}>
|
||||
}
|
||||
|
||||
enum ProviderName {
|
||||
|
||||
27
web/service/access-control/use-member-roles.ts
Normal file
27
web/service/access-control/use-member-roles.ts
Normal file
@ -0,0 +1,27 @@
|
||||
import type { RolesOfMemberResponse } from '@/models/access-control'
|
||||
import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query'
|
||||
import { get, put } from '../base'
|
||||
|
||||
const NAME_SPACE = 'rbac-member-roles'
|
||||
|
||||
export const useRolesOfMember = (memberId: string) => {
|
||||
return useQuery({
|
||||
queryKey: [NAME_SPACE, 'member-roles', memberId],
|
||||
queryFn: () => get<RolesOfMemberResponse>(`/workspaces/current/rbac/members/${memberId}/rbac-roles`),
|
||||
})
|
||||
}
|
||||
|
||||
export const useUpdateRolesOfMember = () => {
|
||||
const queryClient = useQueryClient()
|
||||
|
||||
return useMutation({
|
||||
mutationKey: [NAME_SPACE, 'update-member-roles'],
|
||||
mutationFn: ({ memberId, roleIds }: { memberId: string, roleIds: string[] }) =>
|
||||
put(`/workspaces/current/rbac/members/${memberId}/rbac-roles`, {
|
||||
body: { role_ids: roleIds },
|
||||
}),
|
||||
onSuccess: (_, { memberId }) => {
|
||||
queryClient.invalidateQueries({ queryKey: [NAME_SPACE, 'member-roles', memberId] })
|
||||
},
|
||||
})
|
||||
}
|
||||
11
web/service/access-control/use-permission-keys.ts
Normal file
11
web/service/access-control/use-permission-keys.ts
Normal file
@ -0,0 +1,11 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { get } from '../base'
|
||||
|
||||
const NAME_SPACE = 'workspace-permission-keys'
|
||||
|
||||
export const useWorkspacePermissionKeys = () => {
|
||||
return useQuery({
|
||||
queryKey: [NAME_SPACE],
|
||||
queryFn: () => get('/workspaces/current/rbac/my-permissions'),
|
||||
})
|
||||
}
|
||||
@ -29,13 +29,13 @@ export const useWorkspaceDatasetAccessRules = (params?: PaginationParameters) =>
|
||||
})
|
||||
}
|
||||
|
||||
export const useCreateAccessRule = (resourceType?: AccessPolicyResourceType) => {
|
||||
export const useCreateAccessRule = () => {
|
||||
const queryClient = useQueryClient()
|
||||
|
||||
return useMutation({
|
||||
mutationKey: [NAME_SPACE, 'create', resourceType],
|
||||
mutationFn: (data: CreateAccessPolicyRequest) => {
|
||||
const { name, description, permission_keys } = data
|
||||
mutationKey: [NAME_SPACE, 'create'],
|
||||
mutationFn: (data: CreateAccessPolicyRequest & { resourceType: AccessPolicyResourceType }) => {
|
||||
const { name, description, permission_keys, resourceType } = data
|
||||
return post<AccessPolicy>('/workspaces/current/rbac/access-policies', {
|
||||
body: {
|
||||
resource_type: resourceType,
|
||||
@ -45,20 +45,18 @@ export const useCreateAccessRule = (resourceType?: AccessPolicyResourceType) =>
|
||||
},
|
||||
})
|
||||
},
|
||||
onSuccess: () => {
|
||||
if (resourceType) {
|
||||
queryClient.invalidateQueries({ queryKey: [NAME_SPACE, resourceType] })
|
||||
}
|
||||
onSuccess: (_, { resourceType }) => {
|
||||
queryClient.invalidateQueries({ queryKey: [NAME_SPACE, resourceType] })
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
export const useUpdateAccessRule = (resourceType: AccessPolicyResourceType) => {
|
||||
export const useUpdateAccessRule = () => {
|
||||
const queryClient = useQueryClient()
|
||||
|
||||
return useMutation({
|
||||
mutationKey: [NAME_SPACE, 'update', resourceType],
|
||||
mutationFn: (data: UpdateAccessPolicyRequest) => {
|
||||
mutationKey: [NAME_SPACE, 'update'],
|
||||
mutationFn: (data: UpdateAccessPolicyRequest & { resourceType: AccessPolicyResourceType }) => {
|
||||
const { id, name, description, permission_keys } = data
|
||||
return put<AccessPolicy>(`/workspaces/current/rbac/access-policies/${id}`, {
|
||||
body: {
|
||||
@ -69,7 +67,7 @@ export const useUpdateAccessRule = (resourceType: AccessPolicyResourceType) => {
|
||||
},
|
||||
})
|
||||
},
|
||||
onSuccess: () => {
|
||||
onSuccess: (_, { resourceType }) => {
|
||||
queryClient.invalidateQueries({ queryKey: [NAME_SPACE, resourceType] })
|
||||
},
|
||||
})
|
||||
|
||||
Loading…
Reference in New Issue
Block a user