feat: add missing abilities (#354)

feat: add all missing abilities rules on resolvers
This commit is contained in:
Jérémy M
2023-06-22 20:09:17 +02:00
committed by GitHub
parent 4a2797c491
commit c4ad0171b0
21 changed files with 461 additions and 104 deletions

View File

@ -45,6 +45,11 @@ export class AbilityFactory {
); );
// User // User
can(AbilityAction.Read, 'User', {
workspaceMember: {
workspaceId: workspace.id,
},
});
can(AbilityAction.Update, 'User', { id: user.id }); can(AbilityAction.Update, 'User', { id: user.id });
cannot(AbilityAction.Delete, 'User'); cannot(AbilityAction.Delete, 'User');

View File

@ -2,14 +2,24 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { CommentThreadTargetWhereInput } from 'src/core/@generated/comment-thread-target/comment-thread-target-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
import { subject } from '@casl/ability';
class CommentThreadTargetArgs {
where?: CommentThreadTargetWhereInput;
}
@Injectable() @Injectable()
export class ManageCommentThreadTargetAbilityHandler export class ManageCommentThreadTargetAbilityHandler
implements IAbilityHandler implements IAbilityHandler
{ {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'CommentThreadTarget'); return ability.can(AbilityAction.Manage, 'CommentThreadTarget');
} }
@ -35,8 +45,21 @@ export class CreateCommentThreadTargetAbilityHandler
export class UpdateCommentThreadTargetAbilityHandler export class UpdateCommentThreadTargetAbilityHandler
implements IAbilityHandler implements IAbilityHandler
{ {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'CommentThreadTarget');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CommentThreadTargetArgs>();
const commentThreadTarget =
await this.prismaService.commentThreadTarget.findFirst({
where: args.where,
});
assert(commentThreadTarget, '', NotFoundException);
return ability.can(
AbilityAction.Update,
subject('CommentThreadTarget', commentThreadTarget),
);
} }
} }
@ -44,7 +67,20 @@ export class UpdateCommentThreadTargetAbilityHandler
export class DeleteCommentThreadTargetAbilityHandler export class DeleteCommentThreadTargetAbilityHandler
implements IAbilityHandler implements IAbilityHandler
{ {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'CommentThreadTarget');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CommentThreadTargetArgs>();
const commentThreadTarget =
await this.prismaService.commentThreadTarget.findFirst({
where: args.where,
});
assert(commentThreadTarget, '', NotFoundException);
return ability.can(
AbilityAction.Delete,
subject('CommentThreadTarget', commentThreadTarget),
);
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { GqlExecutionContext } from '@nestjs/graphql';
import { CommentThreadWhereInput } from 'src/core/@generated/comment-thread/comment-thread-where.input';
import { assert } from 'src/utils/assert';
import { subject } from '@casl/ability';
class CommentThreadArgs {
where?: CommentThreadWhereInput;
}
@Injectable() @Injectable()
export class ManageCommentThreadAbilityHandler implements IAbilityHandler { export class ManageCommentThreadAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'CommentThread'); return ability.can(AbilityAction.Manage, 'CommentThread');
} }
@ -29,14 +39,38 @@ export class CreateCommentThreadAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateCommentThreadAbilityHandler implements IAbilityHandler { export class UpdateCommentThreadAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'CommentThread');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CommentThreadArgs>();
const commentThread = await this.prismaService.commentThread.findFirst({
where: args.where,
});
assert(commentThread, '', NotFoundException);
return ability.can(
AbilityAction.Update,
subject('CommentThread', commentThread),
);
} }
} }
@Injectable() @Injectable()
export class DeleteCommentThreadAbilityHandler implements IAbilityHandler { export class DeleteCommentThreadAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'CommentThread');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CommentThreadArgs>();
const commentThread = await this.prismaService.commentThread.findFirst({
where: args.where,
});
assert(commentThread, '', NotFoundException);
return ability.can(
AbilityAction.Delete,
subject('CommentThread', commentThread),
);
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { subject } from '@casl/ability';
import { CommentWhereInput } from 'src/core/@generated/comment/comment-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
class CommentArgs {
where?: CommentWhereInput;
}
@Injectable() @Injectable()
export class ManageCommentAbilityHandler implements IAbilityHandler { export class ManageCommentAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'Comment'); return ability.can(AbilityAction.Manage, 'Comment');
} }
@ -29,14 +39,32 @@ export class CreateCommentAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateCommentAbilityHandler implements IAbilityHandler { export class UpdateCommentAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'Comment');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CommentArgs>();
const comment = await this.prismaService.comment.findFirst({
where: args.where,
});
assert(comment, '', NotFoundException);
return ability.can(AbilityAction.Update, subject('Comment', comment));
} }
} }
@Injectable() @Injectable()
export class DeleteCommentAbilityHandler implements IAbilityHandler { export class DeleteCommentAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'Comment');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CommentArgs>();
const comment = await this.prismaService.comment.findFirst({
where: args.where,
});
assert(comment, '', NotFoundException);
return ability.can(AbilityAction.Delete, subject('Comment', comment));
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { CompanyWhereInput } from 'src/core/@generated/company/company-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
import { subject } from '@casl/ability';
class CompanyArgs {
where?: CompanyWhereInput;
}
@Injectable() @Injectable()
export class ManageCompanyAbilityHandler implements IAbilityHandler { export class ManageCompanyAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'Company'); return ability.can(AbilityAction.Manage, 'Company');
} }
@ -29,14 +39,32 @@ export class CreateCompanyAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateCompanyAbilityHandler implements IAbilityHandler { export class UpdateCompanyAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'Company');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CompanyArgs>();
const company = await this.prismaService.company.findFirst({
where: args.where,
});
assert(company, '', NotFoundException);
return ability.can(AbilityAction.Update, subject('Company', company));
} }
} }
@Injectable() @Injectable()
export class DeleteCompanyAbilityHandler implements IAbilityHandler { export class DeleteCompanyAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'Company');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<CompanyArgs>();
const company = await this.prismaService.company.findFirst({
where: args.where,
});
assert(company, '', NotFoundException);
return ability.can(AbilityAction.Delete, subject('Company', company));
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { PersonWhereInput } from 'src/core/@generated/person/person-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
import { subject } from '@casl/ability';
class PersonArgs {
where?: PersonWhereInput;
}
@Injectable() @Injectable()
export class ManagePersonAbilityHandler implements IAbilityHandler { export class ManagePersonAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'Person'); return ability.can(AbilityAction.Manage, 'Person');
} }
@ -29,14 +39,32 @@ export class CreatePersonAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdatePersonAbilityHandler implements IAbilityHandler { export class UpdatePersonAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'Person');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<PersonArgs>();
const person = await this.prismaService.person.findFirst({
where: args.where,
});
assert(person, '', NotFoundException);
return ability.can(AbilityAction.Update, subject('Person', person));
} }
} }
@Injectable() @Injectable()
export class DeletePersonAbilityHandler implements IAbilityHandler { export class DeletePersonAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'Person');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<PersonArgs>();
const person = await this.prismaService.person.findFirst({
where: args.where,
});
assert(person, '', NotFoundException);
return ability.can(AbilityAction.Delete, subject('Person', person));
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { subject } from '@casl/ability';
import { RefreshTokenWhereInput } from 'src/core/@generated/refresh-token/refresh-token-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
class RefreshTokenArgs {
where?: RefreshTokenWhereInput;
}
@Injectable() @Injectable()
export class ManageRefreshTokenAbilityHandler implements IAbilityHandler { export class ManageRefreshTokenAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'RefreshToken'); return ability.can(AbilityAction.Manage, 'RefreshToken');
} }
@ -29,14 +39,38 @@ export class CreateRefreshTokenAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateRefreshTokenAbilityHandler implements IAbilityHandler { export class UpdateRefreshTokenAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'RefreshToken');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<RefreshTokenArgs>();
const refreshToken = await this.prismaService.refreshToken.findFirst({
where: args.where,
});
assert(refreshToken, '', NotFoundException);
return ability.can(
AbilityAction.Update,
subject('RefreshToken', refreshToken),
);
} }
} }
@Injectable() @Injectable()
export class DeleteRefreshTokenAbilityHandler implements IAbilityHandler { export class DeleteRefreshTokenAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'RefreshToken');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<RefreshTokenArgs>();
const refreshToken = await this.prismaService.refreshToken.findFirst({
where: args.where,
});
assert(refreshToken, '', NotFoundException);
return ability.can(
AbilityAction.Delete,
subject('RefreshToken', refreshToken),
);
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { assert } from 'src/utils/assert';
import { UserWhereInput } from 'src/core/@generated/user/user-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { subject } from '@casl/ability';
class UserArgs {
where?: UserWhereInput;
}
@Injectable() @Injectable()
export class ManageUserAbilityHandler implements IAbilityHandler { export class ManageUserAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'User'); return ability.can(AbilityAction.Manage, 'User');
} }
@ -29,14 +39,32 @@ export class CreateUserAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateUserAbilityHandler implements IAbilityHandler { export class UpdateUserAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'User');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<UserArgs>();
const user = await this.prismaService.user.findFirst({
where: args.where,
});
assert(user, '', NotFoundException);
return ability.can(AbilityAction.Update, subject('User', user));
} }
} }
@Injectable() @Injectable()
export class DeleteUserAbilityHandler implements IAbilityHandler { export class DeleteUserAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'User');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<UserArgs>();
const user = await this.prismaService.user.findFirst({
where: args.where,
});
assert(user, '', NotFoundException);
return ability.can(AbilityAction.Delete, subject('User', user));
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { subject } from '@casl/ability';
import { WorkspaceMemberWhereInput } from 'src/core/@generated/workspace-member/workspace-member-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
class WorksapceMemberArgs {
where?: WorkspaceMemberWhereInput;
}
@Injectable() @Injectable()
export class ManageWorkspaceMemberAbilityHandler implements IAbilityHandler { export class ManageWorkspaceMemberAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'WorkspaceMember'); return ability.can(AbilityAction.Manage, 'WorkspaceMember');
} }
@ -29,14 +39,38 @@ export class CreateWorkspaceMemberAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateWorkspaceMemberAbilityHandler implements IAbilityHandler { export class UpdateWorkspaceMemberAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'WorkspaceMember');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<WorksapceMemberArgs>();
const workspaceMember = await this.prismaService.workspaceMember.findFirst({
where: args.where,
});
assert(workspaceMember, '', NotFoundException);
return ability.can(
AbilityAction.Update,
subject('WorkspaceMember', workspaceMember),
);
} }
} }
@Injectable() @Injectable()
export class DeleteWorkspaceMemberAbilityHandler implements IAbilityHandler { export class DeleteWorkspaceMemberAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'WorkspaceMember');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<WorksapceMemberArgs>();
const workspaceMember = await this.prismaService.workspaceMember.findFirst({
where: args.where,
});
assert(workspaceMember, '', NotFoundException);
return ability.can(
AbilityAction.Delete,
subject('WorkspaceMember', workspaceMember),
);
} }
} }

View File

@ -2,12 +2,22 @@ import { PrismaService } from 'src/database/prisma.service';
import { AbilityAction } from '../ability.action'; import { AbilityAction } from '../ability.action';
import { AppAbility } from '../ability.factory'; import { AppAbility } from '../ability.factory';
import { IAbilityHandler } from '../interfaces/ability-handler.interface'; import { IAbilityHandler } from '../interfaces/ability-handler.interface';
import { Injectable } from '@nestjs/common'; import {
ExecutionContext,
Injectable,
NotFoundException,
} from '@nestjs/common';
import { subject } from '@casl/ability';
import { WorkspaceWhereInput } from 'src/core/@generated/workspace/workspace-where.input';
import { GqlExecutionContext } from '@nestjs/graphql';
import { assert } from 'src/utils/assert';
class WorksapceArgs {
where?: WorkspaceWhereInput;
}
@Injectable() @Injectable()
export class ManageWorkspaceAbilityHandler implements IAbilityHandler { export class ManageWorkspaceAbilityHandler implements IAbilityHandler {
constructor(private readonly prismaService: PrismaService) {}
async handle(ability: AppAbility) { async handle(ability: AppAbility) {
return ability.can(AbilityAction.Manage, 'Workspace'); return ability.can(AbilityAction.Manage, 'Workspace');
} }
@ -29,14 +39,32 @@ export class CreateWorkspaceAbilityHandler implements IAbilityHandler {
@Injectable() @Injectable()
export class UpdateWorkspaceAbilityHandler implements IAbilityHandler { export class UpdateWorkspaceAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Update, 'Workspace');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<WorksapceArgs>();
const workspace = await this.prismaService.workspace.findFirst({
where: args.where,
});
assert(workspace, '', NotFoundException);
return ability.can(AbilityAction.Update, subject('Workspace', workspace));
} }
} }
@Injectable() @Injectable()
export class DeleteWorkspaceAbilityHandler implements IAbilityHandler { export class DeleteWorkspaceAbilityHandler implements IAbilityHandler {
handle(ability: AppAbility) { constructor(private readonly prismaService: PrismaService) {}
return ability.can(AbilityAction.Delete, 'Workspace');
async handle(ability: AppAbility, context: ExecutionContext) {
const gqlContext = GqlExecutionContext.create(context);
const args = gqlContext.getArgs<WorksapceArgs>();
const workspace = await this.prismaService.workspace.findFirst({
where: args.where,
});
assert(workspace, '', NotFoundException);
return ability.can(AbilityAction.Delete, subject('Workspace', workspace));
} }
} }

View File

@ -4,6 +4,7 @@ import { CommentThreadService } from '../services/comment-thread.service';
import { CanActivate } from '@nestjs/common'; import { CanActivate } from '@nestjs/common';
import { CreateOneCommentGuard } from 'src/guards/create-one-comment.guard'; import { CreateOneCommentGuard } from 'src/guards/create-one-comment.guard';
import { CreateOneCommentThreadGuard } from 'src/guards/create-one-comment-thread.guard'; import { CreateOneCommentThreadGuard } from 'src/guards/create-one-comment-thread.guard';
import { AbilityFactory } from 'src/ability/ability.factory';
describe('CommentThreadResolver', () => { describe('CommentThreadResolver', () => {
let resolver: CommentThreadResolver; let resolver: CommentThreadResolver;
@ -18,6 +19,10 @@ describe('CommentThreadResolver', () => {
provide: CommentThreadService, provide: CommentThreadService,
useValue: {}, useValue: {},
}, },
{
provide: AbilityFactory,
useValue: {},
},
], ],
}) })
.overrideGuard(CreateOneCommentGuard) .overrideGuard(CreateOneCommentGuard)

View File

@ -8,13 +8,22 @@ import { CreateOneCommentThreadArgs } from '../../../core/@generated/comment-thr
import { CreateOneCommentThreadGuard } from '../../../guards/create-one-comment-thread.guard'; import { CreateOneCommentThreadGuard } from '../../../guards/create-one-comment-thread.guard';
import { FindManyCommentThreadArgs } from '../../../core/@generated/comment-thread/find-many-comment-thread.args'; import { FindManyCommentThreadArgs } from '../../../core/@generated/comment-thread/find-many-comment-thread.args';
import { CommentThreadService } from '../services/comment-thread.service'; import { CommentThreadService } from '../services/comment-thread.service';
import { prepareFindManyArgs } from 'src/utils/prepare-find-many';
import { UpdateOneCommentThreadArgs } from 'src/core/@generated/comment-thread/update-one-comment-thread.args'; import { UpdateOneCommentThreadArgs } from 'src/core/@generated/comment-thread/update-one-comment-thread.args';
import { Prisma } from '@prisma/client'; import { Prisma } from '@prisma/client';
import { import {
PrismaSelector, PrismaSelector,
PrismaSelect, PrismaSelect,
} from 'src/decorators/prisma-select.decorator'; } from 'src/decorators/prisma-select.decorator';
import { AbilityGuard } from 'src/guards/ability.guard';
import { CheckAbilities } from 'src/decorators/check-abilities.decorator';
import {
CreateCommentThreadAbilityHandler,
ReadCommentThreadAbilityHandler,
UpdateCommentThreadAbilityHandler,
} from 'src/ability/handlers/comment-thread.ability-handler';
import { UserAbility } from 'src/decorators/user-ability.decorator';
import { AppAbility } from 'src/ability/ability.factory';
import { accessibleBy } from '@casl/prisma';
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Resolver(() => CommentThread) @Resolver(() => CommentThread)
@ -25,6 +34,8 @@ export class CommentThreadResolver {
@Mutation(() => CommentThread, { @Mutation(() => CommentThread, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(CreateCommentThreadAbilityHandler)
async createOneCommentThread( async createOneCommentThread(
@Args() args: CreateOneCommentThreadArgs, @Args() args: CreateOneCommentThreadArgs,
@AuthWorkspace() workspace: Workspace, @AuthWorkspace() workspace: Workspace,
@ -53,6 +64,8 @@ export class CommentThreadResolver {
@Mutation(() => CommentThread, { @Mutation(() => CommentThread, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(UpdateCommentThreadAbilityHandler)
async updateOneCommentThread( async updateOneCommentThread(
@Args() args: UpdateOneCommentThreadArgs, @Args() args: UpdateOneCommentThreadArgs,
@PrismaSelector({ modelName: 'CommentThread' }) @PrismaSelector({ modelName: 'CommentThread' })
@ -67,19 +80,20 @@ export class CommentThreadResolver {
} }
@Query(() => [CommentThread]) @Query(() => [CommentThread])
@UseGuards(AbilityGuard)
@CheckAbilities(ReadCommentThreadAbilityHandler)
async findManyCommentThreads( async findManyCommentThreads(
@Args() args: FindManyCommentThreadArgs, @Args() args: FindManyCommentThreadArgs,
@AuthWorkspace() workspace: Workspace, @UserAbility() ability: AppAbility,
@PrismaSelector({ modelName: 'CommentThread' }) @PrismaSelector({ modelName: 'CommentThread' })
prismaSelect: PrismaSelect<'CommentThread'>, prismaSelect: PrismaSelect<'CommentThread'>,
): Promise<Partial<CommentThread>[]> { ): Promise<Partial<CommentThread>[]> {
const preparedArgs = prepareFindManyArgs<FindManyCommentThreadArgs>(
args,
workspace,
);
const result = await this.commentThreadService.findMany({ const result = await this.commentThreadService.findMany({
...preparedArgs, ...args,
where: {
...args.where,
AND: [accessibleBy(ability).CommentThread],
},
select: prismaSelect.value, select: prismaSelect.value,
}); });

View File

@ -3,6 +3,7 @@ import { CommentResolver } from './comment.resolver';
import { CommentService } from '../services/comment.service'; import { CommentService } from '../services/comment.service';
import { CreateOneCommentGuard } from 'src/guards/create-one-comment.guard'; import { CreateOneCommentGuard } from 'src/guards/create-one-comment.guard';
import { CanActivate } from '@nestjs/common'; import { CanActivate } from '@nestjs/common';
import { AbilityFactory } from 'src/ability/ability.factory';
describe('CommentResolver', () => { describe('CommentResolver', () => {
let resolver: CommentResolver; let resolver: CommentResolver;
@ -17,6 +18,10 @@ describe('CommentResolver', () => {
provide: CommentService, provide: CommentService,
useValue: {}, useValue: {},
}, },
{
provide: AbilityFactory,
useValue: {},
},
], ],
}) })
.overrideGuard(CreateOneCommentGuard) .overrideGuard(CreateOneCommentGuard)

View File

@ -12,6 +12,11 @@ import {
PrismaSelector, PrismaSelector,
PrismaSelect, PrismaSelect,
} from 'src/decorators/prisma-select.decorator'; } from 'src/decorators/prisma-select.decorator';
import { AbilityGuard } from 'src/guards/ability.guard';
import { CheckAbilities } from 'src/decorators/check-abilities.decorator';
import { CreateCommentAbilityHandler } from 'src/ability/handlers/comment.ability-handler';
import { AuthUser } from 'src/decorators/auth-user.decorator';
import { User } from 'src/core/@generated/user/user.model';
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Resolver(() => Comment) @Resolver(() => Comment)
@ -22,8 +27,11 @@ export class CommentResolver {
@Mutation(() => Comment, { @Mutation(() => Comment, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(CreateCommentAbilityHandler)
async createOneComment( async createOneComment(
@Args() args: CreateOneCommentArgs, @Args() args: CreateOneCommentArgs,
@AuthUser() user: User,
@AuthWorkspace() workspace: Workspace, @AuthWorkspace() workspace: Workspace,
@PrismaSelector({ modelName: 'Comment' }) @PrismaSelector({ modelName: 'Comment' })
prismaSelect: PrismaSelect<'Comment'>, prismaSelect: PrismaSelect<'Comment'>,

View File

@ -5,6 +5,7 @@ import { UpdateOneGuard } from 'src/guards/update-one.guard';
import { CanActivate } from '@nestjs/common'; import { CanActivate } from '@nestjs/common';
import { DeleteManyGuard } from 'src/guards/delete-many.guard'; import { DeleteManyGuard } from 'src/guards/delete-many.guard';
import { CreateOneGuard } from 'src/guards/create-one.guard'; import { CreateOneGuard } from 'src/guards/create-one.guard';
import { AbilityFactory } from 'src/ability/ability.factory';
describe('CompanyResolver', () => { describe('CompanyResolver', () => {
let resolver: CompanyResolver; let resolver: CompanyResolver;
@ -19,6 +20,10 @@ describe('CompanyResolver', () => {
provide: CompanyService, provide: CompanyService,
useValue: {}, useValue: {},
}, },
{
provide: AbilityFactory,
useValue: {},
},
], ],
}) })
.overrideGuard(UpdateOneGuard) .overrideGuard(UpdateOneGuard)

View File

@ -14,11 +14,21 @@ import { UpdateOneGuard } from '../../guards/update-one.guard';
import { DeleteManyGuard } from '../../guards/delete-many.guard'; import { DeleteManyGuard } from '../../guards/delete-many.guard';
import { CreateOneGuard } from '../../guards/create-one.guard'; import { CreateOneGuard } from '../../guards/create-one.guard';
import { CompanyService } from './company.service'; import { CompanyService } from './company.service';
import { prepareFindManyArgs } from 'src/utils/prepare-find-many';
import { import {
PrismaSelect, PrismaSelect,
PrismaSelector, PrismaSelector,
} from 'src/decorators/prisma-select.decorator'; } from 'src/decorators/prisma-select.decorator';
import { AbilityGuard } from 'src/guards/ability.guard';
import { CheckAbilities } from 'src/decorators/check-abilities.decorator';
import {
CreateCompanyAbilityHandler,
DeleteCompanyAbilityHandler,
ReadCompanyAbilityHandler,
} from 'src/ability/handlers/company.ability-handler';
import { UserAbility } from 'src/decorators/user-ability.decorator';
import { AppAbility } from 'src/ability/ability.factory';
import { accessibleBy } from '@casl/prisma';
import { UpdateCommentAbilityHandler } from 'src/ability/handlers/comment.ability-handler';
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Resolver(() => Company) @Resolver(() => Company)
@ -26,18 +36,20 @@ export class CompanyResolver {
constructor(private readonly companyService: CompanyService) {} constructor(private readonly companyService: CompanyService) {}
@Query(() => [Company]) @Query(() => [Company])
@UseGuards(AbilityGuard)
@CheckAbilities(ReadCompanyAbilityHandler)
async findManyCompany( async findManyCompany(
@Args() args: FindManyCompanyArgs, @Args() args: FindManyCompanyArgs,
@AuthWorkspace() workspace: Workspace, @UserAbility() ability: AppAbility,
@PrismaSelector({ modelName: 'Company' }) @PrismaSelector({ modelName: 'Company' })
prismaSelect: PrismaSelect<'Company'>, prismaSelect: PrismaSelect<'Company'>,
): Promise<Partial<Company>[]> { ): Promise<Partial<Company>[]> {
const preparedArgs = prepareFindManyArgs<FindManyCompanyArgs>(
args,
workspace,
);
return this.companyService.findMany({ return this.companyService.findMany({
...preparedArgs, ...args,
where: {
...args.where,
AND: [accessibleBy(ability).Company],
},
select: prismaSelect.value, select: prismaSelect.value,
}); });
} }
@ -46,6 +58,8 @@ export class CompanyResolver {
@Mutation(() => Company, { @Mutation(() => Company, {
nullable: true, nullable: true,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(UpdateCommentAbilityHandler)
async updateOneCompany( async updateOneCompany(
@Args() args: UpdateOneCompanyArgs, @Args() args: UpdateOneCompanyArgs,
@PrismaSelector({ modelName: 'Company' }) @PrismaSelector({ modelName: 'Company' })
@ -65,6 +79,8 @@ export class CompanyResolver {
@Mutation(() => AffectedRows, { @Mutation(() => AffectedRows, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(DeleteCompanyAbilityHandler)
async deleteManyCompany( async deleteManyCompany(
@Args() args: DeleteManyCompanyArgs, @Args() args: DeleteManyCompanyArgs,
): Promise<AffectedRows> { ): Promise<AffectedRows> {
@ -77,6 +93,8 @@ export class CompanyResolver {
@Mutation(() => Company, { @Mutation(() => Company, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(CreateCompanyAbilityHandler)
async createOneCompany( async createOneCompany(
@Args() args: CreateOneCompanyArgs, @Args() args: CreateOneCompanyArgs,
@AuthWorkspace() workspace: Workspace, @AuthWorkspace() workspace: Workspace,

View File

@ -5,6 +5,7 @@ import { UpdateOneGuard } from 'src/guards/update-one.guard';
import { CanActivate } from '@nestjs/common'; import { CanActivate } from '@nestjs/common';
import { DeleteManyGuard } from 'src/guards/delete-many.guard'; import { DeleteManyGuard } from 'src/guards/delete-many.guard';
import { CreateOneGuard } from 'src/guards/create-one.guard'; import { CreateOneGuard } from 'src/guards/create-one.guard';
import { AbilityFactory } from 'src/ability/ability.factory';
describe('PersonResolver', () => { describe('PersonResolver', () => {
let resolver: PersonResolver; let resolver: PersonResolver;
@ -19,6 +20,10 @@ describe('PersonResolver', () => {
provide: PersonService, provide: PersonService,
useValue: {}, useValue: {},
}, },
{
provide: AbilityFactory,
useValue: {},
},
], ],
}) })
.overrideGuard(UpdateOneGuard) .overrideGuard(UpdateOneGuard)

View File

@ -14,11 +14,21 @@ import { UpdateOneGuard } from '../../guards/update-one.guard';
import { DeleteManyGuard } from '../../guards/delete-many.guard'; import { DeleteManyGuard } from '../../guards/delete-many.guard';
import { CreateOneGuard } from '../../guards/create-one.guard'; import { CreateOneGuard } from '../../guards/create-one.guard';
import { PersonService } from './person.service'; import { PersonService } from './person.service';
import { prepareFindManyArgs } from 'src/utils/prepare-find-many';
import { import {
PrismaSelect, PrismaSelect,
PrismaSelector, PrismaSelector,
} from 'src/decorators/prisma-select.decorator'; } from 'src/decorators/prisma-select.decorator';
import { AbilityGuard } from 'src/guards/ability.guard';
import { CheckAbilities } from 'src/decorators/check-abilities.decorator';
import {
CreatePersonAbilityHandler,
DeletePersonAbilityHandler,
ReadPersonAbilityHandler,
UpdatePersonAbilityHandler,
} from 'src/ability/handlers/person.ability-handler';
import { UserAbility } from 'src/decorators/user-ability.decorator';
import { AppAbility } from 'src/ability/ability.factory';
import { accessibleBy } from '@casl/prisma';
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Resolver(() => Person) @Resolver(() => Person)
@ -28,19 +38,20 @@ export class PersonResolver {
@Query(() => [Person], { @Query(() => [Person], {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(ReadPersonAbilityHandler)
async findManyPerson( async findManyPerson(
@Args() args: FindManyPersonArgs, @Args() args: FindManyPersonArgs,
@AuthWorkspace() workspace: Workspace, @UserAbility() ability: AppAbility,
@PrismaSelector({ modelName: 'Person' }) @PrismaSelector({ modelName: 'Person' })
prismaSelect: PrismaSelect<'Person'>, prismaSelect: PrismaSelect<'Person'>,
): Promise<Partial<Person>[]> { ): Promise<Partial<Person>[]> {
const preparedArgs = prepareFindManyArgs<FindManyPersonArgs>(
args,
workspace,
);
return this.personService.findMany({ return this.personService.findMany({
...preparedArgs, ...args,
where: {
...args.where,
AND: [accessibleBy(ability).Person],
},
select: prismaSelect.value, select: prismaSelect.value,
}); });
} }
@ -49,6 +60,8 @@ export class PersonResolver {
@Mutation(() => Person, { @Mutation(() => Person, {
nullable: true, nullable: true,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(UpdatePersonAbilityHandler)
async updateOnePerson( async updateOnePerson(
@Args() args: UpdateOnePersonArgs, @Args() args: UpdateOnePersonArgs,
@PrismaSelector({ modelName: 'Person' }) @PrismaSelector({ modelName: 'Person' })
@ -68,6 +81,8 @@ export class PersonResolver {
@Mutation(() => AffectedRows, { @Mutation(() => AffectedRows, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(DeletePersonAbilityHandler)
async deleteManyPerson( async deleteManyPerson(
@Args() args: DeleteManyPersonArgs, @Args() args: DeleteManyPersonArgs,
): Promise<AffectedRows> { ): Promise<AffectedRows> {
@ -80,6 +95,8 @@ export class PersonResolver {
@Mutation(() => Person, { @Mutation(() => Person, {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(CreatePersonAbilityHandler)
async createOnePerson( async createOnePerson(
@Args() args: CreateOnePersonArgs, @Args() args: CreateOnePersonArgs,
@AuthWorkspace() workspace: Workspace, @AuthWorkspace() workspace: Workspace,

View File

@ -1,6 +1,7 @@
import { Test, TestingModule } from '@nestjs/testing'; import { Test, TestingModule } from '@nestjs/testing';
import { UserResolver } from './user.resolver'; import { UserResolver } from './user.resolver';
import { UserService } from './user.service'; import { UserService } from './user.service';
import { AbilityFactory } from 'src/ability/ability.factory';
describe('UserResolver', () => { describe('UserResolver', () => {
let resolver: UserResolver; let resolver: UserResolver;
@ -13,6 +14,10 @@ describe('UserResolver', () => {
provide: UserService, provide: UserService,
useValue: {}, useValue: {},
}, },
{
provide: AbilityFactory,
useValue: {},
},
], ],
}).compile(); }).compile();

View File

@ -11,6 +11,12 @@ import {
PrismaSelect, PrismaSelect,
PrismaSelector, PrismaSelector,
} from 'src/decorators/prisma-select.decorator'; } from 'src/decorators/prisma-select.decorator';
import { AbilityGuard } from 'src/guards/ability.guard';
import { CheckAbilities } from 'src/decorators/check-abilities.decorator';
import { ReadUserAbilityHandler } from 'src/ability/handlers/user.ability-handler';
import { UserAbility } from 'src/decorators/user-ability.decorator';
import { AppAbility } from 'src/ability/ability.factory';
import { accessibleBy } from '@casl/prisma';
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Resolver(() => User) @Resolver(() => User)
@ -21,9 +27,12 @@ export class UserResolver {
@Query(() => [User], { @Query(() => [User], {
nullable: false, nullable: false,
}) })
@UseGuards(AbilityGuard)
@CheckAbilities(ReadUserAbilityHandler)
async findManyUser( async findManyUser(
@Args() args: FindManyUserArgs, @Args() args: FindManyUserArgs,
@AuthWorkspace() workspace: Workspace, @AuthWorkspace() workspace: Workspace,
@UserAbility() ability: AppAbility,
@PrismaSelector({ modelName: 'User' }) @PrismaSelector({ modelName: 'User' })
prismaSelect: PrismaSelect<'User'>, prismaSelect: PrismaSelect<'User'>,
): Promise<Partial<User>[]> { ): Promise<Partial<User>[]> {
@ -31,9 +40,7 @@ export class UserResolver {
...args, ...args,
where: { where: {
...args.where, ...args.where,
workspaceMember: { AND: [accessibleBy(ability).User],
is: { workspace: { is: { id: { equals: workspace.id } } } },
},
}, },
select: prismaSelect.value, select: prismaSelect.value,
}); });

View File

@ -1,15 +0,0 @@
import { Workspace } from '@prisma/client';
type FindManyArgsType = { where?: object; orderBy?: object };
export const prepareFindManyArgs = <T extends FindManyArgsType>(
args: T,
workspace: Workspace,
): T => {
args.where = {
...args.where,
...{ workspace: { is: { id: { equals: workspace.id } } } },
};
return args;
};