mirror of
https://github.com/documenso/documenso
synced 2026-04-21 13:27:18 +00:00
## Description - Add a new team page showing team details, global settings, members, and pending invites - Update the organisation page to display organisation usage and global settings - Show the role and ID of each organisation member, with navigation to their teams ## Checklist <!--- Please check the boxes that apply to this pull request. --> <!--- You can add or remove items as needed. --> - [ ] I have tested these changes locally and they work as expected. - [ ] I have added/updated tests that prove the effectiveness of these changes. - [ ] I have updated the documentation to reflect these changes, if applicable. - [ ] I have followed the project's coding style guidelines. - [ ] I have addressed the code review feedback from the previous submission, if applicable.
131 lines
3.4 KiB
TypeScript
131 lines
3.4 KiB
TypeScript
import { OrganisationMemberInviteStatus } from '@prisma/client';
|
|
|
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
|
import { getHighestOrganisationRoleInGroup } from '@documenso/lib/utils/organisations';
|
|
import { getHighestTeamRoleInGroup } from '@documenso/lib/utils/teams';
|
|
import { prisma } from '@documenso/prisma';
|
|
|
|
import { adminProcedure } from '../trpc';
|
|
import { ZGetAdminTeamRequestSchema, ZGetAdminTeamResponseSchema } from './get-admin-team.types';
|
|
|
|
export const getAdminTeamRoute = adminProcedure
|
|
.input(ZGetAdminTeamRequestSchema)
|
|
.output(ZGetAdminTeamResponseSchema)
|
|
.query(async ({ input, ctx }) => {
|
|
const { teamId } = input;
|
|
|
|
ctx.logger.info({
|
|
input: {
|
|
teamId,
|
|
},
|
|
});
|
|
|
|
const team = await prisma.team.findUnique({
|
|
where: {
|
|
id: teamId,
|
|
},
|
|
include: {
|
|
organisation: {
|
|
select: {
|
|
id: true,
|
|
name: true,
|
|
url: true,
|
|
ownerUserId: true,
|
|
},
|
|
},
|
|
teamEmail: true,
|
|
teamGlobalSettings: true,
|
|
},
|
|
});
|
|
|
|
if (!team) {
|
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
message: 'Team not found',
|
|
});
|
|
}
|
|
|
|
const [teamMembers, pendingInvites] = await Promise.all([
|
|
prisma.organisationMember.findMany({
|
|
where: {
|
|
organisationId: team.organisationId,
|
|
organisationGroupMembers: {
|
|
some: {
|
|
group: {
|
|
teamGroups: {
|
|
some: {
|
|
teamId,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
orderBy: {
|
|
createdAt: 'desc',
|
|
},
|
|
select: {
|
|
id: true,
|
|
userId: true,
|
|
createdAt: true,
|
|
organisationGroupMembers: {
|
|
include: {
|
|
group: {
|
|
include: {
|
|
teamGroups: {
|
|
where: {
|
|
teamId,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
user: {
|
|
select: {
|
|
id: true,
|
|
email: true,
|
|
name: true,
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
// Invites are organisation-scoped in the schema (no team relation), so this is intentionally
|
|
// all pending invites for the team's parent organisation.
|
|
prisma.organisationMemberInvite.findMany({
|
|
where: {
|
|
organisationId: team.organisationId,
|
|
status: OrganisationMemberInviteStatus.PENDING,
|
|
},
|
|
orderBy: {
|
|
createdAt: 'desc',
|
|
},
|
|
select: {
|
|
id: true,
|
|
email: true,
|
|
createdAt: true,
|
|
organisationRole: true,
|
|
status: true,
|
|
},
|
|
}),
|
|
]);
|
|
|
|
const mappedTeamMembers = teamMembers.map((teamMember) => {
|
|
const groups = teamMember.organisationGroupMembers.map(({ group }) => group);
|
|
|
|
return {
|
|
id: teamMember.id,
|
|
userId: teamMember.userId,
|
|
createdAt: teamMember.createdAt,
|
|
user: teamMember.user,
|
|
teamRole: getHighestTeamRoleInGroup(groups.flatMap((group) => group.teamGroups)),
|
|
organisationRole: getHighestOrganisationRoleInGroup(groups),
|
|
};
|
|
});
|
|
|
|
return {
|
|
...team,
|
|
memberCount: mappedTeamMembers.length,
|
|
teamMembers: mappedTeamMembers,
|
|
pendingInvites,
|
|
};
|
|
});
|