mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
Ps 976 moving of read only organization collection items to different folder not possible (#2257)
* PS-976 - update PutPartial endpoint to return cipher info, update Cipher_Move sproc to allow users to update a cipher's folder even if they don't have edit permissions * PS-976- fix formatting errors * PS-976 - per cr feedback updated EF query to match cipher_move sproc update, and updated cipher tests to align with existing tests
This commit is contained in:
parent
b5d5e6f65a
commit
b938abab65
@ -272,11 +272,16 @@ public class CiphersController : Controller
|
|||||||
|
|
||||||
[HttpPut("{id}/partial")]
|
[HttpPut("{id}/partial")]
|
||||||
[HttpPost("{id}/partial")]
|
[HttpPost("{id}/partial")]
|
||||||
public async Task PutPartial(string id, [FromBody] CipherPartialRequestModel model)
|
public async Task<CipherResponseModel> PutPartial(string id, [FromBody] CipherPartialRequestModel model)
|
||||||
{
|
{
|
||||||
var userId = _userService.GetProperUserId(User).Value;
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
var folderId = string.IsNullOrWhiteSpace(model.FolderId) ? null : (Guid?)new Guid(model.FolderId);
|
var folderId = string.IsNullOrWhiteSpace(model.FolderId) ? null : (Guid?)new Guid(model.FolderId);
|
||||||
await _cipherRepository.UpdatePartialAsync(new Guid(id), userId, folderId, model.Favorite);
|
var cipherId = new Guid(id);
|
||||||
|
await _cipherRepository.UpdatePartialAsync(cipherId, userId, folderId, model.Favorite);
|
||||||
|
|
||||||
|
var cipher = await _cipherRepository.GetByIdAsync(cipherId, userId);
|
||||||
|
var response = new CipherResponseModel(cipher, _globalSettings);
|
||||||
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPut("{id}/share")]
|
[HttpPut("{id}/share")]
|
||||||
|
@ -345,7 +345,6 @@ public class CipherRepository : Repository<Core.Entities.Cipher, Cipher, Guid>,
|
|||||||
var idsToMove = from ucd in userCipherDetails
|
var idsToMove = from ucd in userCipherDetails
|
||||||
join c in cipherEntities
|
join c in cipherEntities
|
||||||
on ucd.Id equals c.Id
|
on ucd.Id equals c.Id
|
||||||
where ucd.Edit
|
|
||||||
select c;
|
select c;
|
||||||
await idsToMove.ForEachAsync(cipher =>
|
await idsToMove.ForEachAsync(cipher =>
|
||||||
{
|
{
|
||||||
|
@ -15,8 +15,7 @@ BEGIN
|
|||||||
FROM
|
FROM
|
||||||
[dbo].[UserCipherDetails](@UserId)
|
[dbo].[UserCipherDetails](@UserId)
|
||||||
WHERE
|
WHERE
|
||||||
[Edit] = 1
|
[Id] IN (SELECT * FROM @Ids)
|
||||||
AND [Id] IN (SELECT * FROM @Ids)
|
|
||||||
)
|
)
|
||||||
UPDATE
|
UPDATE
|
||||||
[dbo].[Cipher]
|
[dbo].[Cipher]
|
||||||
|
45
test/Api.Test/Controllers/CiphersControllerTests.cs
Normal file
45
test/Api.Test/Controllers/CiphersControllerTests.cs
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
using System.Security.Claims;
|
||||||
|
using Bit.Api.Controllers;
|
||||||
|
using Bit.Api.Models.Request;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using Core.Models.Data;
|
||||||
|
using NSubstitute;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Api.Test.Controllers;
|
||||||
|
|
||||||
|
[ControllerCustomize(typeof(CiphersController))]
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class CiphersControllerTests
|
||||||
|
{
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task PutPartialShouldReturnCipherWithGivenFolderAndFavoriteValues(Guid userId, Guid folderId, SutProvider<CiphersController> sutProvider)
|
||||||
|
{
|
||||||
|
var isFavorite = true;
|
||||||
|
var cipherId = Guid.NewGuid();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IUserService>()
|
||||||
|
.GetProperUserId(Arg.Any<ClaimsPrincipal>())
|
||||||
|
.Returns(userId);
|
||||||
|
|
||||||
|
var cipherDetails = new CipherDetails
|
||||||
|
{
|
||||||
|
Favorite = isFavorite,
|
||||||
|
FolderId = folderId,
|
||||||
|
Type = Core.Enums.CipherType.SecureNote,
|
||||||
|
Data = "{}"
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<ICipherRepository>()
|
||||||
|
.GetByIdAsync(cipherId, userId)
|
||||||
|
.Returns(Task.FromResult(cipherDetails));
|
||||||
|
|
||||||
|
var result = await sutProvider.Sut.PutPartial(cipherId.ToString(), new CipherPartialRequestModel { Favorite = isFavorite, FolderId = folderId.ToString() });
|
||||||
|
|
||||||
|
Assert.Equal(folderId.ToString(), result.FolderId);
|
||||||
|
Assert.Equal(isFavorite, result.Favorite);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,39 @@
|
|||||||
|
-- Remove check for Edit permission. User should be able to move the cipher to a different folder even if they don't have Edit permissions
|
||||||
|
|
||||||
|
ALTER PROCEDURE [dbo].[Cipher_Move]
|
||||||
|
@Ids AS [dbo].[GuidIdArray] READONLY,
|
||||||
|
@FolderId AS UNIQUEIDENTIFIER,
|
||||||
|
@UserId AS UNIQUEIDENTIFIER
|
||||||
|
AS
|
||||||
|
BEGIN
|
||||||
|
SET NOCOUNT ON
|
||||||
|
|
||||||
|
DECLARE @UserIdKey VARCHAR(50) = CONCAT('"', @UserId, '"')
|
||||||
|
DECLARE @UserIdPath VARCHAR(50) = CONCAT('$.', @UserIdKey)
|
||||||
|
|
||||||
|
;WITH [IdsToMoveCTE] AS (
|
||||||
|
SELECT
|
||||||
|
[Id]
|
||||||
|
FROM
|
||||||
|
[dbo].[UserCipherDetails](@UserId)
|
||||||
|
WHERE
|
||||||
|
[Id] IN (SELECT * FROM @Ids)
|
||||||
|
)
|
||||||
|
UPDATE
|
||||||
|
[dbo].[Cipher]
|
||||||
|
SET
|
||||||
|
[Folders] =
|
||||||
|
CASE
|
||||||
|
WHEN @FolderId IS NOT NULL AND [Folders] IS NULL THEN
|
||||||
|
CONCAT('{', @UserIdKey, ':"', @FolderId, '"', '}')
|
||||||
|
WHEN @FolderId IS NOT NULL THEN
|
||||||
|
JSON_MODIFY([Folders], @UserIdPath, CAST(@FolderId AS VARCHAR(50)))
|
||||||
|
ELSE
|
||||||
|
JSON_MODIFY([Folders], @UserIdPath, NULL)
|
||||||
|
END
|
||||||
|
WHERE
|
||||||
|
[Id] IN (SELECT * FROM [IdsToMoveCTE])
|
||||||
|
|
||||||
|
EXEC [dbo].[User_BumpAccountRevisionDate] @UserId
|
||||||
|
END
|
||||||
|
GO
|
Loading…
x
Reference in New Issue
Block a user