diff --git a/plugins/parodos/src/api/fetchAccessRequests.ts b/plugins/parodos/src/api/fetchAccessRequests.ts new file mode 100644 index 0000000..fdc2f03 --- /dev/null +++ b/plugins/parodos/src/api/fetchAccessRequests.ts @@ -0,0 +1,26 @@ +import { FetchApi } from '@backstage/core-plugin-api'; +import { AccessRequest, accessRequests } from '../models/project'; +import * as urls from '../urls'; + +export async function fetchAccessRequests( + fetch: FetchApi['fetch'], + baseUrl: string, + filter?: { projectId?: string; username?: string }, +): Promise { + const response = await fetch(`${baseUrl}${urls.Projects}/access/pending`); + const data = await response.json(); + + if ('error' in data) { + throw new Error(`${data.error}: ${data.path}`); + } + + const requests = accessRequests.parse(data); + + return filter + ? requests.filter( + request => + (filter.projectId ? request.projectId === filter.projectId : true) && + (filter.username ? request.username === filter.username : true), + ) + : requests; +} diff --git a/plugins/parodos/src/api/fetchProjectMembers.ts b/plugins/parodos/src/api/fetchProjectMembers.ts new file mode 100644 index 0000000..2df14e3 --- /dev/null +++ b/plugins/parodos/src/api/fetchProjectMembers.ts @@ -0,0 +1,20 @@ +import { FetchApi } from '@backstage/core-plugin-api'; +import { projectMembers } from '../models/project'; +import * as urls from '../urls'; + +export async function fetchProjectMembers( + fetch: FetchApi['fetch'], + baseUrl: string, + projectId: string, +) { + const response = await fetch( + `${baseUrl}${urls.Projects}/${projectId}/members`, + ); + const data = await response.json(); + + if ('error' in data) { + throw new Error(`${data.error}: ${data.path}`); + } + + return projectMembers.parse(data); +} diff --git a/plugins/parodos/src/api/removeUserFromProject.ts b/plugins/parodos/src/api/removeUserFromProject.ts new file mode 100644 index 0000000..5e1c0c7 --- /dev/null +++ b/plugins/parodos/src/api/removeUserFromProject.ts @@ -0,0 +1,22 @@ +import { FetchApi } from '@backstage/core-plugin-api'; +import * as urls from '../urls'; + +export async function removeUserFromProject( + fetch: FetchApi['fetch'], + baseUrl: string, + projectId: string, + users: string[], +) { + const response = await fetch( + `${baseUrl}${urls.Projects}/${projectId}/users`, + { + method: 'DELETE', + body: JSON.stringify(users), + }, + ); + const data = await response.json(); + + if ('error' in data) { + throw new Error(`${data.error}: ${data.path}`); + } +} diff --git a/plugins/parodos/src/api/responseOnAccessRequest.ts b/plugins/parodos/src/api/responseOnAccessRequest.ts new file mode 100644 index 0000000..1a2a47a --- /dev/null +++ b/plugins/parodos/src/api/responseOnAccessRequest.ts @@ -0,0 +1,26 @@ +import { FetchApi } from '@backstage/core-plugin-api'; +import { AccessStatus } from '../models/project'; +import * as urls from '../urls'; + +export async function responseOnAccessRequest( + fetch: FetchApi['fetch'], + baseUrl: string, + requestId: string, + payload: { comment?: string; status: AccessStatus }, +) { + const response = await fetch( + `${baseUrl}${urls.Projects}/access/${requestId}/status`, + { + method: 'POST', + body: JSON.stringify({ ...payload, comment: payload.comment || '' }), + }, + ); + if (response.status !== 204) { + if (response.status === 400) { + const data = await response.json(); + throw new Error(`${response.status}: ${data.message}`); + } + + throw new Error(`${response.status}`); + } +} diff --git a/plugins/parodos/src/api/updateUserRole.ts b/plugins/parodos/src/api/updateUserRole.ts new file mode 100644 index 0000000..387c1f9 --- /dev/null +++ b/plugins/parodos/src/api/updateUserRole.ts @@ -0,0 +1,23 @@ +import { FetchApi } from '@backstage/core-plugin-api'; +import { AccessRole } from '../models/project'; +import * as urls from '../urls'; + +export async function updateUserRole( + fetch: FetchApi['fetch'], + baseUrl: string, + projectId: string, + payload: { username: string; roles: AccessRole[] }[], +) { + const response = await fetch( + `${baseUrl}${urls.Projects}/${projectId}/users`, + { + method: 'POST', + body: JSON.stringify(payload), + }, + ); + const data = await response.json(); + + if ('error' in data) { + throw new Error(`${data.error}: ${data.path}`); + } +} diff --git a/plugins/parodos/src/components/projects/ProjectAccessTable.tsx b/plugins/parodos/src/components/projects/ProjectAccessTable.tsx index b3ddfc2..2fbf8bd 100644 --- a/plugins/parodos/src/components/projects/ProjectAccessTable.tsx +++ b/plugins/parodos/src/components/projects/ProjectAccessTable.tsx @@ -1,4 +1,10 @@ -import { Table, TableColumn } from '@backstage/core-components'; +import { Progress, Table, TableColumn } from '@backstage/core-components'; +import { + errorApiRef, + fetchApiRef, + identityApiRef, + useApi, +} from '@backstage/core-plugin-api'; import { BackstageTheme } from '@backstage/theme'; import { Button, @@ -10,70 +16,53 @@ import { Snackbar, SnackbarContent, TableCell, + TextField, + Typography, } from '@material-ui/core'; import CloseIcon from '@material-ui/icons/Close'; import classNames from 'classnames'; -import React, { useCallback, useState } from 'react'; -import { AccessRole, Project } from '../../models/project'; +import React, { SyntheticEvent, useCallback, useEffect, useState } from 'react'; +import useAsyncFn from 'react-use/lib/useAsyncFn'; +import { + AccessRequest, + AccessRole, + AccessStatus, + Project, + ProjectMember, +} from '../../models/project'; +import { useStore } from '../../stores/workflowStore/workflowStore'; +import { fetchProjectMembers } from '../../api/fetchProjectMembers'; +import { updateUserRole } from '../../api/updateUserRole'; +import { removeUserFromProject } from '../../api/removeUserFromProject'; +import { fetchAccessRequests } from '../../api/fetchAccessRequests'; +import { responseOnAccessRequest } from '../../api/responseOnAccessRequest'; +import { useNavigate } from 'react-router-dom'; +import { pluginRoutePrefix } from '../ParodosPage/navigationMap'; export interface ProjectAccessTableProps { project: Project; } interface AccessTableData { + requestId?: string; + username: string; member: string; - roles: [AccessRole, boolean][]; + roles: { name: string; disabled: boolean; selected: boolean }[]; + disabled: boolean; + selected: boolean; } -const roles: AccessRole[] = ['Owner', 'Admin', 'Developer']; - -const mockMembers: { name: string; role: AccessRole }[] = [ - { name: 'Luke Shannon', role: 'Owner' }, - { name: 'Piotr Kliczewski', role: 'Admin' }, - { name: 'Richard Wang', role: 'Developer' }, - { name: 'Moti Asayag', role: 'Developer' }, - { name: 'Paul Cowan', role: 'Developer' }, - { name: 'Dmitriy Lazarev', role: 'Developer' }, -]; - -function useMockMembers() { - const [members, setMembers] = useState(mockMembers); - - const addMember = useCallback( - (name: string, role: AccessRole) => - setMembers(prevMembers => [...prevMembers, { name, role }]), - [], - ); - const removeMember = useCallback( - (name: string) => - setMembers(prevMembers => - prevMembers.filter(member => member.name !== name), - ), - [], - ); - const transferOwnership = useCallback( - (name: string) => - setMembers(prevMembers => - prevMembers.map(member => { - if (member.name === name) return { ...member, role: 'Owner' }; - if (member.role === 'Owner') return { ...member, role: 'Developer' }; - return member; - }), - ), - [], - ); - const changeRole = useCallback( - (name: string, role: Exclude) => - setMembers(prevMembers => - prevMembers.map(member => - member.name === name ? { ...member, role } : member, - ), - ), - [], - ); - - return { members, addMember, removeMember, transferOwnership, changeRole }; -} +const accessRoles: AccessRole[] = ['OWNER', 'ADMIN', 'DEVELOPER']; +const accessRoleMap: Record = { + OWNER: 'Owner', + ADMIN: 'Admin', + DEVELOPER: 'Developer', +} as const; +const accessRoleBackMap: Record = { + Owner: 'OWNER', + Admin: 'ADMIN', + Developer: 'DEVELOPER', +}; const useStyles = makeStyles(theme => ({ root: { @@ -105,21 +94,49 @@ const useStyles = makeStyles(theme => ({ success: { backgroundColor: theme.palette.status.ok, }, + roleRequestMessage: { + display: 'flex', + }, + commentTextfield: { + width: '100%', + }, + responseButtons: { + flexShrink: 0, + }, })); export function ProjectAccessTable({ project, }: ProjectAccessTableProps): JSX.Element { - // TODO Use real data when it will be available - const { members, addMember, removeMember, transferOwnership, changeRole } = - useMockMembers(); - + const { fetch } = useApi(fetchApiRef); + const errorApi = useApi(errorApiRef); + const identityApi = useApi(identityApiRef); + const navigate = useNavigate(); + const baseUrl = useStore(state => state.baseUrl); + const fetchProjects = useStore(state => state.fetchProjects); const classes = useStyles(); const [snackbarMessage, setSnackbarMessage] = useState(); const [undoRemove, setUndoRemove] = useState<() => void>(); const [selectedMembers, setSelectedMembers] = useState< - { name: string; role: AccessRole }[] + { username: string; roles: AccessRole[] }[] >([]); + const [members, setMembers] = useState([]); + const [requests, setRequests] = useState([]); + const [currentUser, setCurrentUser] = useState(); + + const [{ error: getMembersError, loading: loadingMembers }, getMembers] = + useAsyncFn( + () => fetchProjectMembers(fetch, baseUrl, project.id), + [fetch, baseUrl, project.id], + ); + + const [ + { error: getAccessRequestsError, loading: loadingAccessRequests }, + getRequests, + ] = useAsyncFn( + () => fetchAccessRequests(fetch, baseUrl, { projectId: project.id }), + [fetch, baseUrl, project.id], + ); const columns: TableColumn[] = [ { @@ -127,13 +144,144 @@ export function ProjectAccessTable({ field: 'member', width: '15%', }, - { title: 'ROLE', field: 'roles' }, + { title: 'ROLES', field: 'roles' }, ]; - const tableData: AccessTableData[] = members.map(member => ({ - member: member.name, - roles: roles.map(role => [role, role === member.role]), - })); + const currentMember = members.find( + ({ username }) => username === currentUser, + ); + const owners = members.filter(({ roles }) => roles.includes('OWNER')); + const selectedOwners = selectedMembers.filter(({ roles }) => + roles.includes('OWNER'), + ); + const isOwner = currentMember?.roles.includes('OWNER') ?? false; + const isAdmin = currentMember?.roles.includes('ADMIN') ?? false; + + const tableData: AccessTableData[] = [...requests, ...members].map(member => { + const selected = selectedMembers.some( + ({ username }) => username === member.username, + ); + const roles = 'role' in member ? [member.role] : member.roles; + return { + requestId: + 'accessRequestId' in member ? member.accessRequestId : undefined, + username: member.username, + member: + 'accessRequestId' in member + ? `${member.firstname} ${member.lastname}` + : `${member.firstName} ${member.lastName}`, + roles: accessRoles.map(role => { + const selectedRole = roles.includes(role); + return { + name: accessRoleMap[role], + selected: selectedRole, + disabled: + (!isOwner && !isAdmin) || + (selectedRole && roles.length <= 1) || + (!isOwner && role === 'OWNER') || + (selectedRole && role === 'OWNER' && owners.length <= 1), + }; + }), + selected, + disabled: + !isOwner || + (!selected && + owners.length - selectedOwners.length <= 1 && + roles.includes('OWNER')), + }; + }); + + useEffect(() => { + if (owners.length === selectedOwners.length) { + setSelectedMembers(prevSelected => + prevSelected.filter(({ roles }) => !roles.includes('OWNER')), + ); + } + }, [owners, selectedOwners]); + + const [{ error: changeRoleError }, changeRole] = useAsyncFn( + async (username: string, role: AccessRole) => { + const roles = members.find(member => member.username === username)!.roles; + if (roles.includes(role)) { + roles.splice(roles.indexOf(role), 1); + } else { + roles.push(role as AccessRole); + } + await updateUserRole(fetch, baseUrl, project.id, [{ username, roles }]); + setMembers(await getMembers()); + setSnackbarMessage( + `User role${ + roles.length > 1 ? 's' : '' + } for ${username} has been successfully changed to ${roles.join( + ', ', + )}.`, + ); + }, + [fetch, baseUrl, project.id, members, getMembers], + ); + + const [{ error: removeMembersError }, removeMembers] = + useAsyncFn(async () => { + const users = selectedMembers.map(({ username }) => username); + await removeUserFromProject(fetch, baseUrl, project.id, users); + setMembers(await getMembers()); + setSelectedMembers([]); + setSnackbarMessage( + `You have successfully removed ${selectedMembers.length} contributor${ + selectedMembers.length > 1 ? 's' : '' + } from this project.`, + ); + if (selectedMembers.some(({ username }) => username === currentUser)) { + fetchProjects(fetch, true); + navigate(`${pluginRoutePrefix}/projects`); + } else { + setUndoRemove(() => async () => { + await updateUserRole(fetch, baseUrl, project.id, selectedMembers); + setMembers(await getMembers()); + }); + } + }, [ + selectedMembers, + fetch, + baseUrl, + project.id, + getMembers, + currentUser, + fetchProjects, + navigate, + ]); + + const [{ error: responseRequestError }, responseRequest] = useAsyncFn( + async (requestId: string, status: AccessStatus, comment?: string) => { + await responseOnAccessRequest(fetch, baseUrl, requestId, { + comment, + status, + }); + setMembers(await getMembers()); + setRequests(await getRequests()); + setSnackbarMessage( + `You have successfully ${ + status === 'APPROVED' ? 'approved' : 'rejected' + } the request.`, + ); + }, + [fetch, baseUrl, getMembers, getRequests], + ); + + const handleRequestAccess = useCallback( + (e: SyntheticEvent) => { + e.preventDefault(); + const { submitter } = e.nativeEvent; + const { requestId, comment, reject } = e.currentTarget; + + responseRequest( + requestId.value, + reject === submitter ? 'REJECTED' : 'APPROVED', + comment.value, + ); + }, + [responseRequest], + ); const handleSelectMember = useCallback( (event: React.ChangeEvent<{}>, selected: boolean) => { @@ -142,52 +290,139 @@ export function ProjectAccessTable({ if (selected) { setSelectedMembers(prevSelected => [ ...prevSelected, - members.find(member => member.name === name)!, + members.find(member => member.username === name)!, ]); } else { setSelectedMembers(prevSelected => - prevSelected.filter(member => member.name !== name), + prevSelected.filter(member => member.username !== name), ); } }, [members], ); - const handleTransferOwnership = useCallback( - (name: string) => { - transferOwnership(name); - setSnackbarMessage( - `User ownership has been successfully transferred to ${name}.`, - ); - }, - [transferOwnership], - ); + useEffect(() => { + const error = + getMembersError || + getAccessRequestsError || + changeRoleError || + removeMembersError || + responseRequestError; + if (error) { + // eslint-disable-next-line no-console + console.error(error); - const handleChangeRole = useCallback( - (name: string, role: Exclude) => { - changeRole(name, role); - setSnackbarMessage( - `User role for ${name} has been successfully changed to ${role}.`, - ); - }, - [changeRole], - ); + if (changeRoleError || removeMembersError || responseRequestError) { + errorApi.post(new Error('Failed to update project access')); + } + if (getMembersError || getAccessRequestsError) { + errorApi.post(new Error('Failed to get project members')); + } + } + }, [ + errorApi, + getMembersError, + getAccessRequestsError, + changeRoleError, + removeMembersError, + responseRequestError, + ]); + + useEffect(() => { + getMembers().then(setMembers); + getRequests().then(setRequests); + identityApi + .getProfileInfo() + .then(({ displayName }) => setCurrentUser(displayName)); + }, [getMembers, getRequests, identityApi]); const Cell = useCallback( ({ columnDef, rowData }) => { - const isOwner = rowData.roles.some( - ([roleName, selected]: [AccessRole, boolean]) => - roleName === 'Owner' && selected, - ); + if (rowData.requestId) { + const { name: roleName } = rowData.roles.find( + ({ + selected, + }: { + name: string; + disabled: boolean; + selected: boolean; + }) => selected, + )!; + if (columnDef.field === 'member') { + return ( + + {rowData.member} + + ); + } else if (columnDef.field === 'roles') { + return ( + +
+ + + + {`Requested access to the project with "${roleName}" role.`} + + + + + + + + + + + + + + + +
+
+ ); + } + } + if (columnDef.field === 'member') { return ( } label={rowData.member} - name={rowData.member} - disabled={isOwner || project.accessRole !== 'Owner'} + name={rowData.username} + disabled={rowData.disabled} onChange={handleSelectMember} + checked={rowData.selected} /> ); @@ -196,25 +431,29 @@ export function ProjectAccessTable({ {rowData.roles.map( - ([roleName, selected]: [AccessRole, boolean]) => ( - - } - label={roleName} - checked={selected} - disabled={ - isOwner || - (project.accessRole !== 'Owner' && roleName === 'Owner') - } - onChange={() => { - if (selected) return; - if (roleName === 'Owner') - handleTransferOwnership(rowData.member); - else handleChangeRole(rowData.member, roleName); - }} - /> - - ), + ({ + name, + selected, + disabled, + }: { + name: string; + disabled: boolean; + selected: boolean; + }) => { + return ( + + } + label={name} + checked={selected} + disabled={disabled} + onChange={() => + changeRole(rowData.username, accessRoleBackMap[name]) + } + /> + + ); + }, )} @@ -222,30 +461,9 @@ export function ProjectAccessTable({ } return {rowData[columnDef.field]}; }, - [ - project.accessRole, - handleSelectMember, - handleTransferOwnership, - handleChangeRole, - ], + [handleRequestAccess, classes, handleSelectMember, changeRole], ); - const handleRemoveSelected = () => { - selectedMembers.forEach(member => removeMember(member.name)); - setSelectedMembers([]); - setSnackbarMessage( - `You have successfully removed ${selectedMembers.length} contributor${ - selectedMembers.length > 1 ? 's' : '' - } from this project.`, - ); - setUndoRemove( - () => () => - [...selectedMembers] - .reverse() - .forEach(member => addMember(member.name, member.role)), - ); - }; - return ( <> + {(loadingMembers || loadingAccessRequests) && } diff --git a/plugins/parodos/src/components/projects/ProjectsTable.tsx b/plugins/parodos/src/components/projects/ProjectsTable.tsx index ee36d7f..24d5512 100644 --- a/plugins/parodos/src/components/projects/ProjectsTable.tsx +++ b/plugins/parodos/src/components/projects/ProjectsTable.tsx @@ -7,7 +7,6 @@ import { Grow, IconButton, Link, - makeStyles, MenuItem, MenuList, Paper, @@ -25,14 +24,8 @@ import { useStore } from '../../stores/workflowStore/workflowStore'; import { WorkflowStatus } from '../../models/workflowTaskSchema'; import { errorApiRef, useApi } from '@backstage/core-plugin-api'; -const useStyles = makeStyles(_ => ({ - manageAccessIcon: { - marginLeft: 'auto', - }, -})); - type ProjectsTableData = { - [K in Pick & + [K in Pick & string]: string; }; @@ -40,7 +33,7 @@ const columns: TableColumn[] = [ { title: 'PROJECT NAME', field: 'name', width: '20%' }, { title: 'OWNER', field: 'createdBy', width: '10%' }, { title: 'CREATED', field: 'createdDate', width: '10%' }, - { title: 'ACCESS', field: 'accessRole', width: '10%' }, + { title: 'ACCESS', field: 'accessRoles', width: '10%' }, { title: '', field: 'view', width: '5%' }, ]; @@ -48,15 +41,14 @@ type AccessRoleFilter = AccessRole | 'All'; const accessRoleMap: Record = { All: 'All', - Admin: 'Admin', - Owner: 'Owner', - Developer: 'Developer', + ADMIN: 'Admin', + OWNER: 'Owner', + DEVELOPER: 'Developer', } as const; type AccessRoleMapKeys = keyof typeof accessRoleMap; export function ProjectsTable({ projects }: { projects: Project[] }) { - const classes = useStyles(); const errorApi = useApi(errorApiRef); const requestAccessStatuses = useStore(state => state.requestAccessStatuses); const filterIconRef = useRef(null); @@ -110,7 +102,9 @@ export function ProjectsTable({ projects }: { projects: Project[] }) { () => projects .filter(p => - accessRoleFilter === 'All' ? true : p.accessRole === accessRoleFilter, + accessRoleFilter === 'All' + ? true + : p.accessRoles?.includes(accessRoleFilter), ) .filter(p => search ? p.name.toLocaleLowerCase().includes(search) : true, @@ -120,7 +114,7 @@ export function ProjectsTable({ projects }: { projects: Project[] }) { name: project.name, createdBy: project.createdBy, createdDate: getHumanReadableDate(project.createdDate), - accessRole: project.accessRole, + accessRoles: project.accessRoles ?? [], })), [accessRoleFilter, projects, search], ); @@ -181,10 +175,10 @@ export function ProjectsTable({ projects }: { projects: Project[] }) { components={{ Cell: ({ columnDef, rowData }) => { const hasAccess = - rowData.accessRole || + rowData.accessRoles.length || requestAccessStatusByProjectId[rowData.id] === 'COMPLETED'; - if (columnDef.field === 'accessRole') { + if (columnDef.field === 'accessRoles') { const isPendingAccess = requestAccessStatusByProjectId[rowData.id] === 'IN_PROGRESS' || requestAccessStatusByProjectId[rowData.id] === 'PENDING'; @@ -194,9 +188,13 @@ export function ProjectsTable({ projects }: { projects: Project[] }) { if (hasAccess) { return ( - - {rowData.accessRole ?? 'Access granted'} - + + + {rowData.accessRoles + .map((r: AccessRole) => accessRoleMap[r]) + .join(', ') ?? 'Access granted'} + + ; +export type AccessStatus = z.infer; + export type ProjectStatus = z.infer; +export const projectMember = z.object({ + username: z.string(), + firstName: z.string(), + lastName: z.string(), + roles: z.array(accessRole), +}); + +export const accessRequest = z.object({ + accessRequestId: z.string(), + comment: z.string().nullable().optional(), + firstname: z.string(), + lastname: z.string(), + projectId: z.string(), + role: accessRole, + status: accessStatus, + username: z.string(), +}); + +export const projectMembers = z.array(projectMember); + +export const accessRequests = z.array(accessRequest); + +export type ProjectMember = z.infer; + +export type AccessRequest = z.infer; + export const projectSchema = z.object({ id: z.string(), name: z.string(), @@ -29,7 +63,7 @@ export const projectSchema = z.object({ .transform(value => value.split('_').join(' ')), modifiedBy: z.string().nullable().optional(), createdBy: z.string().nullable().optional(), - accessRole: accessRole.nullable().optional(), + accessRoles: z.array(accessRole).nullable().optional(), }); export type Project = z.infer; diff --git a/plugins/parodos/src/stores/slices/projectsSlice.ts b/plugins/parodos/src/stores/slices/projectsSlice.ts index a9c27d6..517528f 100644 --- a/plugins/parodos/src/stores/slices/projectsSlice.ts +++ b/plugins/parodos/src/stores/slices/projectsSlice.ts @@ -31,7 +31,7 @@ export const createProjectsSlice: StateCreator< return project; }, projects: [], - async fetchProjects(fetch: FetchApi['fetch']) { + async fetchProjects(fetch: FetchApi['fetch'], force?: boolean) { set(state => { state.projectsLoading = true; }); @@ -44,6 +44,7 @@ export const createProjectsSlice: StateCreator< const newProjects = new Set(projects.map(p => p.id)); if ( + !force && get().initiallyLoaded && existing.size === newProjects.size && [...newProjects].every(id => existing.has(id)) diff --git a/plugins/parodos/src/stores/types.ts b/plugins/parodos/src/stores/types.ts index 40b79f6..954a0d8 100644 --- a/plugins/parodos/src/stores/types.ts +++ b/plugins/parodos/src/stores/types.ts @@ -45,7 +45,7 @@ export interface ProjectsSlice { string, { projectId: string; status: WorkflowStatus['status'] } >; - fetchProjects(fetch: FetchApi['fetch']): Promise; + fetchProjects(fetch: FetchApi['fetch'], force?: boolean): Promise; fetchRequestAccessStatuses(fetch: FetchApi['fetch']): Promise; hasProjects(): boolean; addProject(project: Project): void;