Introduces a capability layer on top of existing roles that controls feature visibility and access. Capabilities follow an area.feature.action taxonomy (~35 capabilities) with sensible defaults per role. Tenant admins can customize via grant/revoke overrides stored in org settings JSONB. Key changes: - Add vice_president role to DB schema - Backend: capability constants, resolution logic, CapabilityGuard (global), @RequireCapability decorator on all 16 tenant controllers - Frontend: permission hooks (useCanEdit, useHasCapability), CapabilityGate component, sidebar filtering by capability, all 17 pages migrated from useIsReadOnly to capability-based checks - New admin UI: /settings/permissions matrix page for per-tenant role customization with grant/revoke delta model - GET /organizations/my-capabilities endpoint for capability refresh - Validation of permissionOverrides in settings updates Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
42 lines
1.4 KiB
TypeScript
42 lines
1.4 KiB
TypeScript
import { Controller, Get, Post, Put, Body, Param, UseGuards } from '@nestjs/common';
|
|
import { ApiTags, ApiBearerAuth } from '@nestjs/swagger';
|
|
import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard';
|
|
import { RequireCapability } from '../../common/decorators/capability.decorator';
|
|
import { AssessmentGroupsService } from './assessment-groups.service';
|
|
|
|
@ApiTags('assessment-groups')
|
|
@Controller('assessment-groups')
|
|
@ApiBearerAuth()
|
|
@UseGuards(JwtAuthGuard)
|
|
export class AssessmentGroupsController {
|
|
constructor(private service: AssessmentGroupsService) {}
|
|
|
|
@Get()
|
|
@RequireCapability('assessments.groups.view')
|
|
findAll() { return this.service.findAll(); }
|
|
|
|
@Get('summary')
|
|
@RequireCapability('assessments.groups.view')
|
|
getSummary() { return this.service.getSummary(); }
|
|
|
|
@Get('default')
|
|
@RequireCapability('assessments.groups.view')
|
|
getDefault() { return this.service.getDefault(); }
|
|
|
|
@Get(':id')
|
|
@RequireCapability('assessments.groups.view')
|
|
findOne(@Param('id') id: string) { return this.service.findOne(id); }
|
|
|
|
@Post()
|
|
@RequireCapability('assessments.groups.edit')
|
|
create(@Body() dto: any) { return this.service.create(dto); }
|
|
|
|
@Put(':id')
|
|
@RequireCapability('assessments.groups.edit')
|
|
update(@Param('id') id: string, @Body() dto: any) { return this.service.update(id, dto); }
|
|
|
|
@Put(':id/set-default')
|
|
@RequireCapability('assessments.groups.edit')
|
|
setDefault(@Param('id') id: string) { return this.service.setDefault(id); }
|
|
}
|