mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
[SM-567] Change how project permission is resolved (#2791)
* Change how project permission is resolved * Fix tests --------- Co-authored-by: Thomas Avery <43214426+Thomas-Avery@users.noreply.github.com>
This commit is contained in:
parent
15954fb679
commit
6a6b15fada
@ -1,7 +1,6 @@
|
|||||||
using System.Linq.Expressions;
|
using System.Linq.Expressions;
|
||||||
using AutoMapper;
|
using AutoMapper;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.SecretsManager.Models.Data;
|
|
||||||
using Bit.Core.SecretsManager.Repositories;
|
using Bit.Core.SecretsManager.Repositories;
|
||||||
using Bit.Infrastructure.EntityFramework.Repositories;
|
using Bit.Infrastructure.EntityFramework.Repositories;
|
||||||
using Bit.Infrastructure.EntityFramework.SecretsManager.Models;
|
using Bit.Infrastructure.EntityFramework.SecretsManager.Models;
|
||||||
@ -28,31 +27,6 @@ public class ProjectRepository : Repository<Core.SecretsManager.Entities.Project
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<ProjectPermissionDetails> GetPermissionDetailsByIdAsync(Guid id, Guid userId)
|
|
||||||
{
|
|
||||||
using var scope = ServiceScopeFactory.CreateScope();
|
|
||||||
var dbContext = GetDatabaseContext(scope);
|
|
||||||
|
|
||||||
var project = await dbContext.Project
|
|
||||||
.Where(c => c.Id == id && c.DeletedDate == null)
|
|
||||||
.Select(p => new ProjectPermissionDetails
|
|
||||||
{
|
|
||||||
Id = p.Id,
|
|
||||||
OrganizationId = p.OrganizationId,
|
|
||||||
Name = p.Name,
|
|
||||||
CreationDate = p.CreationDate,
|
|
||||||
RevisionDate = p.RevisionDate,
|
|
||||||
DeletedDate = p.DeletedDate,
|
|
||||||
Read = p.UserAccessPolicies.Any(ap => ap.OrganizationUser.User.Id == userId && ap.Read)
|
|
||||||
|| p.GroupAccessPolicies.Any(ap =>
|
|
||||||
ap.Group.GroupUsers.Any(gu => gu.OrganizationUser.User.Id == userId && ap.Read)),
|
|
||||||
Write = p.UserAccessPolicies.Any(ap => ap.OrganizationUser.User.Id == userId && ap.Write) ||
|
|
||||||
p.GroupAccessPolicies.Any(ap =>
|
|
||||||
ap.Group.GroupUsers.Any(gu => gu.OrganizationUser.User.Id == userId && ap.Write)),
|
|
||||||
}).FirstOrDefaultAsync();
|
|
||||||
return project;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<IEnumerable<Core.SecretsManager.Entities.Project>> GetManyByOrganizationIdAsync(Guid organizationId, Guid userId, AccessClientType accessType)
|
public async Task<IEnumerable<Core.SecretsManager.Entities.Project>> GetManyByOrganizationIdAsync(Guid organizationId, Guid userId, AccessClientType accessType)
|
||||||
{
|
{
|
||||||
using var scope = ServiceScopeFactory.CreateScope();
|
using var scope = ServiceScopeFactory.CreateScope();
|
||||||
@ -182,4 +156,37 @@ public class ProjectRepository : Repository<Core.SecretsManager.Entities.Project
|
|||||||
await dbContext.SaveChangesAsync();
|
await dbContext.SaveChangesAsync();
|
||||||
return projects;
|
return projects;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task<(bool Read, bool Write)> AccessToProjectAsync(Guid id, Guid userId, AccessClientType accessType)
|
||||||
|
{
|
||||||
|
using var scope = ServiceScopeFactory.CreateScope();
|
||||||
|
var dbContext = GetDatabaseContext(scope);
|
||||||
|
|
||||||
|
var projectQuery = dbContext.Project
|
||||||
|
.Where(s => s.Id == id);
|
||||||
|
|
||||||
|
var query = accessType switch
|
||||||
|
{
|
||||||
|
AccessClientType.NoAccessCheck => projectQuery.Select(_ => new { Read = true, Write = true }),
|
||||||
|
AccessClientType.User => projectQuery.Select(p => new
|
||||||
|
{
|
||||||
|
Read = p.UserAccessPolicies.Any(ap => ap.OrganizationUser.User.Id == userId && ap.Read)
|
||||||
|
|| p.GroupAccessPolicies.Any(ap =>
|
||||||
|
ap.Group.GroupUsers.Any(gu => gu.OrganizationUser.User.Id == userId && ap.Read)),
|
||||||
|
Write = p.UserAccessPolicies.Any(ap => ap.OrganizationUser.User.Id == userId && ap.Write) ||
|
||||||
|
p.GroupAccessPolicies.Any(ap =>
|
||||||
|
ap.Group.GroupUsers.Any(gu => gu.OrganizationUser.User.Id == userId && ap.Write)),
|
||||||
|
}),
|
||||||
|
AccessClientType.ServiceAccount => projectQuery.Select(p => new
|
||||||
|
{
|
||||||
|
Read = p.ServiceAccountAccessPolicies.Any(ap => ap.ServiceAccountId == userId && ap.Read),
|
||||||
|
Write = p.ServiceAccountAccessPolicies.Any(ap => ap.ServiceAccountId == userId && ap.Write),
|
||||||
|
}),
|
||||||
|
_ => projectQuery.Select(_ => new { Read = false, Write = false }),
|
||||||
|
};
|
||||||
|
|
||||||
|
var policy = await query.FirstOrDefaultAsync();
|
||||||
|
|
||||||
|
return (policy.Read, policy.Write);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -82,8 +82,7 @@ public class ProjectsController : Controller
|
|||||||
[HttpGet("projects/{id}")]
|
[HttpGet("projects/{id}")]
|
||||||
public async Task<ProjectPermissionDetailsResponseModel> GetAsync([FromRoute] Guid id)
|
public async Task<ProjectPermissionDetailsResponseModel> GetAsync([FromRoute] Guid id)
|
||||||
{
|
{
|
||||||
var userId = _userService.GetProperUserId(User).Value;
|
var project = await _projectRepository.GetByIdAsync(id);
|
||||||
var project = await _projectRepository.GetPermissionDetailsByIdAsync(id, userId);
|
|
||||||
if (project == null)
|
if (project == null)
|
||||||
{
|
{
|
||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
@ -94,34 +93,18 @@ public class ProjectsController : Controller
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
var orgAdmin = await _currentContext.OrganizationAdmin(project.OrganizationId);
|
var orgAdmin = await _currentContext.OrganizationAdmin(project.OrganizationId);
|
||||||
var accessClient = AccessClientHelper.ToAccessClient(_currentContext.ClientType, orgAdmin);
|
var accessClient = AccessClientHelper.ToAccessClient(_currentContext.ClientType, orgAdmin);
|
||||||
|
|
||||||
bool hasAccess;
|
var access = await _projectRepository.AccessToProjectAsync(id, userId, accessClient);
|
||||||
var read = project.Read;
|
|
||||||
var write = project.Write;
|
|
||||||
|
|
||||||
switch (accessClient)
|
if (!access.Read)
|
||||||
{
|
|
||||||
case AccessClientType.NoAccessCheck:
|
|
||||||
hasAccess = true;
|
|
||||||
write = true;
|
|
||||||
read = true;
|
|
||||||
break;
|
|
||||||
case AccessClientType.User:
|
|
||||||
hasAccess = project.Read;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
hasAccess = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!hasAccess)
|
|
||||||
{
|
{
|
||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ProjectPermissionDetailsResponseModel(project, read, write);
|
return new ProjectPermissionDetailsResponseModel(project, access.Read, access.Write);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPost("projects/delete")]
|
[HttpPost("projects/delete")]
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.SecretsManager.Entities;
|
using Bit.Core.SecretsManager.Entities;
|
||||||
using Bit.Core.SecretsManager.Models.Data;
|
|
||||||
|
|
||||||
namespace Bit.Core.SecretsManager.Repositories;
|
namespace Bit.Core.SecretsManager.Repositories;
|
||||||
|
|
||||||
@ -9,7 +8,6 @@ public interface IProjectRepository
|
|||||||
Task<IEnumerable<Project>> GetManyByOrganizationIdAsync(Guid organizationId, Guid userId, AccessClientType accessType);
|
Task<IEnumerable<Project>> GetManyByOrganizationIdAsync(Guid organizationId, Guid userId, AccessClientType accessType);
|
||||||
Task<IEnumerable<Project>> GetManyByOrganizationIdWriteAccessAsync(Guid organizationId, Guid userId, AccessClientType accessType);
|
Task<IEnumerable<Project>> GetManyByOrganizationIdWriteAccessAsync(Guid organizationId, Guid userId, AccessClientType accessType);
|
||||||
Task<IEnumerable<Project>> GetManyByIds(IEnumerable<Guid> ids);
|
Task<IEnumerable<Project>> GetManyByIds(IEnumerable<Guid> ids);
|
||||||
Task<ProjectPermissionDetails> GetPermissionDetailsByIdAsync(Guid id, Guid userId);
|
|
||||||
Task<Project> GetByIdAsync(Guid id);
|
Task<Project> GetByIdAsync(Guid id);
|
||||||
Task<Project> CreateAsync(Project project);
|
Task<Project> CreateAsync(Project project);
|
||||||
Task ReplaceAsync(Project project);
|
Task ReplaceAsync(Project project);
|
||||||
@ -19,4 +17,5 @@ public interface IProjectRepository
|
|||||||
Task<bool> UserHasWriteAccessToProject(Guid id, Guid userId);
|
Task<bool> UserHasWriteAccessToProject(Guid id, Guid userId);
|
||||||
Task<bool> ServiceAccountHasWriteAccessToProject(Guid id, Guid userId);
|
Task<bool> ServiceAccountHasWriteAccessToProject(Guid id, Guid userId);
|
||||||
Task<bool> ServiceAccountHasReadAccessToProject(Guid id, Guid userId);
|
Task<bool> ServiceAccountHasReadAccessToProject(Guid id, Guid userId);
|
||||||
|
Task<(bool Read, bool Write)> AccessToProjectAsync(Guid id, Guid userId, AccessClientType accessType);
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,6 @@ using Bit.Core.Enums;
|
|||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.SecretsManager.Commands.Projects.Interfaces;
|
using Bit.Core.SecretsManager.Commands.Projects.Interfaces;
|
||||||
using Bit.Core.SecretsManager.Entities;
|
using Bit.Core.SecretsManager.Entities;
|
||||||
using Bit.Core.SecretsManager.Models.Data;
|
|
||||||
using Bit.Core.SecretsManager.Repositories;
|
using Bit.Core.SecretsManager.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Core.Test.SecretsManager.AutoFixture.ProjectsFixture;
|
using Bit.Core.Test.SecretsManager.AutoFixture.ProjectsFixture;
|
||||||
@ -199,13 +198,16 @@ public class ProjectsControllerTests
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
sutProvider.GetDependency<IProjectRepository>().GetPermissionDetailsByIdAsync(Arg.Is(data), Arg.Any<Guid>())
|
sutProvider.GetDependency<IProjectRepository>().GetByIdAsync(Arg.Is(data))
|
||||||
.ReturnsForAnyArgs(new ProjectPermissionDetails() { Id = data, OrganizationId = orgId, Read = true, Write = true });
|
.ReturnsForAnyArgs(new Project { Id = data, OrganizationId = orgId });
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IProjectRepository>().AccessToProjectAsync(default, default, default)
|
||||||
|
.ReturnsForAnyArgs((true, false));
|
||||||
|
|
||||||
await sutProvider.Sut.GetAsync(data);
|
await sutProvider.Sut.GetAsync(data);
|
||||||
|
|
||||||
await sutProvider.GetDependency<IProjectRepository>().Received(1)
|
await sutProvider.GetDependency<IProjectRepository>().Received(1)
|
||||||
.GetPermissionDetailsByIdAsync(Arg.Is(data), Arg.Any<Guid>());
|
.GetByIdAsync(Arg.Is(data));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
[Theory]
|
||||||
|
Loading…
x
Reference in New Issue
Block a user