import Link from 'next/link'; import { redirect } from 'next/navigation'; import { Plural, Trans } from '@lingui/macro'; import { ChevronLeft, Users2 } from 'lucide-react'; import { match } from 'ts-pattern'; import { isUserEnterprise } from '@documenso/ee/server-only/util/is-document-enterprise'; import { DOCUMENSO_ENCRYPTION_KEY } from '@documenso/lib/constants/crypto'; import { getRequiredServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session'; import { getDocumentWithDetailsById } from '@documenso/lib/server-only/document/get-document-with-details-by-id'; import { DocumentVisibility } from '@documenso/lib/types/document-visibility'; import { symmetricDecrypt } from '@documenso/lib/universal/crypto'; import { formatDocumentsPath } from '@documenso/lib/utils/teams'; import type { Team } from '@documenso/prisma/client'; import { TeamMemberRole } from '@documenso/prisma/client'; import { DocumentStatus as InternalDocumentStatus } from '@documenso/prisma/client'; import { EditDocumentForm } from '~/app/(dashboard)/documents/[id]/edit-document'; import { StackAvatarsWithTooltip } from '~/components/(dashboard)/avatar/stack-avatars-with-tooltip'; import { DocumentStatus } from '~/components/formatter/document-status'; export type DocumentEditPageViewProps = { params: { id: string; }; team?: Team & { currentTeamMember: { role: TeamMemberRole } }; }; export const DocumentEditPageView = async ({ params, team }: DocumentEditPageViewProps) => { const { id } = params; const documentId = Number(id); const documentRootPath = formatDocumentsPath(team?.url); if (!documentId || Number.isNaN(documentId)) { redirect(documentRootPath); } const { user } = await getRequiredServerComponentSession(); const document = await getDocumentWithDetailsById({ id: documentId, userId: user.id, teamId: team?.id, }).catch(() => null); if (document?.teamId && !team?.url) { redirect(documentRootPath); } const documentVisibility = document?.visibility; const currentTeamMemberRole = team?.currentTeamMember?.role; const isRecipient = document?.Recipient.find((recipient) => recipient.email === user.email); let canAccessDocument = true; if (!isRecipient && document?.userId !== user.id) { canAccessDocument = match([documentVisibility, currentTeamMemberRole]) .with([DocumentVisibility.EVERYONE, TeamMemberRole.ADMIN], () => true) .with([DocumentVisibility.EVERYONE, TeamMemberRole.MANAGER], () => true) .with([DocumentVisibility.EVERYONE, TeamMemberRole.MEMBER], () => true) .with([DocumentVisibility.MANAGER_AND_ABOVE, TeamMemberRole.ADMIN], () => true) .with([DocumentVisibility.MANAGER_AND_ABOVE, TeamMemberRole.MANAGER], () => true) .with([DocumentVisibility.ADMIN, TeamMemberRole.ADMIN], () => true) .otherwise(() => false); } if (!document) { redirect(documentRootPath); } if (team && !canAccessDocument) { redirect(documentRootPath); } if (document.status === InternalDocumentStatus.COMPLETED) { redirect(`${documentRootPath}/${documentId}`); } const { documentMeta, Recipient: recipients } = document; if (documentMeta?.password) { const key = DOCUMENSO_ENCRYPTION_KEY; if (!key) { throw new Error('Missing DOCUMENSO_ENCRYPTION_KEY'); } const securePassword = Buffer.from( symmetricDecrypt({ key, data: documentMeta.password, }), ).toString('utf-8'); documentMeta.password = securePassword; } const isDocumentEnterprise = await isUserEnterprise({ userId: user.id, teamId: team?.id, }); return (