Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion app/forms/access-util.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ export type EditRoleModalProps = AddRoleModalProps & {
name?: string
identityId: string
identityType: IdentityType
defaultValues: { roleName: RoleKey }
defaultValues: { roleName?: RoleKey }
}

const AccessDocs = () => (
Expand Down
1 change: 1 addition & 0 deletions app/forms/project-access.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,7 @@ export function ProjectAccessEditUserSideModal({
</ResourceLabel>
}
onSubmit={({ roleName }) => {
if (!roleName) return
updatePolicy.mutate({
path: { project },
body: updateRole({ identityId, identityType, roleName }, policy),
Expand Down
1 change: 1 addition & 0 deletions app/forms/silo-access.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,7 @@ export function SiloAccessEditUserSideModal({
</ResourceLabel>
}
onSubmit={({ roleName }) => {
if (!roleName) return
updatePolicy.mutate({
body: updateRole({ identityId, identityType, roleName }, policy),
})
Expand Down
220 changes: 220 additions & 0 deletions app/pages/SiloAccessGroupsTab.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,220 @@
/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
*
* Copyright Oxide Computer Company
*/
import { useQuery } from '@tanstack/react-query'
import { createColumnHelper } from '@tanstack/react-table'
import { useCallback, useMemo, useState } from 'react'

import {
api,
deleteRole,
getListQFn,
q,
queryClient,
useApiMutation,
usePrefetchedQuery,
type Group,
type User,
} from '@oxide/api'
import { PersonGroup16Icon, PersonGroup24Icon } from '@oxide/design-system/icons/react'
import { Badge } from '@oxide/design-system/ui'

import { ReadOnlySideModalForm } from '~/components/form/ReadOnlySideModalForm'
import { HL } from '~/components/HL'
import { SiloAccessEditUserSideModal } from '~/forms/silo-access'
import { titleCrumb } from '~/hooks/use-crumbs'
import { confirmDelete } from '~/stores/confirm-delete'
import { EmptyCell } from '~/table/cells/EmptyCell'
import { ButtonCell } from '~/table/cells/LinkCell'
import { useColsWithActions, type MenuAction } from '~/table/columns/action-col'
import { useQueryTable } from '~/table/QueryTable'
import { EmptyMessage } from '~/ui/lib/EmptyMessage'
import { PropertiesTable } from '~/ui/lib/PropertiesTable'
import { ResourceLabel } from '~/ui/lib/SideModal'
import { Table } from '~/ui/lib/Table'
import { roleColor } from '~/util/access'
import { ALL_ISH } from '~/util/consts'

const policyView = q(api.policyView, {})
const groupList = getListQFn(api.groupList, {})

export async function clientLoader() {
await Promise.all([
queryClient.prefetchQuery(policyView),
queryClient.prefetchQuery(groupList.optionsFn()),
])
return null
}

export const handle = titleCrumb('Groups')

const colHelper = createColumnHelper<Group>()

function MemberCountCell({ groupId }: { groupId: string }) {
const { data } = useQuery(q(api.userList, { query: { group: groupId, limit: ALL_ISH } }))
return data ? <>{data.items.length}</> : null
}

const GroupEmptyState = () => (
<EmptyMessage
icon={<PersonGroup24Icon />}
title="No groups"
body="No groups have been added to this silo"
/>
)

type GroupMembersSideModalProps = {
group: Group
onDismiss: () => void
}

function GroupMembersSideModal({ group, onDismiss }: GroupMembersSideModalProps) {
const { data } = useQuery(q(api.userList, { query: { group: group.id, limit: ALL_ISH } }))
const members = data?.items ?? []

return (
<ReadOnlySideModalForm
title="Group members"
subtitle={
<ResourceLabel>
<PersonGroup16Icon /> {group.displayName}
</ResourceLabel>
}
onDismiss={onDismiss}
animate
>
<PropertiesTable>
<PropertiesTable.IdRow id={group.id} />
</PropertiesTable>
<div className="mt-6">
{members.length === 0 ? (
<EmptyMessage
icon={<PersonGroup24Icon />}
title="No members"
body="This group has no members"
/>
) : (
<table className="ox-table text-sans-md w-full border-separate">
<Table.Header>
<Table.HeaderRow>
<Table.HeadCell>Name</Table.HeadCell>
</Table.HeaderRow>
</Table.Header>
<Table.Body>
{members.map((member: User) => (
<Table.Row key={member.id}>
<Table.Cell>{member.displayName}</Table.Cell>
</Table.Row>
))}
</Table.Body>
</table>
)}
</div>
</ReadOnlySideModalForm>
)
}

export default function SiloAccessGroupsTab() {
const [selectedGroup, setSelectedGroup] = useState<Group | null>(null)
const [editingGroup, setEditingGroup] = useState<Group | null>(null)

const { data: siloPolicy } = usePrefetchedQuery(policyView)

const { mutateAsync: updatePolicy } = useApiMutation(api.policyUpdate, {
onSuccess: () => queryClient.invalidateEndpoint('policyView'),
})

const siloRoleById = useMemo(
() => new Map(siloPolicy.roleAssignments.map((a) => [a.identityId, a.roleName])),
[siloPolicy]
)

const siloRoleCol = useMemo(
() =>
colHelper.display({
id: 'siloRole',
header: 'Silo Role',
cell: ({ row }) => {
const role = siloRoleById.get(row.original.id)
return role ? <Badge color={roleColor[role]}>silo.{role}</Badge> : <EmptyCell />
},
}),
[siloRoleById]
)

const staticColumns = useMemo(
() => [
colHelper.accessor('displayName', {
header: 'Name',
cell: (info) => (
<ButtonCell onClick={() => setSelectedGroup(info.row.original)}>
{info.getValue()}
</ButtonCell>
),
}),
siloRoleCol,
colHelper.display({
id: 'memberCount',
header: 'Users',
cell: ({ row }) => <MemberCountCell groupId={row.original.id} />,
}),
],
[siloRoleCol]
)

const makeActions = useCallback(
(group: Group): MenuAction[] => {
const role = siloRoleById.get(group.id)
return [
{ label: 'Change role', onActivate: () => setEditingGroup(group) },
{
label: 'Remove role',
onActivate: confirmDelete({
doDelete: () => updatePolicy({ body: deleteRole(group.id, siloPolicy) }),
label: (
<span>
the <HL>{role}</HL> role for <HL>{group.displayName}</HL>
</span>
),
}),
disabled: !role && 'This group has no role to remove',
},
]
},
[siloRoleById, siloPolicy, updatePolicy]
)

const columns = useColsWithActions(staticColumns, makeActions)

const { table } = useQueryTable({
query: groupList,
columns,
emptyState: <GroupEmptyState />,
})

return (
<>
{table}
{selectedGroup && (
<GroupMembersSideModal
group={selectedGroup}
onDismiss={() => setSelectedGroup(null)}
/>
)}
{editingGroup && (
<SiloAccessEditUserSideModal
name={editingGroup.displayName}
identityId={editingGroup.id}
identityType="silo_group"
policy={siloPolicy}
defaultValues={{ roleName: siloRoleById.get(editingGroup.id) }}
onDismiss={() => setEditingGroup(null)}
/>
)}
</>
)
}
Loading
Loading