diff --git a/apps/api/package.json b/apps/api/package.json index 3f92db055..ada1af49e 100644 --- a/apps/api/package.json +++ b/apps/api/package.json @@ -1,134 +1,131 @@ { - "name": "@comp/api", - "description": "", - "version": "0.0.1", - "author": "", - "dependencies": { - "@ai-sdk/anthropic": "^2.0.53", - "@ai-sdk/groq": "^2.0.32", - "@ai-sdk/openai": "^2.0.65", - "@aws-sdk/client-s3": "^3.859.0", - "@aws-sdk/client-securityhub": "^3.948.0", - "@aws-sdk/client-sts": "^3.948.0", - "@aws-sdk/s3-request-presigner": "^3.859.0", - "@browserbasehq/sdk": "^2.6.0", - "@browserbasehq/stagehand": "^3.0.5", - "@comp/integration-platform": "workspace:*", - "@mendable/firecrawl-js": "^4.9.3", - "@nestjs/common": "^11.0.1", - "@nestjs/config": "^4.0.2", - "@nestjs/core": "^11.0.1", - "@nestjs/platform-express": "^11.1.5", - "@nestjs/swagger": "^11.2.0", - "@nestjs/throttler": "^6.5.0", - "@prisma/client": "6.18.0", - "@prisma/instrumentation": "^6.13.0", - "@react-email/components": "^0.0.41", - "@trigger.dev/build": "4.0.6", - "@trigger.dev/sdk": "4.0.6", - "@trycompai/db": "1.3.22", - "@trycompai/email": "workspace:*", - "@upstash/redis": "^1.34.2", - "@upstash/vector": "^1.2.2", - "adm-zip": "^0.5.16", - "ai": "^5.0.60", - "archiver": "^7.0.1", - "axios": "^1.12.2", - "better-auth": "^1.3.27", - "class-transformer": "^0.5.1", - "class-validator": "^0.14.2", - "dotenv": "^17.2.3", - "esbuild": "^0.27.1", - "exceljs": "^4.4.0", - "express": "^4.21.2", - "helmet": "^8.1.0", - "jose": "^6.0.12", - "jspdf": "^3.0.3", - "mammoth": "^1.8.0", - "nanoid": "^5.1.6", - "pdf-lib": "^1.17.1", - "playwright-core": "^1.57.0", - "prisma": "6.18.0", - "react": "^19.1.1", - "react-dom": "^19.1.0", - "reflect-metadata": "^0.2.2", - "resend": "^6.4.2", - "rxjs": "^7.8.1", - "safe-stable-stringify": "^2.5.0", - "swagger-ui-express": "^5.0.1", - "xlsx": "^0.18.5", - "zod": "^4.0.14" - }, - "devDependencies": { - "@eslint/eslintrc": "^3.2.0", - "@eslint/js": "^9.18.0", - "@nestjs/cli": "^11.0.0", - "@nestjs/schematics": "^11.0.0", - "@nestjs/testing": "^11.0.1", - "@types/adm-zip": "^0.5.7", - "@types/archiver": "^6.0.3", - "@types/express": "^5.0.0", - "@types/jest": "^30.0.0", - "@types/multer": "^1.4.12", - "@types/node": "^24.0.3", - "@types/supertest": "^6.0.2", - "@types/swagger-ui-express": "^4.1.8", - "eslint": "^9.18.0", - "eslint-config-prettier": "^10.0.1", - "eslint-plugin-prettier": "^5.2.2", - "globals": "^16.0.0", - "jest": "^30.0.0", - "prettier": "^3.5.3", - "source-map-support": "^0.5.21", - "supertest": "^7.0.0", - "ts-jest": "^29.2.5", - "ts-loader": "^9.5.2", - "ts-node": "^10.9.2", - "tsconfig-paths": "^4.2.0", - "typescript": "^5.8.3", - "typescript-eslint": "^8.20.0" - }, - "jest": { - "moduleFileExtensions": [ - "js", - "json", - "ts" - ], - "rootDir": "src", - "testRegex": ".*\\.spec\\.ts$", - "transform": { - "^.+\\.(t|j)s$": "ts-jest" + "name": "@comp/api", + "description": "", + "version": "0.0.1", + "author": "", + "dependencies": { + "@ai-sdk/anthropic": "^2.0.53", + "@ai-sdk/groq": "^2.0.32", + "@ai-sdk/openai": "^2.0.65", + "@aws-sdk/client-s3": "^3.859.0", + "@aws-sdk/client-securityhub": "^3.948.0", + "@aws-sdk/client-sts": "^3.948.0", + "@aws-sdk/s3-request-presigner": "^3.859.0", + "@browserbasehq/sdk": "^2.6.0", + "@browserbasehq/stagehand": "^3.0.5", + "@comp/company": "workspace:*", + "@comp/integration-platform": "workspace:*", + "@mendable/firecrawl-js": "^4.9.3", + "@nestjs/common": "^11.0.1", + "@nestjs/config": "^4.0.2", + "@nestjs/core": "^11.0.1", + "@nestjs/platform-express": "^11.1.5", + "@nestjs/swagger": "^11.2.0", + "@nestjs/throttler": "^6.5.0", + "@prisma/client": "6.18.0", + "@prisma/instrumentation": "^6.13.0", + "@react-email/components": "^0.0.41", + "@trigger.dev/build": "4.0.6", + "@trigger.dev/sdk": "4.0.6", + "@trycompai/db": "1.3.22", + "@trycompai/email": "workspace:*", + "@upstash/redis": "^1.34.2", + "@upstash/vector": "^1.2.2", + "adm-zip": "^0.5.16", + "ai": "^5.0.60", + "archiver": "^7.0.1", + "axios": "^1.12.2", + "better-auth": "^1.3.27", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.2", + "dotenv": "^17.2.3", + "esbuild": "^0.27.1", + "exceljs": "^4.4.0", + "express": "^4.21.2", + "helmet": "^8.1.0", + "jose": "^6.0.12", + "jspdf": "^3.0.3", + "mammoth": "^1.8.0", + "nanoid": "^5.1.6", + "pdf-lib": "^1.17.1", + "playwright-core": "^1.57.0", + "prisma": "6.18.0", + "react": "^19.1.1", + "react-dom": "^19.1.0", + "reflect-metadata": "^0.2.2", + "resend": "^6.4.2", + "rxjs": "^7.8.1", + "safe-stable-stringify": "^2.5.0", + "swagger-ui-express": "^5.0.1", + "xlsx": "^0.18.5", + "zod": "^4.0.14" }, - "collectCoverageFrom": [ - "**/*.(t|j)s" - ], - "coverageDirectory": "../coverage", - "testEnvironment": "node" - }, - "license": "UNLICENSED", - "private": true, - "scripts": { - "build": "nest build", - "build:docker": "bunx prisma generate && nest build", - "db:generate": "bun run db:getschema && bunx prisma generate", - "db:getschema": "node ../../packages/db/scripts/combine-schemas.js && cp ../../packages/db/dist/schema.prisma prisma/schema.prisma", - "db:migrate": "cd ../../packages/db && bunx prisma migrate dev && cd ../../apps/api", - "deploy:trigger-prod": "npx trigger.dev@4.0.6 deploy", - "dev": "bunx concurrently --kill-others --names \"nest,trigger\" --prefix-colors \"green,blue\" \"nest start --watch\" \"bunx trigger.dev@4.0.6 dev\"", - "dev:nest": "nest start --watch", - "dev:trigger": "bunx trigger.dev@4.0.6 dev", - "format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"", - "lint": "eslint \"{src,apps,libs,test}/**/*.ts\" --fix", - "prebuild": "bun run db:generate", - "start": "nest start", - "start:debug": "nest start --debug --watch", - "start:dev": "nest start --watch", - "start:prod": "node dist/main", - "test": "jest", - "test:cov": "jest --coverage", - "test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand", - "test:e2e": "jest --config ./test/jest-e2e.json", - "test:watch": "jest --watch", - "typecheck": "tsc --noEmit" - } + "devDependencies": { + "@eslint/eslintrc": "^3.2.0", + "@eslint/js": "^9.18.0", + "@nestjs/cli": "^11.0.0", + "@nestjs/schematics": "^11.0.0", + "@nestjs/testing": "^11.0.1", + "@types/adm-zip": "^0.5.7", + "@types/archiver": "^6.0.3", + "@types/express": "^5.0.0", + "@types/jest": "^30.0.0", + "@types/multer": "^1.4.12", + "@types/node": "^24.0.3", + "@types/supertest": "^6.0.2", + "@types/swagger-ui-express": "^4.1.8", + "eslint": "^9.18.0", + "eslint-config-prettier": "^10.0.1", + "eslint-plugin-prettier": "^5.2.2", + "globals": "^16.0.0", + "jest": "^30.0.0", + "prettier": "^3.5.3", + "source-map-support": "^0.5.21", + "supertest": "^7.0.0", + "ts-jest": "^29.2.5", + "ts-loader": "^9.5.2", + "ts-node": "^10.9.2", + "tsconfig-paths": "^4.2.0", + "typescript": "^5.8.3", + "typescript-eslint": "^8.20.0" + }, + "jest": { + "moduleFileExtensions": [ + "js", "json", "ts" + ], + "rootDir": "src", + "testRegex": ".*\\.spec\\.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "collectCoverageFrom": ["**/*.(t|j)s"], + "coverageDirectory": "../coverage", + "testEnvironment": "node" + }, + "license": "UNLICENSED", + "private": true, + "scripts": { + "build": "nest build", + "build:docker": "bunx prisma generate && nest build", + "db:generate": "bun run db:getschema && bunx prisma generate", + "db:getschema": "node ../../packages/db/scripts/combine-schemas.js && cp ../../packages/db/dist/schema.prisma prisma/schema.prisma", + "db:migrate": "cd ../../packages/db && bunx prisma migrate dev && cd ../../apps/api", + "deploy:trigger-prod": "npx trigger.dev@4.0.6 deploy", + "dev": "bunx concurrently --kill-others --names \"nest,trigger\" --prefix-colors \"green,blue\" \"nest start --watch\" \"bunx trigger.dev@4.0.6 dev\"", + "dev:nest": "nest start --watch", + "dev:trigger": "bunx trigger.dev@4.0.6 dev", + "format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"", + "lint": "eslint \"{src,apps,libs,test}/**/*.ts\" --fix", + "prebuild": "bun run db:generate", + "start": "nest start", + "start:debug": "nest start --debug --watch", + "start:dev": "nest start --watch", + "start:prod": "node dist/main", + "test": "jest", + "test:cov": "jest --coverage", + "test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand", + "test:e2e": "jest --config ./test/jest-e2e.json", + "test:watch": "jest --watch", + "typecheck": "tsc --noEmit" + } } diff --git a/apps/api/src/app.module.ts b/apps/api/src/app.module.ts index 2645cb341..012a3c550 100644 --- a/apps/api/src/app.module.ts +++ b/apps/api/src/app.module.ts @@ -35,6 +35,7 @@ import { TaskManagementModule } from './task-management/task-management.module'; import { AssistantChatModule } from './assistant-chat/assistant-chat.module'; import { OrgChartModule } from './org-chart/org-chart.module'; import { TrainingModule } from './training/training.module'; +import { EvidenceFormsModule } from './evidence-forms/evidence-forms.module'; @Module({ imports: [ @@ -82,6 +83,7 @@ import { TrainingModule } from './training/training.module'; AssistantChatModule, TrainingModule, OrgChartModule, + EvidenceFormsModule, ], controllers: [AppController], providers: [ diff --git a/apps/api/src/evidence-forms/evidence-forms.controller.ts b/apps/api/src/evidence-forms/evidence-forms.controller.ts new file mode 100644 index 000000000..c478dfb58 --- /dev/null +++ b/apps/api/src/evidence-forms/evidence-forms.controller.ts @@ -0,0 +1,212 @@ +import { AuthContext, OrganizationId } from '@/auth/auth-context.decorator'; +import { HybridAuthGuard } from '@/auth/hybrid-auth.guard'; +import type { AuthContext as AuthContextType } from '@/auth/types'; +import { + Body, + Controller, + Get, + Header, + Param, + Patch, + Post, + Query, + Res, + UseGuards, +} from '@nestjs/common'; +import { ApiHeader, ApiOperation, ApiSecurity, ApiTags } from '@nestjs/swagger'; +import type { Response } from 'express'; +import { EvidenceFormsService } from './evidence-forms.service'; + +@ApiTags('Evidence Forms') +@Controller({ path: 'evidence-forms', version: '1' }) +@UseGuards(HybridAuthGuard) +@ApiSecurity('apikey') +@ApiHeader({ + name: 'X-Organization-Id', + description: + 'Organization ID (required for session auth, optional for API key auth)', + required: false, +}) +export class EvidenceFormsController { + constructor(private readonly evidenceFormsService: EvidenceFormsService) {} + + @Get() + @ApiOperation({ + summary: 'List evidence forms', + description: 'List all available pre-built evidence forms', + }) + listForms() { + return this.evidenceFormsService.listForms(); + } + + @Get('statuses') + @ApiOperation({ + summary: 'Get submission statuses for all forms', + description: + 'Returns the latest submission date per form type for the active organization', + }) + async getFormStatuses(@OrganizationId() organizationId: string) { + return this.evidenceFormsService.getFormStatuses(organizationId); + } + + @Get('my-submissions') + @ApiOperation({ + summary: 'Get current user submissions', + description: + 'Returns all evidence form submissions by the authenticated user for the active organization', + }) + async getMySubmissions( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Query('formType') formType?: string, + ) { + return this.evidenceFormsService.getMySubmissions({ + organizationId, + authContext, + formType, + }); + } + + @Get('my-submissions/pending-count') + @ApiOperation({ + summary: 'Get pending submission count for current user', + description: + 'Returns the count of pending evidence submissions for the authenticated user', + }) + async getPendingSubmissionCount( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + ) { + return this.evidenceFormsService.getPendingSubmissionCount({ + organizationId, + authContext, + }); + } + + @Get(':formType') + @ApiOperation({ + summary: 'Get form definition and submissions', + description: + 'Fetch a specific form definition with submissions for the active organization', + }) + async getFormWithSubmissions( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Param('formType') formType: string, + @Query('search') search?: string, + @Query('limit') limit?: string, + @Query('offset') offset?: string, + ) { + return this.evidenceFormsService.getFormWithSubmissions({ + organizationId, + authContext, + formType, + search, + limit, + offset, + }); + } + + @Get(':formType/submissions/:submissionId') + @ApiOperation({ + summary: 'Get a single submission', + description: + 'Fetch one evidence form submission for the active organization', + }) + async getSubmission( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Param('formType') formType: string, + @Param('submissionId') submissionId: string, + ) { + return this.evidenceFormsService.getSubmission({ + organizationId, + authContext, + formType, + submissionId, + }); + } + + @Post(':formType/submissions') + @ApiOperation({ + summary: 'Submit evidence form entry', + description: + 'Create a new organization-scoped evidence form submission using Zod-validated payloads', + }) + async submitForm( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Param('formType') formType: string, + @Body() body: unknown, + ) { + return this.evidenceFormsService.submitForm({ + organizationId, + formType, + payload: body, + authContext, + }); + } + + @Patch(':formType/submissions/:submissionId/review') + @ApiOperation({ + summary: 'Review a submission', + description: + 'Approve or reject an evidence form submission with an optional reason', + }) + async reviewSubmission( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Param('formType') formType: string, + @Param('submissionId') submissionId: string, + @Body() body: unknown, + ) { + return this.evidenceFormsService.reviewSubmission({ + organizationId, + formType, + submissionId, + payload: body, + authContext, + }); + } + + @Post('uploads') + @ApiOperation({ + summary: 'Upload evidence form file', + description: + 'Upload a file for evidence form fields and return file metadata for submission payload', + }) + async uploadFile( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Body() body: unknown, + ) { + return this.evidenceFormsService.uploadFile({ + organizationId, + authContext, + payload: body, + }); + } + + @Get(':formType/export.csv') + @ApiOperation({ + summary: 'Export form submissions to CSV', + description: 'Export all form submissions for an organization as CSV', + }) + @Header('Content-Type', 'text/csv') + async exportCsv( + @OrganizationId() organizationId: string, + @AuthContext() authContext: AuthContextType, + @Param('formType') formType: string, + @Res() res: Response, + ) { + const csv = await this.evidenceFormsService.exportCsv({ + organizationId, + authContext, + formType, + }); + + const filename = `${formType}-submissions-${new Date().toISOString().slice(0, 10)}.csv`; + res.setHeader('Content-Disposition', `attachment; filename="${filename}"`); + res.send(csv); + } +} diff --git a/apps/api/src/evidence-forms/evidence-forms.definitions.ts b/apps/api/src/evidence-forms/evidence-forms.definitions.ts new file mode 100644 index 000000000..2b94ea7b2 --- /dev/null +++ b/apps/api/src/evidence-forms/evidence-forms.definitions.ts @@ -0,0 +1,11 @@ +// Single source of truth: re-export from shared @comp/company package +export { + evidenceFormTypeSchema, + evidenceFormFileSchema, + evidenceFormSubmissionSchemaMap, + evidenceFormDefinitions, + evidenceFormDefinitionList, + type EvidenceFormType, + type EvidenceFormFieldDefinition, + type EvidenceFormDefinition, +} from '@comp/company'; diff --git a/apps/api/src/evidence-forms/evidence-forms.module.ts b/apps/api/src/evidence-forms/evidence-forms.module.ts new file mode 100644 index 000000000..333ffaa5d --- /dev/null +++ b/apps/api/src/evidence-forms/evidence-forms.module.ts @@ -0,0 +1,13 @@ +import { Module } from '@nestjs/common'; +import { AttachmentsModule } from '@/attachments/attachments.module'; +import { AuthModule } from '@/auth/auth.module'; +import { EvidenceFormsController } from './evidence-forms.controller'; +import { EvidenceFormsService } from './evidence-forms.service'; + +@Module({ + imports: [AuthModule, AttachmentsModule], + controllers: [EvidenceFormsController], + providers: [EvidenceFormsService], + exports: [EvidenceFormsService], +}) +export class EvidenceFormsModule {} diff --git a/apps/api/src/evidence-forms/evidence-forms.service.ts b/apps/api/src/evidence-forms/evidence-forms.service.ts new file mode 100644 index 000000000..0e63d7467 --- /dev/null +++ b/apps/api/src/evidence-forms/evidence-forms.service.ts @@ -0,0 +1,619 @@ +import { AttachmentsService } from '@/attachments/attachments.service'; +import type { AuthContext } from '@/auth/types'; +import { db } from '@trycompai/db'; +import { + BadRequestException, + Injectable, + NotFoundException, + UnauthorizedException, +} from '@nestjs/common'; +import { z } from 'zod'; +import { + evidenceFormDefinitionList, + evidenceFormDefinitions, + evidenceFormSubmissionSchemaMap, + evidenceFormTypeSchema, + type EvidenceFormFieldDefinition, + type EvidenceFormType, +} from './evidence-forms.definitions'; + +const listQuerySchema = z.object({ + search: z.string().trim().optional(), + limit: z.coerce.number().int().min(1).max(200).optional().default(50), + offset: z.coerce.number().int().min(0).optional().default(0), +}); + +const uploadSchema = z.object({ + formType: evidenceFormTypeSchema, + fileName: z.string().min(1), + fileType: z.string().min(1), + fileData: z.string().min(1), +}); + +const reviewSchema = z.object({ + action: z.enum(['approved', 'rejected']), + reason: z.string().trim().optional(), +}); + +const EVIDENCE_FORM_REVIEWER_ROLES = ['owner', 'admin', 'auditor'] as const; +const MAX_UPLOAD_FILE_SIZE_BYTES = 100 * 1024 * 1024; +const MAX_UPLOAD_BASE64_LENGTH = Math.ceil(MAX_UPLOAD_FILE_SIZE_BYTES / 3) * 4; + +function toCsvRow(values: string[]): string { + return values.map((value) => `"${value.replace(/"/g, '""')}"`).join(','); +} + +function flattenValue(value: unknown): string { + if (value === null || value === undefined) { + return ''; + } + + if (typeof value === 'object') { + if ( + 'fileName' in value && + typeof value.fileName === 'string' && + 'downloadUrl' in value && + typeof value.downloadUrl === 'string' + ) { + return value.downloadUrl; + } + return JSON.stringify(value); + } + + if (typeof value === 'string') { + return value; + } + if ( + typeof value === 'number' || + typeof value === 'boolean' || + typeof value === 'bigint' + ) { + return value.toString(); + } + if (typeof value === 'symbol') { + return value.description ?? ''; + } + + return ''; +} + +function flattenMatrixRows( + value: unknown, + field: EvidenceFormFieldDefinition, +): string { + if (!Array.isArray(value)) { + return ''; + } + + const columns = Array.isArray(field.columns) ? field.columns : []; + if (columns.length === 0) { + return JSON.stringify(value); + } + + return value + .filter((row) => row && typeof row === 'object') + .map((row) => { + const rowRecord = row as Record; + return columns + .map((column) => { + const cellValue = rowRecord[column.key]; + const normalizedValue = + typeof cellValue === 'string' ? cellValue : ''; + return `${column.label}: ${normalizedValue}`; + }) + .join(' | '); + }) + .join(' || '); +} + +@Injectable() +export class EvidenceFormsService { + constructor(private readonly attachmentsService: AttachmentsService) {} + + private requireJwtUser(authContext: AuthContext): string { + if (authContext.isApiKey || authContext.authType === 'api-key') { + throw new UnauthorizedException( + 'This endpoint requires JWT authentication and does not support API key authentication', + ); + } + + if (!authContext.userId) { + throw new UnauthorizedException('Authenticated user session is required'); + } + + return authContext.userId; + } + + private requirePrivilegedEvidenceAccess(authContext: AuthContext): string { + const userId = this.requireJwtUser(authContext); + const roles = authContext.userRoles ?? []; + const hasRequiredRole = EVIDENCE_FORM_REVIEWER_ROLES.some((role) => + roles.includes(role), + ); + + if (!hasRequiredRole) { + throw new UnauthorizedException( + `Access denied. Required one of roles: ${EVIDENCE_FORM_REVIEWER_ROLES.join(', ')}`, + ); + } + + return userId; + } + + private decodeBase64File(fileData: string): Buffer { + const normalized = fileData.trim(); + if (normalized.length === 0 || normalized.length % 4 !== 0) { + throw new BadRequestException( + 'Invalid file data. Expected base64 string.', + ); + } + + const base64Pattern = /^[A-Za-z0-9+/]+={0,2}$/; + if (!base64Pattern.test(normalized)) { + throw new BadRequestException( + 'Invalid file data. Expected base64 string.', + ); + } + + const fileBuffer = Buffer.from(normalized, 'base64'); + if (!fileBuffer.length) { + throw new BadRequestException('File cannot be empty'); + } + + return fileBuffer; + } + + listForms() { + return evidenceFormDefinitionList; + } + + async getFormStatuses(organizationId: string) { + const results = await db.evidenceSubmission.groupBy({ + by: ['formType'], + where: { organizationId }, + _max: { submittedAt: true }, + }); + + const statuses: Record = {}; + + for (const form of evidenceFormDefinitionList) { + const match = results.find((r) => r.formType === form.type); + statuses[form.type] = { + lastSubmittedAt: match?._max.submittedAt?.toISOString() ?? null, + }; + } + + return statuses; + } + + async getFormWithSubmissions(params: { + organizationId: string; + authContext: AuthContext; + formType: string; + search?: string; + limit?: string; + offset?: string; + }) { + const { organizationId, formType } = params; + this.requirePrivilegedEvidenceAccess(params.authContext); + + const parsedType = evidenceFormTypeSchema.safeParse(formType); + if (!parsedType.success) { + throw new BadRequestException('Unsupported form type'); + } + + const parsedQuery = listQuerySchema.safeParse({ + search: params.search, + limit: params.limit, + offset: params.offset, + }); + if (!parsedQuery.success) { + throw new BadRequestException(parsedQuery.error.flatten()); + } + const query = parsedQuery.data; + + const submissions = await db.evidenceSubmission.findMany({ + where: { + organizationId, + formType: parsedType.data, + }, + include: { + submittedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + }, + orderBy: { + submittedAt: 'desc', + }, + }); + + const filtered = query.search + ? submissions.filter((submission) => { + const searchTarget = JSON.stringify(submission.data).toLowerCase(); + return searchTarget.includes(query.search!.toLowerCase()); + }) + : submissions; + + const paginated = filtered.slice(query.offset, query.offset + query.limit); + + return { + form: evidenceFormDefinitions[parsedType.data], + submissions: paginated, + total: filtered.length, + }; + } + + async getSubmission(params: { + organizationId: string; + authContext: AuthContext; + formType: string; + submissionId: string; + }) { + this.requirePrivilegedEvidenceAccess(params.authContext); + + const parsedType = evidenceFormTypeSchema.safeParse(params.formType); + if (!parsedType.success) { + throw new BadRequestException('Unsupported form type'); + } + + const submission = await db.evidenceSubmission.findFirst({ + where: { + id: params.submissionId, + organizationId: params.organizationId, + formType: parsedType.data, + }, + include: { + submittedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + reviewedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + }, + }); + + if (!submission) { + throw new NotFoundException('Submission not found'); + } + + return { + form: evidenceFormDefinitions[parsedType.data], + submission, + }; + } + + async submitForm(params: { + organizationId: string; + formType: string; + payload: unknown; + authContext: AuthContext; + }) { + const parsedType = evidenceFormTypeSchema.safeParse(params.formType); + if (!parsedType.success) { + throw new BadRequestException('Unsupported form type'); + } + + if (!params.authContext.userId) { + throw new BadRequestException( + 'Authenticated user session is required to submit evidence forms', + ); + } + + const formDefinition = evidenceFormDefinitions[parsedType.data]; + const nowIso = new Date().toISOString(); + + if (!params.payload || typeof params.payload !== 'object') { + throw new BadRequestException('Submission payload must be an object'); + } + + const payloadObject: Record = { + ...(params.payload as Record), + }; + + if (formDefinition.submissionDateMode === 'auto') { + payloadObject.submissionDate = nowIso; + } + + const schema = evidenceFormSubmissionSchemaMap[parsedType.data]; + const parsedPayload = schema.safeParse(payloadObject); + if (!parsedPayload.success) { + throw new BadRequestException(parsedPayload.error.flatten()); + } + + return await db.evidenceSubmission.create({ + data: { + organizationId: params.organizationId, + formType: parsedType.data, + submittedById: params.authContext.userId, + data: parsedPayload.data, + }, + include: { + submittedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + }, + }); + } + + async uploadFile(params: { + organizationId: string; + authContext: AuthContext; + payload: unknown; + }) { + if (!params.authContext.userId) { + throw new BadRequestException( + 'Authenticated user session is required to upload evidence files', + ); + } + + const parsed = uploadSchema.safeParse(params.payload); + if (!parsed.success) { + throw new BadRequestException(parsed.error.flatten()); + } + + if (parsed.data.fileData.length > MAX_UPLOAD_BASE64_LENGTH) { + throw new BadRequestException( + `File exceeds the ${MAX_UPLOAD_FILE_SIZE_BYTES / (1024 * 1024)}MB limit`, + ); + } + + const fileBuffer = this.decodeBase64File(parsed.data.fileData); + if (fileBuffer.length > MAX_UPLOAD_FILE_SIZE_BYTES) { + throw new BadRequestException( + `File exceeds the ${MAX_UPLOAD_FILE_SIZE_BYTES / (1024 * 1024)}MB limit`, + ); + } + + const fileKey = await this.attachmentsService.uploadToS3( + fileBuffer, + parsed.data.fileName, + parsed.data.fileType, + params.organizationId, + 'evidence-forms', + parsed.data.formType, + ); + + const downloadUrl = + await this.attachmentsService.getPresignedDownloadUrl(fileKey); + + return { + fileName: parsed.data.fileName, + fileKey, + downloadUrl, + }; + } + + async exportCsv(params: { + organizationId: string; + formType: string; + authContext: AuthContext; + }) { + this.requirePrivilegedEvidenceAccess(params.authContext); + + const parsedType = evidenceFormTypeSchema.safeParse(params.formType); + if (!parsedType.success) { + throw new BadRequestException('Unsupported form type'); + } + + const formType: EvidenceFormType = parsedType.data; + const form = evidenceFormDefinitions[formType]; + + const submissions = await db.evidenceSubmission.findMany({ + where: { + organizationId: params.organizationId, + formType, + }, + include: { + submittedBy: { + select: { + name: true, + email: true, + }, + }, + }, + orderBy: { + submittedAt: 'desc', + }, + }); + + if (submissions.length === 0) { + throw new BadRequestException( + 'No submissions available for export for this form', + ); + } + + const headers = [ + 'submissionId', + 'submissionDate', + 'submittedByName', + 'submittedByEmail', + ...form.fields + .filter((field) => field.key !== 'submissionDate') + .map((field) => field.key), + ]; + + const rows = await Promise.all( + submissions.map(async (submission) => { + const data = submission.data as Record; + const fieldValues = await Promise.all( + form.fields + .filter((field) => field.key !== 'submissionDate') + .map(async (field) => { + const rawValue = data[field.key]; + if ( + rawValue && + typeof rawValue === 'object' && + 'fileKey' in rawValue && + typeof rawValue.fileKey === 'string' + ) { + const signedUrl = + await this.attachmentsService.getPresignedDownloadUrl( + rawValue.fileKey, + ); + return signedUrl; + } + if (field.type === 'matrix') { + return flattenMatrixRows(rawValue, field); + } + return flattenValue(rawValue); + }), + ); + + return [ + submission.id, + typeof data.submissionDate === 'string' + ? data.submissionDate + : submission.submittedAt.toISOString(), + submission.submittedBy?.name ?? '', + submission.submittedBy?.email ?? '', + ...fieldValues, + ]; + }), + ); + + const csvLines = [toCsvRow(headers), ...rows.map((row) => toCsvRow(row))]; + return csvLines.join('\n'); + } + + async reviewSubmission(params: { + organizationId: string; + formType: string; + submissionId: string; + payload: unknown; + authContext: AuthContext; + }) { + const parsedType = evidenceFormTypeSchema.safeParse(params.formType); + if (!parsedType.success) { + throw new BadRequestException('Unsupported form type'); + } + + const reviewerUserId = this.requirePrivilegedEvidenceAccess( + params.authContext, + ); + + const parsed = reviewSchema.safeParse(params.payload); + if (!parsed.success) { + throw new BadRequestException(parsed.error.flatten()); + } + + if (parsed.data.action === 'rejected' && !parsed.data.reason) { + throw new BadRequestException( + 'A reason is required when rejecting a submission', + ); + } + + const submission = await db.evidenceSubmission.findFirst({ + where: { + id: params.submissionId, + organizationId: params.organizationId, + formType: parsedType.data, + }, + }); + + if (!submission) { + throw new NotFoundException('Submission not found'); + } + + if (submission.status !== 'pending') { + throw new BadRequestException( + 'Submission must be pending to be reviewed', + ); + } + + return await db.evidenceSubmission.update({ + where: { id: params.submissionId }, + data: { + status: parsed.data.action, + reviewedById: reviewerUserId, + reviewedAt: new Date(), + reviewReason: parsed.data.reason ?? null, + }, + include: { + submittedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + reviewedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + }, + }); + } + + async getMySubmissions(params: { + organizationId: string; + authContext: AuthContext; + formType?: string; + }) { + const userId = this.requireJwtUser(params.authContext); + + const where: Record = { + organizationId: params.organizationId, + submittedById: userId, + }; + + if (params.formType) { + const parsedType = evidenceFormTypeSchema.safeParse(params.formType); + if (!parsedType.success) { + throw new BadRequestException('Unsupported form type'); + } + where.formType = parsedType.data; + } + + return await db.evidenceSubmission.findMany({ + where, + include: { + reviewedBy: { + select: { + id: true, + name: true, + email: true, + }, + }, + }, + orderBy: { + submittedAt: 'desc', + }, + }); + } + + async getPendingSubmissionCount(params: { + organizationId: string; + authContext: AuthContext; + }) { + const userId = this.requireJwtUser(params.authContext); + + const count = await db.evidenceSubmission.count({ + where: { + organizationId: params.organizationId, + submittedById: userId, + status: 'pending', + }, + }); + + return { count }; + } +} diff --git a/apps/api/src/findings/dto/create-finding.dto.ts b/apps/api/src/findings/dto/create-finding.dto.ts index ada5d1ee9..c759a68a8 100644 --- a/apps/api/src/findings/dto/create-finding.dto.ts +++ b/apps/api/src/findings/dto/create-finding.dto.ts @@ -12,10 +12,20 @@ export class CreateFindingDto { @ApiProperty({ description: 'Task ID this finding is associated with', example: 'tsk_abc123', + required: false, }) @IsString() - @IsNotEmpty() - taskId: string; + @IsOptional() + taskId?: string; + + @ApiProperty({ + description: 'Evidence submission ID this finding is associated with', + example: 'evs_abc123', + required: false, + }) + @IsString() + @IsOptional() + evidenceSubmissionId?: string; @ApiProperty({ description: 'Type of finding (SOC 2 or ISO 27001)', diff --git a/apps/api/src/findings/finding-audit.service.ts b/apps/api/src/findings/finding-audit.service.ts index d7698cead..19aa45966 100644 --- a/apps/api/src/findings/finding-audit.service.ts +++ b/apps/api/src/findings/finding-audit.service.ts @@ -17,8 +17,10 @@ export class FindingAuditService { */ async logFindingCreated( params: FindingAuditParams & { - taskId: string; - taskTitle: string; + taskId?: string; + taskTitle?: string; + evidenceSubmissionId?: string; + evidenceSubmissionFormType?: string; content: string; type: FindingType; }, @@ -37,6 +39,8 @@ export class FindingAuditService { findingId: params.findingId, taskId: params.taskId, taskTitle: params.taskTitle, + evidenceSubmissionId: params.evidenceSubmissionId, + evidenceSubmissionFormType: params.evidenceSubmissionFormType, content: params.content, type: params.type, status: FindingStatus.open, @@ -147,8 +151,10 @@ export class FindingAuditService { */ async logFindingDeleted( params: FindingAuditParams & { - taskId: string; - taskTitle: string; + taskId?: string; + taskTitle?: string; + evidenceSubmissionId?: string; + evidenceSubmissionFormType?: string; content: string; }, ): Promise { @@ -166,6 +172,8 @@ export class FindingAuditService { findingId: params.findingId, taskId: params.taskId, taskTitle: params.taskTitle, + evidenceSubmissionId: params.evidenceSubmissionId, + evidenceSubmissionFormType: params.evidenceSubmissionFormType, content: params.content, }, }, diff --git a/apps/api/src/findings/finding-notifier.service.ts b/apps/api/src/findings/finding-notifier.service.ts index d4ea71b8e..b22ca0175 100644 --- a/apps/api/src/findings/finding-notifier.service.ts +++ b/apps/api/src/findings/finding-notifier.service.ts @@ -32,8 +32,11 @@ interface Recipient { interface NotificationParams { organizationId: string; findingId: string; - taskId: string; - taskTitle: string; + taskId?: string; + taskTitle?: string; + evidenceSubmissionId?: string; + evidenceSubmissionFormType?: string; + evidenceSubmissionSubmittedById?: string | null; findingContent: string; findingType: FindingType; actorUserId: string; @@ -84,6 +87,19 @@ function getAppUrl(): string { ); } +function getDocumentContextTitle( + formType?: string, + evidenceSubmissionId?: string, +): string { + if (formType) { + return `Document submission (${formType})`; + } + if (evidenceSubmissionId) { + return `Document submission (${evidenceSubmissionId})`; + } + return 'Document submission'; +} + // ============================================================================ // Service // ============================================================================ @@ -107,29 +123,40 @@ export class FindingNotifierService { organizationId, taskId, taskTitle, + evidenceSubmissionId, + evidenceSubmissionFormType, + evidenceSubmissionSubmittedById, findingType, actorUserId, actorName, } = params; - const recipients = await this.getTaskAssigneeAndAdmins( - organizationId, - taskId, - actorUserId, - ); + const recipients = taskId + ? await this.getTaskAssigneeAndAdmins(organizationId, taskId, actorUserId) + : await this.getSubmissionSubmitterAndAdmins( + organizationId, + evidenceSubmissionId, + evidenceSubmissionSubmittedById, + actorUserId, + ); if (recipients.length === 0) { this.logger.log('No recipients for finding created notification'); return; } + const contextTitle = + taskTitle ?? + getDocumentContextTitle(evidenceSubmissionFormType, evidenceSubmissionId); + const contextLabel = taskId ? 'task' : 'document submission'; + await this.sendNotifications({ ...params, action: 'created', recipients, - subject: `New finding on task: ${taskTitle}`, + subject: `New finding on ${contextLabel}: ${contextTitle}`, heading: 'New Finding Created', - message: `${actorName} created a new ${TYPE_LABELS[findingType]} finding on the task "${taskTitle}".`, + message: `${actorName} created a new ${TYPE_LABELS[findingType]} finding on the ${contextLabel} "${contextTitle}".`, }); } @@ -143,6 +170,8 @@ export class FindingNotifierService { const { findingId, taskTitle, + evidenceSubmissionId, + evidenceSubmissionFormType, actorUserId, actorName, findingCreatorMemberId, @@ -168,13 +197,17 @@ export class FindingNotifierService { `[notifyReadyForReview] Finding ${findingId}: Sending to ${recipients.length} recipient(s): ${recipients.map((r) => r.email).join(', ')}`, ); + const contextTitle = + taskTitle ?? + getDocumentContextTitle(evidenceSubmissionFormType, evidenceSubmissionId); + await this.sendNotifications({ ...params, action: 'ready_for_review', recipients, - subject: `Finding ready for review: ${taskTitle}`, + subject: `Finding ready for review: ${contextTitle}`, heading: 'Finding Ready for Review', - message: `${actorName} marked a finding on "${taskTitle}" as ready for your review.`, + message: `${actorName} marked a finding on "${contextTitle}" as ready for your review.`, newStatus: FindingStatus.ready_for_review, }); } @@ -184,27 +217,42 @@ export class FindingNotifierService { * Recipients: Task assignee + Organization admins/owners */ async notifyNeedsRevision(params: NotificationParams): Promise { - const { organizationId, taskId, taskTitle, actorUserId, actorName } = - params; - - const recipients = await this.getTaskAssigneeAndAdmins( + const { organizationId, taskId, + taskTitle, + evidenceSubmissionId, + evidenceSubmissionFormType, + evidenceSubmissionSubmittedById, actorUserId, - ); + actorName, + } = params; + + const recipients = taskId + ? await this.getTaskAssigneeAndAdmins(organizationId, taskId, actorUserId) + : await this.getSubmissionSubmitterAndAdmins( + organizationId, + evidenceSubmissionId, + evidenceSubmissionSubmittedById, + actorUserId, + ); if (recipients.length === 0) { this.logger.log('No recipients for needs revision notification'); return; } + const contextTitle = + taskTitle ?? + getDocumentContextTitle(evidenceSubmissionFormType, evidenceSubmissionId); + await this.sendNotifications({ ...params, action: 'needs_revision', recipients, - subject: `Finding needs revision: ${taskTitle}`, + subject: `Finding needs revision: ${contextTitle}`, heading: 'Finding Needs Revision', - message: `${actorName} reviewed a finding on "${taskTitle}" and marked it as needing revision.`, + message: `${actorName} reviewed a finding on "${contextTitle}" and marked it as needing revision.`, newStatus: FindingStatus.needs_revision, }); } @@ -214,27 +262,42 @@ export class FindingNotifierService { * Recipients: Task assignee + Organization admins/owners */ async notifyFindingClosed(params: NotificationParams): Promise { - const { organizationId, taskId, taskTitle, actorUserId, actorName } = - params; - - const recipients = await this.getTaskAssigneeAndAdmins( + const { organizationId, taskId, + taskTitle, + evidenceSubmissionId, + evidenceSubmissionFormType, + evidenceSubmissionSubmittedById, actorUserId, - ); + actorName, + } = params; + + const recipients = taskId + ? await this.getTaskAssigneeAndAdmins(organizationId, taskId, actorUserId) + : await this.getSubmissionSubmitterAndAdmins( + organizationId, + evidenceSubmissionId, + evidenceSubmissionSubmittedById, + actorUserId, + ); if (recipients.length === 0) { this.logger.log('No recipients for finding closed notification'); return; } + const contextTitle = + taskTitle ?? + getDocumentContextTitle(evidenceSubmissionFormType, evidenceSubmissionId); + await this.sendNotifications({ ...params, action: 'closed', recipients, - subject: `Finding closed: ${taskTitle}`, + subject: `Finding closed: ${contextTitle}`, heading: 'Finding Closed', - message: `${actorName} closed a finding on "${taskTitle}". The issue has been resolved.`, + message: `${actorName} closed a finding on "${contextTitle}". The issue has been resolved.`, newStatus: FindingStatus.closed, }); } @@ -255,6 +318,8 @@ export class FindingNotifierService { findingId, taskId, taskTitle, + evidenceSubmissionId, + evidenceSubmissionFormType, findingContent, findingType, action, @@ -272,7 +337,13 @@ export class FindingNotifierService { }); const organizationName = organization?.name ?? 'your organization'; - const findingUrl = `${getAppUrl()}/${organizationId}/tasks/${taskId}`; + const contextTitle = + taskTitle ?? + getDocumentContextTitle(evidenceSubmissionFormType, evidenceSubmissionId); + const findingUrl = + evidenceSubmissionId && evidenceSubmissionFormType + ? `${getAppUrl()}/${organizationId}/documents/${evidenceSubmissionFormType}/submissions/${evidenceSubmissionId}` + : `${getAppUrl()}/${organizationId}/tasks/${taskId}`; const typeLabel = TYPE_LABELS[findingType]; const statusLabel = newStatus ? STATUS_LABELS[newStatus] : undefined; @@ -285,7 +356,7 @@ export class FindingNotifierService { organizationName, findingId, taskId, - taskTitle, + taskTitle: contextTitle, findingContent, findingType: typeLabel, action, @@ -307,7 +378,7 @@ export class FindingNotifierService { organizationId: string; organizationName: string; findingId: string; - taskId: string; + taskId?: string; taskTitle: string; findingContent: string; findingType: string; @@ -460,7 +531,7 @@ export class FindingNotifierService { organizationId: string; organizationName: string; findingId: string; - taskId: string; + taskId?: string; taskTitle: string; findingType: string; findingContent: string; @@ -605,6 +676,103 @@ export class FindingNotifierService { } } + private async getSubmissionSubmitterAndAdmins( + organizationId: string, + evidenceSubmissionId: string | undefined, + submitterUserId: string | null | undefined, + excludeUserId: string, + ): Promise { + try { + const allMembers = await db.member.findMany({ + where: { + organizationId, + deactivated: false, + }, + select: { + role: true, + user: { select: { id: true, email: true, name: true } }, + }, + }); + + const adminMembers = allMembers.filter( + (member) => + member.role.includes('admin') || member.role.includes('owner'), + ); + + const recipients: Recipient[] = []; + const addedUserIds = new Set(); + + if (submitterUserId) { + const submitter = await db.user.findUnique({ + where: { id: submitterUserId }, + select: { id: true, email: true, name: true }, + }); + + if ( + submitter && + submitter.id !== excludeUserId && + submitter.email && + !addedUserIds.has(submitter.id) + ) { + recipients.push({ + userId: submitter.id, + email: submitter.email, + name: submitter.name || submitter.email, + }); + addedUserIds.add(submitter.id); + } + } else if (evidenceSubmissionId) { + const submission = await db.evidenceSubmission.findUnique({ + where: { id: evidenceSubmissionId }, + select: { + submittedBy: { + select: { id: true, email: true, name: true }, + }, + }, + }); + + const submitter = submission?.submittedBy; + if ( + submitter && + submitter.id !== excludeUserId && + submitter.email && + !addedUserIds.has(submitter.id) + ) { + recipients.push({ + userId: submitter.id, + email: submitter.email, + name: submitter.name || submitter.email, + }); + addedUserIds.add(submitter.id); + } + } + + for (const member of adminMembers) { + const user = member.user; + if ( + user.id !== excludeUserId && + user.email && + !addedUserIds.has(user.id) + ) { + recipients.push({ + userId: user.id, + email: user.email, + name: user.name || user.email, + }); + addedUserIds.add(user.id); + } + } + + return recipients; + } catch (error) { + this.logger.error( + 'Failed to get submission recipients:', + error instanceof Error ? error.message : 'Unknown error', + ); + return []; + } + } + /** * Get the finding creator as recipient (for Ready for Review notifications). * Excludes the actor (person who triggered the action). diff --git a/apps/api/src/findings/findings.controller.ts b/apps/api/src/findings/findings.controller.ts index a4006a2dd..937716f50 100644 --- a/apps/api/src/findings/findings.controller.ts +++ b/apps/api/src/findings/findings.controller.ts @@ -54,10 +54,16 @@ export class FindingsController { }) @ApiQuery({ name: 'taskId', - required: true, + required: false, description: 'Task ID to get findings for', example: 'tsk_abc123', }) + @ApiQuery({ + name: 'evidenceSubmissionId', + required: false, + description: 'Evidence submission ID to get findings for', + example: 'evs_abc123', + }) @ApiResponse({ status: 200, description: 'List of findings for the task', @@ -72,14 +78,31 @@ export class FindingsController { }) async getFindingsByTask( @Query('taskId') taskId: string, + @Query('evidenceSubmissionId') evidenceSubmissionId: string, @AuthContext() authContext: AuthContextType, ) { - if (!taskId) { - throw new BadRequestException('taskId query parameter is required'); + if (!taskId && !evidenceSubmissionId) { + throw new BadRequestException( + 'Either taskId or evidenceSubmissionId query parameter is required', + ); + } + + if (taskId && evidenceSubmissionId) { + throw new BadRequestException( + 'Provide only one target: taskId or evidenceSubmissionId', + ); } - return await this.findingsService.findByTaskId( + + if (taskId) { + return await this.findingsService.findByTaskId( + authContext.organizationId, + taskId, + ); + } + + return await this.findingsService.findByEvidenceSubmissionId( authContext.organizationId, - taskId, + evidenceSubmissionId, ); } diff --git a/apps/api/src/findings/findings.service.ts b/apps/api/src/findings/findings.service.ts index 57e6b8f26..8c6f11549 100644 --- a/apps/api/src/findings/findings.service.ts +++ b/apps/api/src/findings/findings.service.ts @@ -14,6 +14,41 @@ import { FindingNotifierService } from './finding-notifier.service'; @Injectable() export class FindingsService { private readonly logger = new Logger(FindingsService.name); + private readonly findingInclude = { + createdBy: { + include: { + user: { + select: { + id: true, + name: true, + email: true, + image: true, + }, + }, + }, + }, + template: { + select: { + id: true, + category: true, + title: true, + }, + }, + task: { + select: { + id: true, + title: true, + }, + }, + evidenceSubmission: { + select: { + id: true, + formType: true, + submittedAt: true, + submittedById: true, + }, + }, + }; constructor( private readonly findingAuditService: FindingAuditService, @@ -37,33 +72,7 @@ export class FindingsService { const findings = await db.finding.findMany({ where: { taskId, organizationId }, - include: { - createdBy: { - include: { - user: { - select: { - id: true, - name: true, - email: true, - image: true, - }, - }, - }, - }, - template: { - select: { - id: true, - category: true, - title: true, - }, - }, - task: { - select: { - id: true, - title: true, - }, - }, - }, + include: this.findingInclude, orderBy: [ // Sort by status: open first, then ready_for_review, needs_revision, closed { status: 'asc' }, @@ -75,6 +84,35 @@ export class FindingsService { return findings; } + /** + * Get all findings for a specific evidence submission + */ + async findByEvidenceSubmissionId( + organizationId: string, + evidenceSubmissionId: string, + ) { + const submission = await db.evidenceSubmission.findFirst({ + where: { id: evidenceSubmissionId, organizationId }, + }); + + if (!submission) { + throw new NotFoundException( + `Evidence submission with ID ${evidenceSubmissionId} not found in organization`, + ); + } + + const findings = await db.finding.findMany({ + where: { evidenceSubmissionId, organizationId }, + include: this.findingInclude, + orderBy: [{ status: 'asc' }, { createdAt: 'desc' }], + }); + + this.logger.log( + `Retrieved ${findings.length} findings for evidence submission ${evidenceSubmissionId}`, + ); + return findings; + } + /** * Get all findings for an organization */ @@ -84,33 +122,7 @@ export class FindingsService { organizationId, ...(status && { status }), }, - include: { - createdBy: { - include: { - user: { - select: { - id: true, - name: true, - email: true, - image: true, - }, - }, - }, - }, - template: { - select: { - id: true, - category: true, - title: true, - }, - }, - task: { - select: { - id: true, - title: true, - }, - }, - }, + include: this.findingInclude, orderBy: [{ status: 'asc' }, { createdAt: 'desc' }], }); @@ -126,33 +138,7 @@ export class FindingsService { async findById(organizationId: string, findingId: string) { const finding = await db.finding.findFirst({ where: { id: findingId, organizationId }, - include: { - createdBy: { - include: { - user: { - select: { - id: true, - name: true, - email: true, - image: true, - }, - }, - }, - }, - template: { - select: { - id: true, - category: true, - title: true, - }, - }, - task: { - select: { - id: true, - title: true, - }, - }, - }, + include: this.findingInclude, }); if (!finding) { @@ -173,16 +159,64 @@ export class FindingsService { userId: string, createDto: CreateFindingDto, ) { - // Verify task belongs to organization - const task = await db.task.findFirst({ - where: { id: createDto.taskId, organizationId }, - }); - - if (!task) { - throw new NotFoundException( - `Task with ID ${createDto.taskId} not found in organization`, + const hasTaskTarget = Boolean(createDto.taskId); + const hasSubmissionTarget = Boolean(createDto.evidenceSubmissionId); + if (!hasTaskTarget && !hasSubmissionTarget) { + throw new BadRequestException( + 'Either taskId or evidenceSubmissionId is required', ); } + if (hasTaskTarget && hasSubmissionTarget) { + throw new BadRequestException( + 'Provide only one target: taskId or evidenceSubmissionId', + ); + } + + let task: + | { + id: string; + title: string; + } + | null = null; + let evidenceSubmission: + | { + id: string; + formType: string; + submittedAt: Date; + submittedById: string | null; + } + | null = null; + + if (createDto.taskId) { + task = await db.task.findFirst({ + where: { id: createDto.taskId, organizationId }, + select: { id: true, title: true }, + }); + + if (!task) { + throw new NotFoundException( + `Task with ID ${createDto.taskId} not found in organization`, + ); + } + } + + if (createDto.evidenceSubmissionId) { + evidenceSubmission = await db.evidenceSubmission.findFirst({ + where: { id: createDto.evidenceSubmissionId, organizationId }, + select: { + id: true, + formType: true, + submittedAt: true, + submittedById: true, + }, + }); + + if (!evidenceSubmission) { + throw new NotFoundException( + `Evidence submission with ID ${createDto.evidenceSubmissionId} not found in organization`, + ); + } + } // Verify template exists if provided if (createDto.templateId) { @@ -199,7 +233,8 @@ export class FindingsService { const finding = await db.finding.create({ data: { - taskId: createDto.taskId, + taskId: createDto.taskId ?? null, + evidenceSubmissionId: createDto.evidenceSubmissionId ?? null, type: createDto.type, content: createDto.content, templateId: createDto.templateId, @@ -207,33 +242,7 @@ export class FindingsService { organizationId, status: FindingStatus.open, }, - include: { - createdBy: { - include: { - user: { - select: { - id: true, - name: true, - email: true, - image: true, - }, - }, - }, - }, - template: { - select: { - id: true, - category: true, - title: true, - }, - }, - task: { - select: { - id: true, - title: true, - }, - }, - }, + include: this.findingInclude, }); // Log to audit trail @@ -242,8 +251,10 @@ export class FindingsService { organizationId, userId, memberId, - taskId: createDto.taskId, - taskTitle: task.title, + taskId: task?.id, + taskTitle: task?.title, + evidenceSubmissionId: evidenceSubmission?.id, + evidenceSubmissionFormType: evidenceSubmission?.formType, content: createDto.content, type: createDto.type ?? FindingType.soc2, }); @@ -256,17 +267,21 @@ export class FindingsService { void this.findingNotifierService.notifyFindingCreated({ organizationId, findingId: finding.id, - taskId: createDto.taskId, - taskTitle: task.title, + taskId: task?.id, + taskTitle: task?.title, + evidenceSubmissionId: evidenceSubmission?.id, + evidenceSubmissionFormType: evidenceSubmission?.formType, + evidenceSubmissionSubmittedById: evidenceSubmission?.submittedById, findingContent: createDto.content, findingType: createDto.type ?? FindingType.soc2, actorUserId: userId, actorName, }); - this.logger.log( - `Created finding ${finding.id} for task ${createDto.taskId}`, - ); + const target = task + ? `task ${task.id}` + : `evidence submission ${evidenceSubmission?.id}`; + this.logger.log(`Created finding ${finding.id} for ${target}`); return finding; } @@ -418,8 +433,12 @@ export class FindingsService { const notificationParams = { organizationId, findingId, - taskId: finding.taskId, - taskTitle: finding.task.title, + taskId: finding.task?.id, + taskTitle: finding.task?.title, + evidenceSubmissionId: finding.evidenceSubmission?.id, + evidenceSubmissionFormType: finding.evidenceSubmission?.formType, + evidenceSubmissionSubmittedById: + finding.evidenceSubmission?.submittedById, findingContent: updatedFinding.content, findingType: updatedFinding.type, actorUserId: userId, @@ -496,17 +515,20 @@ export class FindingsService { organizationId, userId, memberId, - taskId: finding.taskId, - taskTitle: finding.task.title, + taskId: finding.task?.id, + taskTitle: finding.task?.title, + evidenceSubmissionId: finding.evidenceSubmission?.id, + evidenceSubmissionFormType: finding.evidenceSubmission?.formType, content: finding.content, }); - this.logger.log(`Deleted finding ${findingId} from task ${finding.taskId}`); + this.logger.log(`Deleted finding ${findingId}`); return { message: 'Finding deleted successfully', deletedFinding: { id: finding.id, taskId: finding.taskId, + evidenceSubmissionId: finding.evidenceSubmissionId, }, }; } diff --git a/apps/api/src/tasks/task-notifier.service.ts b/apps/api/src/tasks/task-notifier.service.ts index 180b6733c..b8fae1b90 100644 --- a/apps/api/src/tasks/task-notifier.service.ts +++ b/apps/api/src/tasks/task-notifier.service.ts @@ -624,45 +624,49 @@ export class TaskNotifierService { } = params; try { - const [organization, changedByUser, oldAssigneeMember, newAssigneeMember] = - await Promise.all([ - db.organization.findUnique({ - where: { id: organizationId }, - select: { name: true }, - }), - db.user.findUnique({ - where: { id: changedByUserId }, - select: { name: true, email: true }, - }), - oldAssigneeId - ? db.member.findUnique({ - where: { id: oldAssigneeId }, - select: { - user: { - select: { - id: true, - name: true, - email: true, - }, + const [ + organization, + changedByUser, + oldAssigneeMember, + newAssigneeMember, + ] = await Promise.all([ + db.organization.findUnique({ + where: { id: organizationId }, + select: { name: true }, + }), + db.user.findUnique({ + where: { id: changedByUserId }, + select: { name: true, email: true }, + }), + oldAssigneeId + ? db.member.findUnique({ + where: { id: oldAssigneeId }, + select: { + user: { + select: { + id: true, + name: true, + email: true, }, }, - }) - : Promise.resolve(null), - newAssigneeId - ? db.member.findUnique({ - where: { id: newAssigneeId }, - select: { - user: { - select: { - id: true, - name: true, - email: true, - }, + }, + }) + : Promise.resolve(null), + newAssigneeId + ? db.member.findUnique({ + where: { id: newAssigneeId }, + select: { + user: { + select: { + id: true, + name: true, + email: true, }, }, - }) - : Promise.resolve(null), - ]); + }, + }) + : Promise.resolve(null), + ]); const organizationName = organization?.name ?? 'your organization'; const changedByName = @@ -792,31 +796,39 @@ export class TaskNotifierService { submittedByUserId: string; approverMemberId: string; }): Promise { - const { organizationId, taskId, taskTitle, submittedByUserId, approverMemberId } = params; + const { + organizationId, + taskId, + taskTitle, + submittedByUserId, + approverMemberId, + } = params; try { - const [organization, submittedByUser, approverMember] = await Promise.all([ - db.organization.findUnique({ - where: { id: organizationId }, - select: { name: true }, - }), - db.user.findUnique({ - where: { id: submittedByUserId }, - select: { name: true, email: true }, - }), - db.member.findUnique({ - where: { id: approverMemberId }, - select: { - user: { - select: { - id: true, - name: true, - email: true, + const [organization, submittedByUser, approverMember] = await Promise.all( + [ + db.organization.findUnique({ + where: { id: organizationId }, + select: { name: true }, + }), + db.user.findUnique({ + where: { id: submittedByUserId }, + select: { name: true, email: true }, + }), + db.member.findUnique({ + where: { id: approverMemberId }, + select: { + user: { + select: { + id: true, + name: true, + email: true, + }, }, }, - }, - }), - ]); + }), + ], + ); const organizationName = organization?.name ?? 'your organization'; const submittedByName = @@ -825,7 +837,9 @@ export class TaskNotifierService { 'Someone'; if (!approverMember?.user?.id || !approverMember.user.email) { - this.logger.warn('Approver not found, skipping review request notification'); + this.logger.warn( + 'Approver not found, skipping review request notification', + ); return; } @@ -836,7 +850,10 @@ export class TaskNotifierService { const recipient = { id: approverMember.user.id, - name: approverMember.user.name?.trim() || approverMember.user.email?.trim() || 'User', + name: + approverMember.user.name?.trim() || + approverMember.user.email?.trim() || + 'User', email: approverMember.user.email, }; @@ -925,41 +942,48 @@ export class TaskNotifierService { submittedByUserId: string; approverMemberId: string; }): Promise { - const { organizationId, taskIds, taskCount, submittedByUserId, approverMemberId } = params; + const { + organizationId, + taskIds, + taskCount, + submittedByUserId, + approverMemberId, + } = params; try { - const [organization, submittedByUser, approverMember, tasks] = await Promise.all([ - db.organization.findUnique({ - where: { id: organizationId }, - select: { name: true }, - }), - db.user.findUnique({ - where: { id: submittedByUserId }, - select: { name: true, email: true }, - }), - db.member.findUnique({ - where: { id: approverMemberId }, - select: { - user: { - select: { - id: true, - name: true, - email: true, + const [organization, submittedByUser, approverMember, tasks] = + await Promise.all([ + db.organization.findUnique({ + where: { id: organizationId }, + select: { name: true }, + }), + db.user.findUnique({ + where: { id: submittedByUserId }, + select: { name: true, email: true }, + }), + db.member.findUnique({ + where: { id: approverMemberId }, + select: { + user: { + select: { + id: true, + name: true, + email: true, + }, }, }, - }, - }), - db.task.findMany({ - where: { - id: { in: taskIds }, - organizationId, - }, - select: { - id: true, - title: true, - }, - }), - ]); + }), + db.task.findMany({ + where: { + id: { in: taskIds }, + organizationId, + }, + select: { + id: true, + title: true, + }, + }), + ]); const organizationName = organization?.name ?? 'your organization'; const submittedByName = @@ -968,7 +992,9 @@ export class TaskNotifierService { 'Someone'; if (!approverMember?.user?.id || !approverMember.user.email) { - this.logger.warn('Approver not found, skipping bulk review notification'); + this.logger.warn( + 'Approver not found, skipping bulk review notification', + ); return; } @@ -979,7 +1005,10 @@ export class TaskNotifierService { const recipient = { id: approverMember.user.id, - name: approverMember.user.name?.trim() || approverMember.user.email?.trim() || 'User', + name: + approverMember.user.name?.trim() || + approverMember.user.email?.trim() || + 'User', email: approverMember.user.email, }; diff --git a/apps/api/src/tasks/tasks.controller.ts b/apps/api/src/tasks/tasks.controller.ts index 92bfa929a..e14eb0e0b 100644 --- a/apps/api/src/tasks/tasks.controller.ts +++ b/apps/api/src/tasks/tasks.controller.ts @@ -408,7 +408,8 @@ export class TasksController { @Get(':taskId/activity') @ApiOperation({ summary: 'Get task activity', - description: 'Retrieve audit log activity for a specific task with pagination', + description: + 'Retrieve audit log activity for a specific task with pagination', }) @ApiParam({ name: 'taskId', @@ -424,8 +425,15 @@ export class TasksController { @Query('take') take?: string, ) { const parsedSkip = skip ? Math.max(0, parseInt(skip, 10) || 0) : 0; - const parsedTake = take ? Math.min(50, Math.max(1, parseInt(take, 10) || 10)) : 10; - return await this.tasksService.getTaskActivity(organizationId, taskId, parsedSkip, parsedTake); + const parsedTake = take + ? Math.min(50, Math.max(1, parseInt(take, 10) || 10)) + : 10; + return await this.tasksService.getTaskActivity( + organizationId, + taskId, + parsedSkip, + parsedTake, + ); } @Patch(':taskId') @@ -552,8 +560,7 @@ export class TasksController { @Post(':taskId/submit-for-review') @ApiOperation({ summary: 'Submit task for review', - description: - 'Move task status to in_review and assign an approver.', + description: 'Move task status to in_review and assign an approver.', }) @ApiParam({ name: 'taskId', diff --git a/apps/api/src/tasks/tasks.service.ts b/apps/api/src/tasks/tasks.service.ts index ee5403c07..88d799c41 100644 --- a/apps/api/src/tasks/tasks.service.ts +++ b/apps/api/src/tasks/tasks.service.ts @@ -571,7 +571,10 @@ export class TasksService { approverMemberId: approverId, }) .catch((error) => { - console.error('Failed to send evidence review request notifications:', error); + console.error( + 'Failed to send evidence review request notifications:', + error, + ); }); return updatedTask; @@ -652,7 +655,10 @@ export class TasksService { approverMemberId: approverId, }) .catch((error) => { - console.error('Failed to send bulk evidence review request notifications:', error); + console.error( + 'Failed to send bulk evidence review request notifications:', + error, + ); }); return { submittedCount: tasks.length }; @@ -772,9 +778,8 @@ export class TasksService { throw new ForbiddenException('User is not a member of this organization'); } - const memberRoles = currentMember.role - ?.split(',') - .map((r: string) => r.trim()) ?? []; + const memberRoles = + currentMember.role?.split(',').map((r: string) => r.trim()) ?? []; const isAdminOrOwner = memberRoles.includes('admin') || memberRoles.includes('owner'); const isApprover = task.approverId === currentMember.id; @@ -801,7 +806,7 @@ export class TasksService { }); const assigneeName = task.assignee - ? (task.assignee.user.name || task.assignee.user.email) + ? task.assignee.user.name || task.assignee.user.email : 'Unknown'; await tx.auditLog.create({ diff --git a/apps/api/src/trust-portal/badge-svgs-new.ts b/apps/api/src/trust-portal/badge-svgs-new.ts index 7907c50df..e94a41f48 100644 --- a/apps/api/src/trust-portal/badge-svgs-new.ts +++ b/apps/api/src/trust-portal/badge-svgs-new.ts @@ -31,4 +31,4 @@ export const BADGE_ICON_MAP: Record = { icon: 'data:image/svg+xml;base64,<svg
    width="168"
    height="210"
    viewBox="0 0 168 210"
    fill="none"
    xmlns="http://www.w3.org/2000/svg"
  >
    <path
      d="M157.227 2.80371H10.478C6.348 2.80371 3 6.15161 3 10.2814V160.436C3 163.226 4.553 165.783 7.028 167.07L81.297 205.69C83.482 206.827 86.086 206.814 88.26 205.657L160.74 167.086C163.18 165.787 164.705 163.249 164.705 160.485V10.2814C164.705 6.15161 161.357 2.80371 157.227 2.80371Z"
      fill="white"
    />
    <path
      d="M161.705 10.2818C161.705 7.7007 159.613 5.6083 157.032 5.6083H10.282C7.701 5.6083 5.608 7.7007 5.608 10.2818V160.436L5.619 160.762C5.732 162.377 6.676 163.829 8.126 164.583L82.395 203.203L82.654 203.328C83.963 203.91 85.473 203.86 86.747 203.183L159.227 164.611L159.506 164.45C160.867 163.6 161.705 162.105 161.705 160.485V10.2818ZM167.313 160.485C167.313 164.167 165.346 167.558 162.172 169.39L161.862 169.561L89.382 208.133C86.486 209.674 83.035 209.738 80.091 208.321L79.808 208.178L5.538 169.559C2.241 167.844 0.135 164.49 0.007 160.795L0 160.436V10.2818C0 4.6033 4.604 0 10.282 0H157.032C162.71 0 167.313 4.6033 167.313 10.2818V160.485Z"
      fill="#16171B"
    />
    <path
      d="M103.489 5.6084H63.2969V36.249L83.6249 46.7357L103.489 36.249V5.6084Z"
      fill="#16171B"
    />
    <path
      d="M89.234 17.083L88.501 16.5803L83.394 13.085L71.708 21.081V25.6512L83.393 33.6487L95.076 25.6512V21.081L89.233 17.083H89.234ZM83.394 15.0988L87.031 17.5887L85.163 18.8662L84.997 18.98L83.396 17.8844L78.721 21.0825L80.323 22.1796L81.793 23.1865L83.394 24.2836L88.067 21.084L86.467 19.9884L86.633 19.8745L88.501 18.5971L92.136 21.0855L90.268 22.3644L83.394 27.0691L79.757 24.5793L78.287 23.5739L76.518 22.3644L74.651 21.0855L83.394 15.0988Z"
      fill="white"
    />
    <path
      d="M51.8604 85.885V57.6232H58.7864V85.885H51.8604ZM72.7964 86.5218C69.0284 86.5218 66.0834 85.6196 63.9604 83.8151C61.8374 81.984 60.6694 79.5426 60.4574 76.4909L67.3834 76.1724C67.5954 77.7116 68.1394 78.8925 69.0154 79.7151C69.9174 80.5378 71.2044 80.9491 72.8764 80.9491C74.2294 80.9491 75.2644 80.7103 75.9814 80.2326C76.7244 79.7549 77.0954 79.0384 77.0954 78.0831C77.0954 77.4993 76.9494 76.9951 76.6584 76.5705C76.3924 76.1194 75.8484 75.7213 75.0254 75.3763C74.2294 75.0048 73.0224 74.6466 71.4034 74.3016C68.9094 73.7709 66.8924 73.1605 65.3534 72.4705C63.8404 71.754 62.7264 70.8651 62.0094 69.8036C61.3194 68.7156 60.9744 67.3357 60.9744 65.6638C60.9744 63.9389 61.4124 62.4263 62.2884 61.126C63.1634 59.8257 64.4244 58.8173 66.0694 58.1008C67.7154 57.3578 69.6784 56.9863 71.9604 56.9863C74.4024 56.9863 76.4454 57.4109 78.0904 58.26C79.7624 59.0827 81.0494 60.2238 81.9524 61.6833C82.8804 63.1428 83.4514 64.8279 83.6634 66.7386L76.8174 67.057C76.6844 65.6771 76.1934 64.5891 75.3444 63.793C74.4954 62.9703 73.3404 62.559 71.8814 62.559C70.6604 62.559 69.6924 62.8244 68.9754 63.3551C68.2854 63.8859 67.9404 64.5758 67.9404 65.425C67.9404 66.4069 68.3124 67.1499 69.0554 67.6541C69.8244 68.1583 71.2444 68.6094 73.3144 69.0075C76.0214 69.5117 78.1574 70.1751 79.7234 70.9977C81.2884 71.7939 82.4034 72.7625 83.0664 73.9035C83.7304 75.0181 84.0614 76.3051 84.0614 77.7647C84.0614 80.4714 83.0534 82.6076 81.0364 84.1733C79.0464 85.739 76.2994 86.5218 72.7964 86.5218ZM99.3994 86.5218C96.4804 86.5218 93.9724 85.9248 91.8764 84.7306C89.7794 83.5364 88.1604 81.8381 87.0194 79.6355C85.9054 77.4329 85.3484 74.8191 85.3484 71.7939C85.3484 68.7687 85.9054 66.1548 87.0194 63.9522C88.1604 61.7231 89.7794 60.0115 91.8764 58.8173C93.9724 57.5966 96.4804 56.9863 99.3994 56.9863C102.318 56.9863 104.826 57.5966 106.922 58.8173C109.018 60.0115 110.624 61.7231 111.738 63.9522C112.88 66.1548 113.45 68.7687 113.45 71.7939C113.45 74.8191 112.88 77.4329 111.738 79.6355C110.624 81.8381 109.018 83.5364 106.922 84.7306C104.826 85.9248 102.318 86.5218 99.3994 86.5218ZM99.3994 80.8695C101.575 80.8695 103.273 80.0866 104.494 78.521C105.715 76.9287 106.325 74.6864 106.325 71.7939C106.325 68.9013 105.715 66.659 104.494 65.0668C103.273 63.448 101.575 62.6386 99.3994 62.6386C97.2234 62.6386 95.5244 63.448 94.3044 65.0668C93.0834 66.659 92.4734 68.9013 92.4734 71.7939C92.4734 74.6864 93.0834 76.9287 94.3044 78.521C95.5244 80.0866 97.2234 80.8695 99.3994 80.8695Z"
      fill="#16171B"
    />
    <path
      d="M45.7547 94.1157C47.365 94.1157 48.8899 94.3721 50.3293 94.8849C51.7687 95.3886 53.0416 96.1848 54.1482 97.2733C55.2637 98.3619 56.1363 99.7833 56.7661 101.538C57.4048 103.283 57.7287 105.397 57.7377 107.88C57.7467 110.147 57.4633 112.185 56.8875 113.993C56.3208 115.792 55.5066 117.326 54.445 118.594C53.3925 119.863 52.124 120.834 50.6396 121.509C49.1643 122.175 47.518 122.508 45.7007 122.508C43.6586 122.508 41.8638 122.116 40.3165 121.334C38.7691 120.551 37.5367 119.498 36.6191 118.176C35.7014 116.854 35.1662 115.383 35.0132 113.763H42.4081C42.588 114.6 42.9839 115.207 43.5956 115.585C44.2074 115.954 44.9091 116.138 45.7007 116.138C47.2841 116.138 48.4401 115.455 49.1688 114.087C49.9065 112.711 50.2798 110.876 50.2888 108.582H50.1269C49.776 109.427 49.2452 110.151 48.5345 110.754C47.8238 111.357 46.9962 111.82 46.0516 112.144C45.107 112.468 44.1084 112.63 43.0558 112.63C41.3915 112.63 39.9342 112.257 38.6837 111.51C37.4332 110.754 36.4571 109.724 35.7554 108.42C35.0537 107.106 34.6984 105.613 34.6894 103.94C34.6804 101.978 35.1437 100.26 36.0793 98.7847C37.0149 97.3093 38.3148 96.1623 39.9791 95.3437C41.6434 94.525 43.5686 94.1157 45.7547 94.1157ZM45.8087 99.7833C45.062 99.7833 44.3963 99.9542 43.8115 100.296C43.2358 100.638 42.7815 101.101 42.4486 101.686C42.1247 102.271 41.9673 102.932 41.9763 103.67C41.9853 104.407 42.1517 105.069 42.4756 105.653C42.8084 106.238 43.2583 106.701 43.825 107.043C44.4008 107.385 45.062 107.556 45.8087 107.556C46.3575 107.556 46.8657 107.462 47.3335 107.273C47.8013 107.075 48.2062 106.8 48.548 106.449C48.8989 106.09 49.1688 105.676 49.3577 105.208C49.5556 104.731 49.6501 104.218 49.6411 103.67C49.6321 102.932 49.4611 102.271 49.1283 101.686C48.7954 101.101 48.3411 100.638 47.7654 100.296C47.1896 99.9542 46.5374 99.7833 45.8087 99.7833ZM73.0099 122.886C70.4999 122.877 68.3318 122.296 66.5056 121.145C64.6794 119.993 63.2715 118.333 62.2819 116.165C61.2923 113.997 60.802 111.397 60.811 108.366C60.82 105.325 61.3148 102.743 62.2954 100.62C63.2849 98.4968 64.6884 96.882 66.5056 95.7755C68.3318 94.6689 70.4999 94.1157 73.0099 94.1157C75.5198 94.1157 77.6879 94.6734 79.5141 95.789C81.3403 96.8955 82.7483 98.5103 83.7378 100.633C84.7274 102.757 85.2177 105.334 85.2087 108.366C85.2087 111.415 84.7139 114.024 83.7243 116.192C82.7348 118.36 81.3269 120.02 79.5006 121.172C77.6834 122.314 75.5198 122.886 73.0099 122.886ZM73.0099 116.786C74.3053 116.786 75.3669 116.111 76.1945 114.762C77.0312 113.404 77.445 111.271 77.436 108.366C77.436 106.467 77.2471 104.916 76.8692 103.71C76.4914 102.505 75.9696 101.614 75.3039 101.038C74.6382 100.454 73.8735 100.161 73.0099 100.161C71.7144 100.161 70.6573 100.818 69.8387 102.131C69.02 103.445 68.6017 105.523 68.5837 108.366C68.5747 110.3 68.7591 111.892 69.137 113.143C69.5148 114.384 70.0366 115.302 70.7023 115.896C71.377 116.489 72.1462 116.786 73.0099 116.786ZM100.768 122.886C98.2577 122.877 96.0896 122.296 94.2634 121.145C92.4372 119.993 91.0293 118.333 90.0397 116.165C89.0501 113.997 88.5598 111.397 88.5688 108.366C88.5778 105.325 89.0726 102.743 90.0532 100.62C91.0428 98.4968 92.4462 96.882 94.2634 95.7755C96.0896 94.6689 98.2577 94.1157 100.768 94.1157C103.278 94.1157 105.446 94.6734 107.272 95.789C109.098 96.8955 110.506 98.5103 111.496 100.633C112.485 102.757 112.976 105.334 112.967 108.366C112.967 111.415 112.472 114.024 111.482 116.192C110.493 118.36 109.085 120.02 107.258 121.172C105.441 122.314 103.278 122.886 100.768 122.886ZM100.768 116.786C102.063 116.786 103.125 116.111 103.952 114.762C104.789 113.404 105.203 111.271 105.194 108.366C105.194 106.467 105.005 104.916 104.627 103.71C104.249 102.505 103.727 101.614 103.062 101.038C102.396 100.454 101.631 100.161 100.768 100.161C99.4722 100.161 98.4152 100.818 97.5965 102.131C96.7778 103.445 96.3595 105.523 96.3415 108.366C96.3325 110.3 96.517 111.892 96.8948 113.143C97.2726 114.384 97.7944 115.302 98.4601 115.896C99.1349 116.489 99.904 116.786 100.768 116.786ZM130.037 94.4935V122.13H122.534V101.403H122.372L116.327 105.019V98.6498L123.128 94.4935H130.037Z"
      fill="#16171B"
    />
    <path
      d="M45.5167 141.11C44.7307 141.11 44.0337 140.932 43.4267 140.576C42.8197 140.219 42.3417 139.705 41.9937 139.033C41.6447 138.36 41.4707 137.555 41.4707 136.615C41.4707 135.7 41.6367 134.906 41.9687 134.234C42.3087 133.554 42.7827 133.027 43.3907 132.655C44.0057 132.282 44.7267 132.096 45.5527 132.096C46.6787 132.096 47.5537 132.375 48.1767 132.934C48.8007 133.485 49.2057 134.274 49.3917 135.303L47.4727 135.376C47.3747 134.833 47.1647 134.412 46.8407 134.112C46.5167 133.805 46.0877 133.651 45.5527 133.651C44.8647 133.651 44.3297 133.922 43.9487 134.465C43.5687 134.999 43.3777 135.716 43.3777 136.615C43.3777 137.522 43.5727 138.239 43.9617 138.765C44.3497 139.292 44.8767 139.555 45.5407 139.555C46.1157 139.555 46.5647 139.393 46.8887 139.069C47.2127 138.737 47.4157 138.279 47.4967 137.696L49.4277 137.769C49.2497 138.822 48.8327 139.644 48.1767 140.235C47.5287 140.818 46.6417 141.11 45.5167 141.11ZM53.0807 140.916V132.29H59.0817V133.845H54.9267V135.825H58.9357V137.356H54.9267V139.361H59.1787V140.916H53.0807ZM63.0317 140.916V132.29H66.8107C67.4417 132.29 67.9887 132.395 68.4507 132.606C68.9117 132.817 69.2687 133.116 69.5197 133.505C69.7787 133.886 69.9077 134.339 69.9077 134.866C69.9077 135.392 69.7667 135.838 69.4827 136.202C69.2077 136.566 68.8427 136.818 68.3897 136.955C69.2397 137.077 69.7017 137.571 69.7747 138.437L69.9927 140.916H68.1097L67.9517 138.729C67.9277 138.397 67.8227 138.154 67.6367 138C67.4577 137.838 67.1507 137.757 66.7127 137.757H64.8787V140.916H63.0317ZM64.8787 136.214H66.6037C67.0487 136.214 67.3937 136.113 67.6367 135.91C67.8877 135.7 68.0127 135.404 68.0127 135.024C68.0127 134.635 67.8877 134.343 67.6367 134.149C67.3857 133.946 67.0207 133.845 66.5427 133.845H64.8787V136.214ZM75.5287 140.916V133.845H72.9537V132.29H79.9757V133.845H77.3877V140.916H75.5287ZM83.2077 140.916V132.29H85.0547V140.916H83.2077ZM89.1137 140.916V132.29H95.0667V133.845H90.9607V135.935H94.8477V137.465H90.9607V140.916H89.1137ZM98.7097 140.916V132.29H100.556V140.916H98.7097ZM104.615 140.916V132.29H110.617V133.845H106.462V135.825H110.471V137.356H106.462V139.361H110.714V140.916H104.615ZM114.567 140.916V132.29H117.543C118.936 132.29 120.009 132.667 120.763 133.42C121.524 134.165 121.905 135.23 121.905 136.615C121.905 137.992 121.532 139.053 120.787 139.798C120.042 140.543 118.985 140.916 117.616 140.916H114.567ZM116.413 139.361H117.543C118.378 139.361 118.993 139.138 119.39 138.692C119.795 138.239 119.997 137.542 119.997 136.603C119.997 135.671 119.795 134.979 119.39 134.525C118.993 134.072 118.378 133.845 117.543 133.845H116.413V139.361Z"
      fill="#16171B"
    />
    <path
      d="M59.5575 5.14028L56.5195 4.91588V3.73828H66.8015V11.9814H64.0965C64.0965 12.3739 63.7635 12.1764 63.7635 8.87918C63.7635 5.58188 61.1815 5.14028 59.5575 5.14028Z"
      fill="#16171B"
    />
    <path
      d="M56.5188 4.91551H56.0518V5.34961L56.4848 5.38161L56.5188 4.91551ZM59.5568 5.13991L59.5228 5.60601L59.5398 5.60731H59.5568V5.13991ZM64.0958 11.981V11.5136H63.6278V11.981H64.0958ZM66.8008 11.981V12.4484H67.2688V11.981H66.8008ZM66.8008 3.73791H67.2688V3.27051H66.8008V3.73791ZM56.5188 3.73791V3.27051H56.0518V3.73791H56.5188ZM56.5188 4.91551L56.4848 5.38161L59.5228 5.60601L59.5568 5.13991L59.5918 4.67391L56.5538 4.44941L56.5188 4.91551ZM59.5568 5.13991V5.60731C60.3418 5.60731 61.2868 5.71721 62.0238 6.17091C62.7208 6.60011 63.2958 7.37611 63.2958 8.87881H63.7628H64.2308C64.2308 7.08421 63.5138 5.99081 62.5138 5.37491C61.5538 4.78351 60.3958 4.67261 59.5568 4.67261V5.13991ZM63.7628 8.87881H63.2958C63.2958 10.5335 63.3788 11.4456 63.4708 11.9058C63.4948 12.0216 63.5218 12.1246 63.5548 12.2097C63.5708 12.2517 63.5928 12.3027 63.6258 12.3528C63.6508 12.3918 63.7168 12.4878 63.8428 12.5475C63.9138 12.5814 64.0068 12.6047 64.1128 12.5913C64.2188 12.578 64.3018 12.5327 64.3608 12.4843C64.4658 12.3987 64.5068 12.2935 64.5218 12.2512C64.5568 12.1525 64.5628 12.0514 64.5628 11.981H64.0958H63.6278C63.6278 12.0087 63.6238 11.9873 63.6408 11.9401C63.6458 11.9235 63.6768 11.837 63.7678 11.7618C63.8208 11.7185 63.8978 11.6763 63.9958 11.6639C64.0948 11.6515 64.1808 11.6735 64.2448 11.7036C64.3538 11.7559 64.4038 11.835 64.4108 11.8463C64.4258 11.8687 64.4298 11.8813 64.4258 11.8729C64.4198 11.8573 64.4058 11.8117 64.3878 11.7213C64.3138 11.3565 64.2308 10.5214 64.2308 8.87881H63.7628ZM64.0958 11.981V12.4484H66.8008V11.981V11.5136H64.0958V11.981ZM66.8008 11.981H67.2688V3.73791H66.8008H66.3338V11.981H66.8008ZM66.8008 3.73791V3.27051H56.5188V3.73791V4.20521H66.8008V3.73791ZM56.5188 3.73791H56.0518V4.91551H56.5188H56.9868V3.73791H56.5188Z"
      fill="#16171B"
    />
    <path
      d="M107.928 5.14028L110.966 4.91588V3.73828H100.684V11.9814L102.553 12.1507C102.553 12.5432 103.021 11.6833 103.021 8.87918C103.021 5.58188 106.305 5.14028 107.928 5.14028Z"
      fill="#16171B"
    />
    <path
      d="M110.965 4.91551H111.432V5.34961L110.999 5.38161L110.965 4.91551ZM107.927 5.13991L107.961 5.60601L107.944 5.60731H107.927V5.13991ZM102.552 12.1503L102.595 11.6848L103.02 11.7233V12.1503H102.552ZM100.683 11.981L100.641 12.4464L100.216 12.4079V11.981H100.683ZM100.683 3.73791H100.216V3.27051H100.683V3.73791ZM110.965 3.73791V3.27051H111.432V3.73791H110.965ZM110.965 4.91551L110.999 5.38161L107.961 5.60601L107.927 5.13991L107.893 4.67391L110.93 4.44941L110.965 4.91551ZM107.927 5.13991V5.60731C107.142 5.60731 106.003 5.71641 105.077 6.18921C104.62 6.42231 104.228 6.73741 103.948 7.16091C103.671 7.58161 103.487 8.13591 103.487 8.87881H103.02H102.552C102.552 7.97311 102.779 7.23571 103.168 6.64621C103.555 6.05961 104.085 5.64611 104.652 5.35661C105.774 4.78431 107.088 4.67261 107.927 4.67261V5.13991ZM103.02 8.87881H103.487C103.487 10.3033 103.369 11.256 103.242 11.8164C103.183 12.0786 103.111 12.3063 103.02 12.449C102.998 12.4838 102.949 12.5562 102.867 12.6151C102.823 12.6465 102.746 12.6902 102.642 12.7035C102.527 12.7183 102.41 12.6909 102.312 12.6255C102.15 12.5169 102.111 12.3532 102.103 12.3158C102.088 12.2525 102.085 12.1932 102.085 12.1503H102.552H103.02C103.02 12.1564 103.021 12.1364 103.013 12.1022C103.011 12.094 102.982 11.9494 102.833 11.8494C102.742 11.7882 102.632 11.7625 102.524 11.7763C102.427 11.7886 102.36 11.8287 102.324 11.8538C102.259 11.9 102.231 11.9489 102.23 11.9491C102.228 11.9533 102.273 11.8643 102.33 11.6101C102.437 11.1377 102.552 10.2584 102.552 8.87881H103.02ZM102.552 12.1503L102.51 12.6157L100.641 12.4464L100.683 11.981L100.725 11.5155L102.595 11.6848L102.552 12.1503ZM100.683 11.981H100.216V3.73791H100.683H101.15V11.981H100.683ZM100.683 3.73791V3.27051H110.965V3.73791V4.20521H100.683V3.73791ZM110.965 3.73791H111.432V4.91551H110.965H110.497V3.73791H110.965Z"
      fill="#16171B"
    />
    <path
      d="M82.4061 162.627C83.1771 162.731 83.9581 162.731 84.7291 162.628L153.375 153.443C155.288 153.187 156.962 154.733 156.859 156.661C156.801 157.747 156.174 158.722 155.21 159.226L85.5911 195.594C84.3231 196.256 82.8121 196.256 81.5441 195.594L12.0461 159.289C11.0151 158.751 10.3691 157.684 10.3691 156.521C10.3691 154.631 12.0351 153.174 13.9081 153.426L82.4061 162.627Z"
      fill="#004F3B"
    />
  </svg>', label: 'ISO 9001', }, -}; \ No newline at end of file +}; diff --git a/apps/api/src/trust-portal/dto/trust-vendor.dto.ts b/apps/api/src/trust-portal/dto/trust-vendor.dto.ts index a790c4fb7..47396e54d 100644 --- a/apps/api/src/trust-portal/dto/trust-vendor.dto.ts +++ b/apps/api/src/trust-portal/dto/trust-vendor.dto.ts @@ -1,7 +1,16 @@ import { z } from 'zod'; const ComplianceBadgeSchema = z.object({ - type: z.enum(['soc2', 'iso27001', 'iso42001', 'gdpr', 'hipaa', 'pci_dss', 'nen7510', 'iso9001']), + type: z.enum([ + 'soc2', + 'iso27001', + 'iso42001', + 'gdpr', + 'hipaa', + 'pci_dss', + 'nen7510', + 'iso9001', + ]), verified: z.boolean(), }); @@ -12,6 +21,8 @@ export const UpdateVendorTrustSettingsSchema = z.object({ complianceBadges: z.array(ComplianceBadgeSchema).optional().nullable(), }); -export type UpdateVendorTrustSettingsDto = z.infer; +export type UpdateVendorTrustSettingsDto = z.infer< + typeof UpdateVendorTrustSettingsSchema +>; export type ComplianceBadge = z.infer; diff --git a/apps/api/src/trust-portal/trust-access.service.ts b/apps/api/src/trust-portal/trust-access.service.ts index 6427cbf2c..003b9acef 100644 --- a/apps/api/src/trust-portal/trust-access.service.ts +++ b/apps/api/src/trust-portal/trust-access.service.ts @@ -25,7 +25,6 @@ import archiver from 'archiver'; import { PassThrough, Readable } from 'stream'; import { PDFDocument, rgb, StandardFonts } from 'pdf-lib'; - @Injectable() export class TrustAccessService { /** @@ -2522,7 +2521,9 @@ export class TrustAccessService { } return { ...vendor, - complianceBadges: this.formatComplianceBadgeLabels(vendor.complianceBadges), + complianceBadges: this.formatComplianceBadgeLabels( + vendor.complianceBadges, + ), trustPortalUrl, }; }); @@ -2532,7 +2533,9 @@ export class TrustAccessService { * Format compliance badges as simple type + label pairs for external rendering. * Does NOT include branded icons to avoid implying vendors were certified through us. */ - private formatComplianceBadgeLabels(badges: unknown): { type: string; label: string }[] { + private formatComplianceBadgeLabels( + badges: unknown, + ): { type: string; label: string }[] { if (!badges || !Array.isArray(badges)) { return []; } diff --git a/apps/api/src/trust-portal/trust-portal.controller.ts b/apps/api/src/trust-portal/trust-portal.controller.ts index 4cc5e5a18..1117467ab 100644 --- a/apps/api/src/trust-portal/trust-portal.controller.ts +++ b/apps/api/src/trust-portal/trust-portal.controller.ts @@ -325,7 +325,11 @@ export class TrustPortalController { @AuthContext() authContext: AuthContextType, ) { const dto = UpdateCustomLinkSchema.parse(body); - return this.trustPortalService.updateCustomLink(linkId, dto, authContext.organizationId); + return this.trustPortalService.updateCustomLink( + linkId, + dto, + authContext.organizationId, + ); } @Post('custom-links/:linkId/delete') @@ -341,7 +345,10 @@ export class TrustPortalController { @Param('linkId') linkId: string, @AuthContext() authContext: AuthContextType, ) { - return this.trustPortalService.deleteCustomLink(linkId, authContext.organizationId); + return this.trustPortalService.deleteCustomLink( + linkId, + authContext.organizationId, + ); } @Post('custom-links/reorder') @@ -394,7 +401,11 @@ export class TrustPortalController { @AuthContext() authContext: AuthContextType, ) { const dto = UpdateVendorTrustSettingsSchema.parse(body); - return this.trustPortalService.updateVendorTrustSettings(vendorId, dto, authContext.organizationId); + return this.trustPortalService.updateVendorTrustSettings( + vendorId, + dto, + authContext.organizationId, + ); } @Get('vendors') diff --git a/apps/api/src/trust-portal/trust-portal.service.ts b/apps/api/src/trust-portal/trust-portal.service.ts index ea21950a6..cae381605 100644 --- a/apps/api/src/trust-portal/trust-portal.service.ts +++ b/apps/api/src/trust-portal/trust-portal.service.ts @@ -644,11 +644,14 @@ export class TrustPortalService { } } - async updateOverview(organizationId: string, data: { - overviewTitle?: string | null; - overviewContent?: string | null; - showOverview?: boolean; - }) { + async updateOverview( + organizationId: string, + data: { + overviewTitle?: string | null; + overviewContent?: string | null; + showOverview?: boolean; + }, + ) { const trust = await db.trust.findUnique({ where: { organizationId }, }); @@ -684,11 +687,14 @@ export class TrustPortalService { return trust; } - async createCustomLink(organizationId: string, data: { - title: string; - description?: string | null; - url: string; - }) { + async createCustomLink( + organizationId: string, + data: { + title: string; + description?: string | null; + url: string; + }, + ) { const maxOrder = await db.trustCustomLink.findFirst({ where: { organizationId }, orderBy: { order: 'desc' }, @@ -708,12 +714,16 @@ export class TrustPortalService { }); } - async updateCustomLink(linkId: string, data: { - title?: string; - description?: string | null; - url?: string; - isActive?: boolean; - }, organizationId: string) { + async updateCustomLink( + linkId: string, + data: { + title?: string; + description?: string | null; + url?: string; + isActive?: boolean; + }, + organizationId: string, + ) { const link = await db.trustCustomLink.findUnique({ where: { id: linkId }, }); @@ -723,7 +733,9 @@ export class TrustPortalService { } if (link.organizationId !== organizationId) { - throw new BadRequestException('You can only modify custom links belonging to your organization'); + throw new BadRequestException( + 'You can only modify custom links belonging to your organization', + ); } return db.trustCustomLink.update({ @@ -742,7 +754,9 @@ export class TrustPortalService { } if (link.organizationId !== organizationId) { - throw new BadRequestException('You can only delete custom links belonging to your organization'); + throw new BadRequestException( + 'You can only delete custom links belonging to your organization', + ); } await db.trustCustomLink.delete({ @@ -761,7 +775,9 @@ export class TrustPortalService { const invalidIds = linkIds.filter((id) => !linkIdSet.has(id)); if (invalidIds.length > 0) { - throw new BadRequestException('Some link IDs do not belong to this organization'); + throw new BadRequestException( + 'Some link IDs do not belong to this organization', + ); } await db.$transaction( @@ -790,10 +806,7 @@ export class TrustPortalService { isSubProcessor: true, showOnTrustPortal: true, }, - orderBy: [ - { trustPortalOrder: 'asc' }, - { name: 'asc' }, - ], + orderBy: [{ trustPortalOrder: 'asc' }, { name: 'asc' }], select: { id: true, name: true, @@ -805,12 +818,16 @@ export class TrustPortalService { }); } - async updateVendorTrustSettings(vendorId: string, data: { - logoUrl?: string | null; - showOnTrustPortal?: boolean; - trustPortalOrder?: number | null; - complianceBadges?: any; - }, organizationId: string) { + async updateVendorTrustSettings( + vendorId: string, + data: { + logoUrl?: string | null; + showOnTrustPortal?: boolean; + trustPortalOrder?: number | null; + complianceBadges?: any; + }, + organizationId: string, + ) { const vendor = await db.vendor.findUnique({ where: { id: vendorId }, }); @@ -820,7 +837,9 @@ export class TrustPortalService { } if (vendor.organizationId !== organizationId) { - throw new BadRequestException('You can only modify vendors belonging to your organization'); + throw new BadRequestException( + 'You can only modify vendors belonging to your organization', + ); } return db.vendor.update({ diff --git a/apps/app/next.config.ts b/apps/app/next.config.ts index ab6dc8956..d5fffca64 100644 --- a/apps/app/next.config.ts +++ b/apps/app/next.config.ts @@ -48,6 +48,7 @@ const config: NextConfig = { '@prisma/client', '@trycompai/design-system', '@carbon/icons-react', + '@comp/company', ], images: { remotePatterns: [ diff --git a/apps/app/package.json b/apps/app/package.json index 19c7b1f2b..5ab0118ae 100644 --- a/apps/app/package.json +++ b/apps/app/package.json @@ -1,191 +1,192 @@ { - "name": "@comp/app", - "version": "0.1.0", - "type": "module", - "dependencies": { - "@ai-sdk/anthropic": "^2.0.0", - "@ai-sdk/groq": "^2.0.0", - "@ai-sdk/openai": "^2.0.80", - "@ai-sdk/provider": "^2.0.0", - "@ai-sdk/react": "^2.0.60", - "@ai-sdk/rsc": "^1.0.0", - "@aws-sdk/client-ec2": "^3.911.0", - "@aws-sdk/client-lambda": "^3.891.0", - "@aws-sdk/client-s3": "^3.859.0", - "@aws-sdk/client-sts": "^3.808.0", - "@aws-sdk/s3-request-presigner": "^3.859.0", - "@azure/core-rest-pipeline": "^1.21.0", - "@browserbasehq/sdk": "^2.5.0", - "@browserbasehq/stagehand": "^3.0.5", - "@calcom/atoms": "^1.0.102-framer", - "@calcom/embed-react": "^1.5.3", - "@comp/integration-platform": "workspace:*", - "@date-fns/tz": "^1.2.0", - "@dnd-kit/core": "^6.3.1", - "@dnd-kit/modifiers": "^9.0.0", - "@dnd-kit/sortable": "^10.0.0", - "@dnd-kit/utilities": "^3.2.2", - "@dub/analytics": "^0.0.27", - "@dub/better-auth": "^0.0.6", - "@dub/embed-react": "^0.0.16", - "@hookform/resolvers": "^5.1.1", - "@mendable/firecrawl-js": "^1.24.0", - "@monaco-editor/react": "^4.7.0", - "@nangohq/frontend": "^0.53.2", - "@next/third-parties": "^15.3.1", - "@novu/api": "^1.6.0", - "@novu/nextjs": "^3.10.1", - "@number-flow/react": "^0.5.9", - "@prisma/client": "6.18.0", - "@prisma/instrumentation": "6.18.0", - "@prisma/nextjs-monorepo-workaround-plugin": "6.18.0", - "@radix-ui/react-slot": "^1.2.3", - "@react-email/components": "^0.0.41", - "@react-email/render": "^1.1.2", - "@react-three/drei": "^10.3.0", - "@react-three/fiber": "^9.1.2", - "@react-three/postprocessing": "^3.0.4", - "@t3-oss/env-nextjs": "^0.13.8", - "@tanstack/react-form": "^1.23.8", - "@tanstack/react-query": "^5.90.7", - "@tanstack/react-table": "^8.21.3", - "@tiptap/extension-mention": "3.16.0", - "@tiptap/extension-table": "3.16.0", - "@tiptap/react": "3.16.0", - "@trigger.dev/react-hooks": "4.0.6", - "@trigger.dev/sdk": "4.0.6", - "@trycompai/db": "1.3.22", - "@trycompai/design-system": "^1.0.32", - "@trycompai/email": "workspace:*", - "@types/canvas-confetti": "^1.9.0", - "@types/react-syntax-highlighter": "^15.5.13", - "@types/three": "^0.180.0", - "@uiw/react-json-view": "^2.0.0-alpha.40", - "@uploadthing/react": "^7.3.0", - "@upstash/ratelimit": "^2.0.5", - "@vercel/analytics": "^1.5.0", - "@vercel/sandbox": "^0.0.21", - "@vercel/sdk": "^1.7.1", - "@xyflow/react": "^12.10.0", - "ai": "^5.0.108", - "ai-elements": "^1.6.1", - "axios": "^1.9.0", - "better-auth": "^1.3.27", - "botid": "^1.5.5", - "canvas-confetti": "^1.9.3", - "d3": "^7.9.0", - "date-fns": "^4.1.0", - "diff": "^8.0.2", - "dub": "^0.66.1", - "framer-motion": "^12.18.1", - "geist": "^1.3.1", - "jspdf": "^3.0.2", - "lucide-react": "^0.544.0", - "mammoth": "^1.11.0", - "motion": "^12.9.2", - "next": "^16.0.10", - "next-safe-action": "^8.0.3", - "next-themes": "^0.4.4", - "nuqs": "^2.4.3", - "pdf-parse": "^2.4.5", - "playwright-core": "^1.52.0", - "posthog-js": "^1.236.6", - "posthog-node": "^5.8.2", - "prisma": "6.18.0", - "puppeteer-core": "^24.7.2", - "react": "^19.2.3", - "react-dom": "^19.2.3", - "react-email": "^4.0.15", - "react-hook-form": "^7.61.1", - "react-hotkeys-hook": "^5.1.0", - "react-intersection-observer": "^9.16.0", - "react-markdown": "10.1.0", - "react-spinners": "^0.17.0", - "react-syntax-highlighter": "^15.6.6", - "react-textarea-autosize": "^8.5.9", - "react-use-draggable-scroll": "^0.4.7", - "react-wrap-balancer": "^1.1.1", - "rehype-raw": "^7.0.0", - "remark-gfm": "^4.0.1", - "remark-parse": "^11.0.0", - "resend": "^4.4.1", - "sonner": "^2.0.5", - "stripe": "^20.0.0", - "swr": "^2.3.4", - "three": "^0.182.0", - "ts-pattern": "^5.7.0", - "use-debounce": "^10.0.4", - "use-long-press": "^3.3.0", - "use-stick-to-bottom": "^1.1.1", - "xlsx": "^0.18.5", - "xml2js": "^0.6.2", - "zaraz-ts": "^1.2.0", - "zod": "^4.0.0", - "zustand": "^5.0.3" - }, - "devDependencies": { - "@playwright/experimental-ct-react": "^1.53.1", - "@playwright/test": "^1.53.1", - "@tailwindcss/postcss": "^4.1.10", - "@testing-library/dom": "^10.4.0", - "@testing-library/jest-dom": "^6.6.3", - "@testing-library/react": "^16.3.0", - "@trigger.dev/build": "4.0.6", - "@types/d3": "^7.4.3", - "@types/jspdf": "^2.0.0", - "@types/node": "^24.0.3", - "@vitejs/plugin-react": "^4.6.0", - "@vitest/ui": "^3.2.4", - "eslint": "^9.18.0", - "eslint-config-next": "15.5.2", - "fleetctl": "^4.68.1", - "glob": "^11.0.3", - "jsdom": "^26.1.0", - "postcss": "^8.5.4", - "raw-loader": "^4.0.2", - "tailwindcss": "^4.1.8", - "typescript": "^5.8.3", - "vite-tsconfig-paths": "^5.1.4", - "vitest": "^3.2.4" - }, - "exports": { - "./src/lib/encryption": "./src/lib/encryption.ts" - }, - "peerDependencies": { - "react": "^19.1.1", - "react-dom": "^19.1.0" - }, - "pnpm": { - "overrides": { - "tiptap-extension-global-drag-handle": "^0.1.18" + "name": "@comp/app", + "version": "0.1.0", + "type": "module", + "dependencies": { + "@ai-sdk/anthropic": "^2.0.0", + "@ai-sdk/groq": "^2.0.0", + "@ai-sdk/openai": "^2.0.80", + "@ai-sdk/provider": "^2.0.0", + "@ai-sdk/react": "^2.0.60", + "@ai-sdk/rsc": "^1.0.0", + "@aws-sdk/client-ec2": "^3.911.0", + "@aws-sdk/client-lambda": "^3.891.0", + "@aws-sdk/client-s3": "^3.859.0", + "@aws-sdk/client-sts": "^3.808.0", + "@aws-sdk/s3-request-presigner": "^3.859.0", + "@azure/core-rest-pipeline": "^1.21.0", + "@browserbasehq/sdk": "^2.5.0", + "@browserbasehq/stagehand": "^3.0.5", + "@calcom/atoms": "^1.0.102-framer", + "@calcom/embed-react": "^1.5.3", + "@comp/company": "workspace:*", + "@comp/integration-platform": "workspace:*", + "@date-fns/tz": "^1.2.0", + "@dnd-kit/core": "^6.3.1", + "@dnd-kit/modifiers": "^9.0.0", + "@dnd-kit/sortable": "^10.0.0", + "@dnd-kit/utilities": "^3.2.2", + "@dub/analytics": "^0.0.27", + "@dub/better-auth": "^0.0.6", + "@dub/embed-react": "^0.0.16", + "@hookform/resolvers": "^5.1.1", + "@mendable/firecrawl-js": "^1.24.0", + "@monaco-editor/react": "^4.7.0", + "@nangohq/frontend": "^0.53.2", + "@next/third-parties": "^15.3.1", + "@novu/api": "^1.6.0", + "@novu/nextjs": "^3.10.1", + "@number-flow/react": "^0.5.9", + "@prisma/client": "6.18.0", + "@prisma/instrumentation": "6.18.0", + "@prisma/nextjs-monorepo-workaround-plugin": "6.18.0", + "@radix-ui/react-slot": "^1.2.3", + "@react-email/components": "^0.0.41", + "@react-email/render": "^1.1.2", + "@react-three/drei": "^10.3.0", + "@react-three/fiber": "^9.1.2", + "@react-three/postprocessing": "^3.0.4", + "@t3-oss/env-nextjs": "^0.13.8", + "@tanstack/react-form": "^1.23.8", + "@tanstack/react-query": "^5.90.7", + "@tanstack/react-table": "^8.21.3", + "@tiptap/extension-mention": "3.16.0", + "@tiptap/extension-table": "3.16.0", + "@tiptap/react": "3.16.0", + "@trigger.dev/react-hooks": "4.0.6", + "@trigger.dev/sdk": "4.0.6", + "@trycompai/db": "1.3.22", + "@trycompai/design-system": "^1.0.32", + "@trycompai/email": "workspace:*", + "@types/canvas-confetti": "^1.9.0", + "@types/react-syntax-highlighter": "^15.5.13", + "@types/three": "^0.180.0", + "@uiw/react-json-view": "^2.0.0-alpha.40", + "@uploadthing/react": "^7.3.0", + "@upstash/ratelimit": "^2.0.5", + "@vercel/analytics": "^1.5.0", + "@vercel/sandbox": "^0.0.21", + "@vercel/sdk": "^1.7.1", + "@xyflow/react": "^12.10.0", + "ai": "^5.0.108", + "ai-elements": "^1.6.1", + "axios": "^1.9.0", + "better-auth": "^1.3.27", + "botid": "^1.5.5", + "canvas-confetti": "^1.9.3", + "d3": "^7.9.0", + "date-fns": "^4.1.0", + "diff": "^8.0.2", + "dub": "^0.66.1", + "framer-motion": "^12.18.1", + "geist": "^1.3.1", + "jspdf": "^3.0.2", + "lucide-react": "^0.544.0", + "mammoth": "^1.11.0", + "motion": "^12.9.2", + "next": "^16.0.10", + "next-safe-action": "^8.0.3", + "next-themes": "^0.4.4", + "nuqs": "^2.4.3", + "pdf-parse": "^2.4.5", + "playwright-core": "^1.52.0", + "posthog-js": "^1.236.6", + "posthog-node": "^5.8.2", + "prisma": "6.18.0", + "puppeteer-core": "^24.7.2", + "react": "^19.2.3", + "react-dom": "^19.2.3", + "react-email": "^4.0.15", + "react-hook-form": "^7.61.1", + "react-hotkeys-hook": "^5.1.0", + "react-intersection-observer": "^9.16.0", + "react-markdown": "10.1.0", + "react-spinners": "^0.17.0", + "react-syntax-highlighter": "^15.6.6", + "react-textarea-autosize": "^8.5.9", + "react-use-draggable-scroll": "^0.4.7", + "react-wrap-balancer": "^1.1.1", + "rehype-raw": "^7.0.0", + "remark-gfm": "^4.0.1", + "remark-parse": "^11.0.0", + "resend": "^4.4.1", + "sonner": "^2.0.5", + "stripe": "^20.0.0", + "swr": "^2.3.4", + "three": "^0.182.0", + "ts-pattern": "^5.7.0", + "use-debounce": "^10.0.4", + "use-long-press": "^3.3.0", + "use-stick-to-bottom": "^1.1.1", + "xlsx": "^0.18.5", + "xml2js": "^0.6.2", + "zaraz-ts": "^1.2.0", + "zod": "^4.0.0", + "zustand": "^5.0.3" + }, + "devDependencies": { + "@playwright/experimental-ct-react": "^1.53.1", + "@playwright/test": "^1.53.1", + "@tailwindcss/postcss": "^4.1.10", + "@testing-library/dom": "^10.4.0", + "@testing-library/jest-dom": "^6.6.3", + "@testing-library/react": "^16.3.0", + "@trigger.dev/build": "4.0.6", + "@types/d3": "^7.4.3", + "@types/jspdf": "^2.0.0", + "@types/node": "^24.0.3", + "@vitejs/plugin-react": "^4.6.0", + "@vitest/ui": "^3.2.4", + "eslint": "^9.18.0", + "eslint-config-next": "15.5.2", + "fleetctl": "^4.68.1", + "glob": "^11.0.3", + "jsdom": "^26.1.0", + "postcss": "^8.5.4", + "raw-loader": "^4.0.2", + "tailwindcss": "^4.1.8", + "typescript": "^5.8.3", + "vite-tsconfig-paths": "^5.1.4", + "vitest": "^3.2.4" + }, + "exports": { + "./src/lib/encryption": "./src/lib/encryption.ts" + }, + "peerDependencies": { + "react": "^19.1.1", + "react-dom": "^19.1.0" + }, + "pnpm": { + "overrides": { + "tiptap-extension-global-drag-handle": "^0.1.18" + } + }, + "private": true, + "scripts": { + "analyze-locale-usage": "bunx tsx src/locales/analyze-locale-usage.ts", + "build": "next build", + "build:docker": "prisma generate && next build", + "db:generate": "bun run db:getschema && prisma generate", + "db:getschema": "node ../../packages/db/scripts/combine-schemas.js && cp ../../packages/db/dist/schema.prisma prisma/schema.prisma", + "db:migrate": "cd ../../packages/db && bunx prisma migrate dev && cd ../../apps/app", + "deploy:trigger-prod": "npx trigger.dev@4.0.6 deploy", + "dev": "bun i && bunx concurrently --kill-others --names \"next,trigger\" --prefix-colors \"yellow,blue\" \"next dev --turbo -p 3000\" \"bunx trigger.dev@4.0.6 dev\"", + "lint": "eslint . && prettier --check .", + "prebuild": "bun run db:generate", + "postinstall": "prisma generate --schema=./prisma/schema.prisma || exit 0", + "start": "next start", + "test": "vitest", + "test:all": "./scripts/test-all.sh", + "test:coverage": "vitest run --coverage", + "test:e2e": "playwright test", + "test:e2e:debug": "playwright test --debug", + "test:e2e:headed": "playwright test --headed", + "test:e2e:install": "playwright install --with-deps", + "test:e2e:report": "playwright show-report", + "test:e2e:setup": "./scripts/setup-e2e.sh", + "test:e2e:ui": "playwright test --ui", + "test:ui": "vitest --ui", + "test:watch": "vitest --watch", + "typecheck": "tsc --noEmit" } - }, - "private": true, - "scripts": { - "analyze-locale-usage": "bunx tsx src/locales/analyze-locale-usage.ts", - "build": "next build", - "build:docker": "prisma generate && next build", - "db:generate": "bun run db:getschema && prisma generate", - "db:getschema": "node ../../packages/db/scripts/combine-schemas.js && cp ../../packages/db/dist/schema.prisma prisma/schema.prisma", - "db:migrate": "cd ../../packages/db && bunx prisma migrate dev && cd ../../apps/app", - "deploy:trigger-prod": "npx trigger.dev@4.0.6 deploy", - "dev": "bun i && bunx concurrently --kill-others --names \"next,trigger\" --prefix-colors \"yellow,blue\" \"next dev --turbo -p 3000\" \"bunx trigger.dev@4.0.6 dev\"", - "lint": "eslint . && prettier --check .", - "prebuild": "bun run db:generate", - "postinstall": "prisma generate --schema=./prisma/schema.prisma || exit 0", - "start": "next start", - "test": "vitest", - "test:all": "./scripts/test-all.sh", - "test:coverage": "vitest run --coverage", - "test:e2e": "playwright test", - "test:e2e:debug": "playwright test --debug", - "test:e2e:headed": "playwright test --headed", - "test:e2e:install": "playwright install --with-deps", - "test:e2e:report": "playwright show-report", - "test:e2e:setup": "./scripts/setup-e2e.sh", - "test:e2e:ui": "playwright test --ui", - "test:ui": "vitest --ui", - "test:watch": "vitest --watch", - "typecheck": "tsc --noEmit" - } -} \ No newline at end of file +} diff --git a/apps/app/src/actions/organization/update-organization-access-request-form-action.ts b/apps/app/src/actions/organization/update-organization-access-request-form-action.ts new file mode 100644 index 000000000..adfd12e7a --- /dev/null +++ b/apps/app/src/actions/organization/update-organization-access-request-form-action.ts @@ -0,0 +1,46 @@ +'use server'; + +import { db } from '@db'; +import { revalidatePath, revalidateTag } from 'next/cache'; +import { headers } from 'next/headers'; +import { authActionClient } from '../safe-action'; +import { organizationAccessRequestFormSchema } from '../schema'; + +export const updateOrganizationAccessRequestFormAction = authActionClient + .inputSchema(organizationAccessRequestFormSchema) + .metadata({ + name: 'update-organization-access-request-form', + track: { + event: 'update-organization-access-request-form', + channel: 'server', + }, + }) + .action(async ({ parsedInput, ctx }) => { + const { accessRequestFormEnabled } = parsedInput; + const { activeOrganizationId } = ctx.session; + + if (!activeOrganizationId) { + throw new Error('No active organization'); + } + + try { + await db.organization.update({ + where: { id: activeOrganizationId }, + data: { accessRequestFormEnabled }, + }); + + const headersList = await headers(); + let path = headersList.get('x-pathname') || headersList.get('referer') || ''; + path = path.replace(/\/[a-z]{2}\//, '/'); + + revalidatePath(path); + revalidateTag(`organization_${activeOrganizationId}`, 'max'); + + return { + success: true, + }; + } catch (error) { + console.error(error); + throw new Error('Failed to update access request form setting'); + } + }); diff --git a/apps/app/src/actions/organization/update-organization-whistleblower-report-action.ts b/apps/app/src/actions/organization/update-organization-whistleblower-report-action.ts new file mode 100644 index 000000000..e01cceb2c --- /dev/null +++ b/apps/app/src/actions/organization/update-organization-whistleblower-report-action.ts @@ -0,0 +1,46 @@ +'use server'; + +import { db } from '@db'; +import { revalidatePath, revalidateTag } from 'next/cache'; +import { headers } from 'next/headers'; +import { authActionClient } from '../safe-action'; +import { organizationWhistleblowerReportSchema } from '../schema'; + +export const updateOrganizationWhistleblowerReportAction = authActionClient + .inputSchema(organizationWhistleblowerReportSchema) + .metadata({ + name: 'update-organization-whistleblower-report', + track: { + event: 'update-organization-whistleblower-report', + channel: 'server', + }, + }) + .action(async ({ parsedInput, ctx }) => { + const { whistleblowerReportEnabled } = parsedInput; + const { activeOrganizationId } = ctx.session; + + if (!activeOrganizationId) { + throw new Error('No active organization'); + } + + try { + await db.organization.update({ + where: { id: activeOrganizationId }, + data: { whistleblowerReportEnabled }, + }); + + const headersList = await headers(); + let path = headersList.get('x-pathname') || headersList.get('referer') || ''; + path = path.replace(/\/[a-z]{2}\//, '/'); + + revalidatePath(path); + revalidateTag(`organization_${activeOrganizationId}`, 'max'); + + return { + success: true, + }; + } catch (error) { + console.error(error); + throw new Error('Failed to update whistleblower report setting'); + } + }); diff --git a/apps/app/src/actions/schema.ts b/apps/app/src/actions/schema.ts index 3f7034ce6..9a91c755f 100644 --- a/apps/app/src/actions/schema.ts +++ b/apps/app/src/actions/schema.ts @@ -77,6 +77,14 @@ export const organizationSecurityTrainingStepSchema = z.object({ securityTrainingStepEnabled: z.boolean(), }); +export const organizationWhistleblowerReportSchema = z.object({ + whistleblowerReportEnabled: z.boolean(), +}); + +export const organizationAccessRequestFormSchema = z.object({ + accessRequestFormEnabled: z.boolean(), +}); + // Risks export const createRiskSchema = z.object({ title: z diff --git a/apps/app/src/app/(app)/[orgId]/components/AppShellWrapper.tsx b/apps/app/src/app/(app)/[orgId]/components/AppShellWrapper.tsx index 3b57abb8c..2833f0fc7 100644 --- a/apps/app/src/app/(app)/[orgId]/components/AppShellWrapper.tsx +++ b/apps/app/src/app/(app)/[orgId]/components/AppShellWrapper.tsx @@ -243,7 +243,13 @@ function AppShellWrapperContent({ {isSettingsActive ? ( diff --git a/apps/app/src/app/(app)/[orgId]/components/AppSidebar.tsx b/apps/app/src/app/(app)/[orgId]/components/AppSidebar.tsx index d5b58307f..f92cfeb30 100644 --- a/apps/app/src/app/(app)/[orgId]/components/AppSidebar.tsx +++ b/apps/app/src/app/(app)/[orgId]/components/AppSidebar.tsx @@ -1,6 +1,7 @@ 'use client'; import { + Catalog, Chemistry, Dashboard, Document, @@ -74,6 +75,12 @@ export function AppSidebar({ name: 'Evidence', icon: , }, + { + id: 'documents', + path: `/${organization.id}/documents`, + name: 'Documents', + icon: , + }, { id: 'people', path: `/${organization.id}/people/all`, diff --git a/apps/app/src/app/(app)/[orgId]/components/app-shell-search-groups.tsx b/apps/app/src/app/(app)/[orgId]/components/app-shell-search-groups.tsx index 5d260829b..ad944cf2d 100644 --- a/apps/app/src/app/(app)/[orgId]/components/app-shell-search-groups.tsx +++ b/apps/app/src/app/(app)/[orgId]/components/app-shell-search-groups.tsx @@ -1,4 +1,5 @@ import { + Catalog, Chemistry, Dashboard, Document, @@ -124,6 +125,14 @@ export const getAppShellSearchGroups = ({ }), ] : []), + createNavItem({ + id: 'documents', + label: 'Documents', + icon: , + path: `/${organizationId}/documents`, + keywords: ['company', 'tasks', 'forms', 'evidence submissions', 'documents'], + router, + }), createNavItem({ id: 'people', label: 'People', diff --git a/apps/app/src/app/(app)/[orgId]/documents/[formType]/new/page.tsx b/apps/app/src/app/(app)/[orgId]/documents/[formType]/new/page.tsx new file mode 100644 index 000000000..eec663891 --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/[formType]/new/page.tsx @@ -0,0 +1,49 @@ +import { CompanySubmissionWizard } from '@/app/(app)/[orgId]/documents/components/CompanySubmissionWizard'; +import { conciseFormDescriptions } from '@/app/(app)/[orgId]/documents/form-descriptions'; +import { Breadcrumb, PageHeader, PageLayout, Text } from '@trycompai/design-system'; +import Link from 'next/link'; +import { notFound } from 'next/navigation'; +import { evidenceFormDefinitions, evidenceFormTypeSchema } from '../../forms'; + +export default async function NewCompanySubmissionPage({ + params, +}: { + params: Promise<{ orgId: string; formType: string }>; +}) { + const { orgId, formType } = await params; + const parsedType = evidenceFormTypeSchema.safeParse(formType); + + if (!parsedType.success) { + notFound(); + } + + const parsedFormType = parsedType.data; + const formDefinition = evidenceFormDefinitions[parsedFormType]; + + return ( + + }, + }, + { + label: formDefinition.title, + href: `/${orgId}/documents/${parsedFormType}`, + props: { render: }, + }, + { label: 'New Submission', isCurrent: true }, + ]} + /> + +
+ + {conciseFormDescriptions[parsedFormType] ?? formDefinition.description} + + +
+
+ ); +} diff --git a/apps/app/src/app/(app)/[orgId]/documents/[formType]/page.tsx b/apps/app/src/app/(app)/[orgId]/documents/[formType]/page.tsx new file mode 100644 index 000000000..f1651d3c6 --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/[formType]/page.tsx @@ -0,0 +1,36 @@ +import { CompanyFormPageClient } from '@/app/(app)/[orgId]/documents/components/CompanyFormPageClient'; +import { Breadcrumb, PageLayout } from '@trycompai/design-system'; +import Link from 'next/link'; +import { notFound } from 'next/navigation'; +import { evidenceFormDefinitions, evidenceFormTypeSchema } from '../forms'; + +export default async function CompanyFormDetailPage({ + params, +}: { + params: Promise<{ orgId: string; formType: string }>; +}) { + const { orgId, formType } = await params; + const parsedType = evidenceFormTypeSchema.safeParse(formType); + + if (!parsedType.success) { + notFound(); + } + + const formDefinition = evidenceFormDefinitions[parsedType.data]; + + return ( + + }, + }, + { label: formDefinition.title, isCurrent: true }, + ]} + /> + + + ); +} diff --git a/apps/app/src/app/(app)/[orgId]/documents/[formType]/submissions/[submissionId]/page.tsx b/apps/app/src/app/(app)/[orgId]/documents/[formType]/submissions/[submissionId]/page.tsx new file mode 100644 index 000000000..60c26ddf1 --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/[formType]/submissions/[submissionId]/page.tsx @@ -0,0 +1,63 @@ +import { CompanySubmissionDetailPageClient } from '@/app/(app)/[orgId]/documents/components/CompanySubmissionDetailPageClient'; +import { auth } from '@/utils/auth'; +import { db } from '@db'; +import { Breadcrumb, PageHeader, PageLayout } from '@trycompai/design-system'; +import { headers } from 'next/headers'; +import Link from 'next/link'; +import { notFound } from 'next/navigation'; +import { evidenceFormDefinitions, evidenceFormTypeSchema } from '../../../forms'; + +export default async function CompanySubmissionDetailPage({ + params, +}: { + params: Promise<{ orgId: string; formType: string; submissionId: string }>; +}) { + const { orgId, formType, submissionId } = await params; + const parsedType = evidenceFormTypeSchema.safeParse(formType); + + if (!parsedType.success) { + notFound(); + } + + const parsedFormType = parsedType.data; + const formDefinition = evidenceFormDefinitions[parsedFormType]; + const session = await auth.api.getSession({ + headers: await headers(), + }); + + let isPlatformAdmin = false; + if (session?.user?.id) { + const user = await db.user.findUnique({ + where: { id: session.user.id }, + select: { isPlatformAdmin: true }, + }); + isPlatformAdmin = user?.isPlatformAdmin ?? false; + } + + return ( + + }, + }, + { + label: formDefinition.title, + href: `/${orgId}/documents/${parsedFormType}`, + props: { render: }, + }, + { label: 'Submission Details', isCurrent: true }, + ]} + /> + + + + ); +} diff --git a/apps/app/src/app/(app)/[orgId]/documents/components/CompanyFormPageClient.tsx b/apps/app/src/app/(app)/[orgId]/documents/components/CompanyFormPageClient.tsx new file mode 100644 index 000000000..dafe1db6e --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/components/CompanyFormPageClient.tsx @@ -0,0 +1,300 @@ +'use client'; + +import { + evidenceFormDefinitions, + type EvidenceFormType, +} from '@/app/(app)/[orgId]/documents/forms'; +import { conciseFormDescriptions } from '@/app/(app)/[orgId]/documents/form-descriptions'; +import { api } from '@/lib/api-client'; +import { jwtManager } from '@/utils/jwt-manager'; +import { + Button, + Empty, + EmptyDescription, + EmptyHeader, + EmptyMedia, + EmptyTitle, + InputGroup, + InputGroupAddon, + InputGroupInput, + PageHeader, + Table, + TableBody, + TableCell, + TableHead, + TableHeader, + TableRow, + Text, +} from '@trycompai/design-system'; +import { Add, Catalog, Download, Search } from '@trycompai/design-system/icons'; +import Link from 'next/link'; +import { useRouter } from 'next/navigation'; +import { useState } from 'react'; +import { toast } from 'sonner'; +import useSWR from 'swr'; +import { StatusBadge, formatSubmissionDate } from './submission-utils'; + +// ─── Types ─────────────────────────────────────────────────── + +type EvidenceSubmissionRow = { + id: string; + submittedAt: string; + status: string; + data: Record; + submittedBy?: { + name: string | null; + email: string; + } | null; +}; + +type EvidenceFormResponse = { + form: (typeof evidenceFormDefinitions)[EvidenceFormType]; + submissions: EvidenceSubmissionRow[]; + total: number; +}; + +const submissionDateColumnWidth = 128; +const submittedByColumnWidth = 128; +const statusColumnWidth = 176; +const summaryColumnWidth = 280; + +// ─── Helpers ───────────────────────────────────────────────── + +function truncate(str: string, max: number) { + if (str.length <= max) return str; + return `${str.slice(0, max)}…`; +} + +function getMatrixRowCount(value: unknown): number { + if (!Array.isArray(value)) return 0; + return value.filter((row) => row && typeof row === 'object').length; +} + +// ─── Main Component ────────────────────────────────────────── + +export function CompanyFormPageClient({ + organizationId, + formType, +}: { + organizationId: string; + formType: EvidenceFormType; +}) { + const router = useRouter(); + const [search, setSearch] = useState(''); + const [isExporting, setIsExporting] = useState(false); + + const formDefinition = evidenceFormDefinitions[formType]; + const summaryField = formDefinition.fields.find((field) => field.type === 'textarea'); + const matrixSummaryField = formDefinition.fields.find((field) => field.type === 'matrix'); + const hasMatrixSummary = Boolean(matrixSummaryField); + const showSummaryColumn = Boolean(summaryField) || hasMatrixSummary; + + const query = search.trim() ? `?search=${encodeURIComponent(search.trim())}` : ''; + const swrKey: readonly [string, string] = [ + `/v1/evidence-forms/${formType}${query}`, + organizationId, + ]; + + const { data, isLoading } = useSWR( + swrKey, + async ([endpoint, orgId]: readonly [string, string]) => { + const response = await api.get(endpoint, orgId); + if (response.error || !response.data) { + throw new Error(response.error ?? 'Failed to load evidence form submissions'); + } + return response.data; + }, + ); + + const handleExportCsv = async () => { + if (!data || data.total === 0) { + toast.error('No submissions available to export'); + return; + } + + setIsExporting(true); + try { + const token = await jwtManager.getValidToken(); + if (!token) { + throw new Error('Authentication failed'); + } + + const response = await fetch( + `${process.env.NEXT_PUBLIC_API_URL ?? 'http://localhost:3333'}/v1/evidence-forms/${formType}/export.csv`, + { + method: 'GET', + headers: { + Authorization: `Bearer ${token}`, + 'X-Organization-Id': organizationId, + }, + credentials: 'include', + }, + ); + + if (!response.ok) { + throw new Error(await response.text()); + } + + const blob = await response.blob(); + const url = URL.createObjectURL(blob); + const anchor = document.createElement('a'); + anchor.href = url; + anchor.download = `${formType}-submissions.csv`; + document.body.appendChild(anchor); + anchor.click(); + document.body.removeChild(anchor); + URL.revokeObjectURL(url); + } catch (error) { + toast.error(error instanceof Error ? error.message : 'CSV export failed'); + } finally { + setIsExporting(false); + } + }; + + return ( +
+ + + + + +
+ } + /> +
+ + {conciseFormDescriptions[formType] ?? formDefinition.description} + +
+ + {/* ─── Submissions List ─── */} +
+
+ + + + + setSearch(event.target.value)} + /> + +
+ + {isLoading ? ( + + + + + + No submissions yet. + + Start by creating a new submission, click the New Submission button above. + + + + ) : !data || data.submissions.length === 0 ? ( + + + + + + No submissions yet + + Start by creating a new submission, click the New Submission button above. + + + + ) : ( + + + + + {formType === 'access-request' && } + {showSummaryColumn && } + + + + +
Submission Date
+
+ +
Submitted By
+
+ {formType === 'access-request' && ( + +
Status
+
+ )} + {showSummaryColumn && Summary} +
+
+ + {data.submissions.map((submission) => { + const summaryValue = summaryField + ? String(submission.data[summaryField.key] ?? '') + : ''; + const matrixSummary = matrixSummaryField + ? `${getMatrixRowCount(submission.data[matrixSummaryField.key])} row(s)` + : ''; + const rowSummary = summaryField ? truncate(summaryValue, 80) : matrixSummary; + + return ( + + router.push( + `/${organizationId}/documents/${formType}/submissions/${submission.id}`, + ) + } + style={{ cursor: 'pointer' }} + > + +
+ {formatSubmissionDate( + submission.data.submissionDate, + submission.submittedAt, + )} +
+
+ + + {submission.submittedBy?.name ?? submission.submittedBy?.email ?? 'Unknown'} + + + {formType === 'access-request' && ( + +
+ +
+
+ )} + {showSummaryColumn && ( + + + {rowSummary || '—'} + + + )} +
+ ); + })} +
+
+ )} +
+ + ); +} diff --git a/apps/app/src/app/(app)/[orgId]/documents/components/CompanyOverviewCards.tsx b/apps/app/src/app/(app)/[orgId]/documents/components/CompanyOverviewCards.tsx new file mode 100644 index 000000000..50e791e14 --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/components/CompanyOverviewCards.tsx @@ -0,0 +1,101 @@ +'use client'; + +import { conciseFormDescriptions } from '@/app/(app)/[orgId]/documents/form-descriptions'; +import { api } from '@/lib/api-client'; +import { + Badge, + Card, + CardContent, + CardDescription, + CardHeader, + CardTitle, + Stack, + Text, +} from '@trycompai/design-system'; +import Link from 'next/link'; +import useSWR from 'swr'; +import { evidenceFormDefinitionList } from '../forms'; + +type FormStatuses = Record; + +const SIX_MONTHS_MS = 6 * 30 * 24 * 60 * 60 * 1000; + +function isTodo(lastSubmittedAt: string | null): boolean { + if (!lastSubmittedAt) return true; + const elapsed = Date.now() - new Date(lastSubmittedAt).getTime(); + return elapsed > SIX_MONTHS_MS; +} + +export function CompanyOverviewCards({ organizationId }: { organizationId: string }) { + const swrKey: readonly [string, string] = ['/v1/evidence-forms/statuses', organizationId]; + + const { data: statuses } = useSWR( + swrKey, + async ([endpoint, orgId]: readonly [string, string]) => { + const response = await api.get(endpoint, orgId); + if (response.error || !response.data) { + throw new Error(response.error ?? 'Failed to load form statuses'); + } + return response.data; + }, + ); + + // Group forms by category + const categories = new Map(); + for (const form of evidenceFormDefinitionList) { + const cat = form.category; + if (!categories.has(cat)) { + categories.set(cat, []); + } + categories.get(cat)!.push(form); + } + + return ( + + {Array.from(categories.entries()).map(([category, forms]) => ( +
+
+ + {category} + + {forms.length} +
+
+ {forms.map((form) => { + const status = statuses?.[form.type]; + const showTodo = statuses ? isTodo(status?.lastSubmittedAt ?? null) : false; + + return ( + + + + {form.title} +
+ + {conciseFormDescriptions[form.type] ?? form.description} + +
+
+ + {statuses ? ( + showTodo ? ( + TODO + ) : ( + Complete + ) + ) : ( + + {form.fields.length} fields + + )} + +
+ + ); + })} +
+
+ ))} +
+ ); +} diff --git a/apps/app/src/app/(app)/[orgId]/documents/components/CompanySidebar.tsx b/apps/app/src/app/(app)/[orgId]/documents/components/CompanySidebar.tsx new file mode 100644 index 000000000..1ff6f77f0 --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/components/CompanySidebar.tsx @@ -0,0 +1,56 @@ +'use client'; + +import { AppShellNav, AppShellNavItem } from '@trycompai/design-system'; +import Link from 'next/link'; +import { usePathname } from 'next/navigation'; +import { evidenceFormDefinitionList } from '../forms'; + +interface CompanySidebarProps { + orgId: string; +} + +export function CompanySidebar({ orgId }: CompanySidebarProps) { + const pathname = usePathname() ?? ''; + + const isPathActive = (path: string) => { + if (path === `/${orgId}/documents`) { + return pathname === path; + } + return pathname.startsWith(path); + }; + + // Group forms by category preserving insertion order + const categories = new Map>(); + for (const form of evidenceFormDefinitionList) { + const cat = form.category; + if (!categories.has(cat)) { + categories.set(cat, []); + } + categories.get(cat)!.push({ + id: form.type, + label: form.title, + path: `/${orgId}/documents/${form.type}`, + }); + } + + return ( + + + Overview + + + {Array.from(categories.entries()).map(([category, items]) => ( +
+
+ {category} +
+ {items.map((item) => ( + + {item.label} + + ))} +
+ ))} +
+ ); +} diff --git a/apps/app/src/app/(app)/[orgId]/documents/components/CompanySubmissionDetailPageClient.tsx b/apps/app/src/app/(app)/[orgId]/documents/components/CompanySubmissionDetailPageClient.tsx new file mode 100644 index 000000000..88e964bc5 --- /dev/null +++ b/apps/app/src/app/(app)/[orgId]/documents/components/CompanySubmissionDetailPageClient.tsx @@ -0,0 +1,370 @@ +'use client'; + +import { + evidenceFormDefinitions, + type EvidenceFormType, +} from '@/app/(app)/[orgId]/documents/forms'; +import { api } from '@/lib/api-client'; +import { useActiveMember } from '@/utils/auth-client'; +import { + Button, + Empty, + EmptyMedia, + EmptyDescription, + EmptyHeader, + EmptyTitle, + Field, + FieldLabel, + Section, + Table, + TableBody, + TableCell, + TableHead, + TableHeader, + TableRow, + Text, + Textarea, +} from '@trycompai/design-system'; +import { Document } from '@trycompai/design-system/icons'; +import { useState } from 'react'; +import ReactMarkdown from 'react-markdown'; +import remarkGfm from 'remark-gfm'; +import { toast } from 'sonner'; +import useSWR from 'swr'; +import { + StatusBadge, + formatSubmissionDate, + isMatrixField, + normalizeMatrixRows, + renderSubmissionValue, +} from './submission-utils'; +import { DocumentFindingsSection } from './DocumentFindingsSection'; + +type EvidenceSubmissionRow = { + id: string; + submittedAt: string; + status: string; + reviewedAt?: string | null; + reviewReason?: string | null; + data: Record; + submittedBy?: { + name: string | null; + email: string; + } | null; + reviewedBy?: { + name: string | null; + email: string; + } | null; +}; + +type EvidenceSubmissionResponse = { + form: (typeof evidenceFormDefinitions)[EvidenceFormType]; + submission: EvidenceSubmissionRow; +}; + +function MarkdownPreview({ content }: { content: string }) { + return ( +
+ {content} +
+ ); +} + +export function CompanySubmissionDetailPageClient({ + organizationId, + formType, + submissionId, + isPlatformAdmin, +}: { + organizationId: string; + formType: EvidenceFormType; + submissionId: string; + isPlatformAdmin: boolean; +}) { + const endpoint = `/v1/evidence-forms/${formType}/submissions/${submissionId}`; + const swrKey: readonly [string, string] = [endpoint, organizationId]; + + const { data, isLoading, error, mutate } = useSWR( + swrKey, + async ([path, orgId]: readonly [string, string]) => { + const response = await api.get(path, orgId); + if (response.error || !response.data) { + throw new Error(response.error ?? 'Failed to load submission'); + } + return response.data; + }, + ); + + const [reviewReason, setReviewReason] = useState(''); + const [isSubmittingReview, setIsSubmittingReview] = useState(false); + const { data: activeMember } = useActiveMember(); + const memberRoles = activeMember?.role?.split(',').map((role: string) => role.trim()) || []; + const isAuditor = memberRoles.includes('auditor'); + const isAdminOrOwner = memberRoles.includes('admin') || memberRoles.includes('owner'); + + const handleReview = async (action: 'approved' | 'rejected') => { + if (action === 'rejected' && !reviewReason.trim()) { + toast.error('A reason is required when rejecting a submission'); + return; + } + + setIsSubmittingReview(true); + try { + const response = await api.patch( + `/v1/evidence-forms/${formType}/submissions/${submissionId}/review`, + { action, reason: reviewReason.trim() || undefined }, + organizationId, + ); + + if (response.error) { + toast.error(response.error); + return; + } + + toast.success(action === 'approved' ? 'Submission approved' : 'Submission rejected'); + setReviewReason(''); + mutate(); + } catch { + toast.error('Failed to submit review'); + } finally { + setIsSubmittingReview(false); + } + }; + + if (isLoading) { + return ( + + + + + + Loading submission... + Fetching the selected document details. + + + ); + } + + if (error || !data?.submission) { + return ( + + + Submission not found + + This submission may have been removed or you may not have access. + + + + ); + } + + const submission = data.submission; + const fields = data.form.fields.filter((field) => field.key !== 'submissionDate'); + const compactFields = fields.filter((f) => f.type === 'text' || f.type === 'date'); + const selectFields = fields.filter((f) => f.type === 'select'); + const textareaFields = fields.filter((f) => f.type === 'textarea'); + const fileFields = fields.filter((f) => f.type === 'file'); + const matrixFields = fields.filter(isMatrixField); + + return ( +
+
+
+
+ {formType === 'access-request' && ( +
+
+ Status +
+
+ +
+
+ )} +
+
+ Submission Date +
+
+ {formatSubmissionDate(submission.data.submissionDate, submission.submittedAt)} +
+
+
+
+ Submitted By +
+
+ {submission.submittedBy?.name ?? submission.submittedBy?.email ?? 'Unknown'} +
+
+ {formType === 'access-request' && submission.status !== 'pending' && ( + <> +
+
+ Reviewed By +
+
+ {submission.reviewedBy?.name ?? submission.reviewedBy?.email ?? '—'} +
+
+ {submission.reviewReason && ( +
+
+ Review Reason +
+
+ {submission.reviewReason} +
+
+ )} + + )} + {compactFields.map((field) => ( +
+
+ {field.label} +
+
+ {renderSubmissionValue(submission.data[field.key], field)} +
+
+ ))} + {selectFields.map((field) => ( +
+
+ {field.label} +
+
+ {renderSubmissionValue(submission.data[field.key], field)} +
+
+ ))} + {textareaFields.map((field) => { + const value = submission.data[field.key]; + const content = typeof value === 'string' ? value : ''; + return ( +
+
+ {field.label} +
+
+ {content ? ( + + ) : ( + + — + + )} +
+
+ ); + })} + {fileFields.map((field) => ( +
+
+ {field.label} +
+
+ {renderSubmissionValue(submission.data[field.key], field)} +
+
+ ))} + {matrixFields.map((field) => { + const rows = normalizeMatrixRows(submission.data[field.key]); + return ( +
+
+ {field.label} +
+
+ {rows.length === 0 ? ( + + — + + ) : ( + + + + {field.columns.map((column) => ( + + {column.label} + + ))} + + + + {rows.map((row, rowIndex) => ( + + {field.columns.map((column) => ( + +
{row[column.key] || '—'}
+
+ ))} +
+ ))} +
+
+ )} +
+
+ ); + })} +
+
+ + {/* Review action area — only for access requests */} + {formType === 'access-request' && submission.status === 'pending' && ( +
+
+
+ + Review this submission + +
+
+ + Reason (required for rejection) +