1
0
mirror of https://github.com/bitwarden/server.git synced 2025-04-05 05:00:19 -05:00

[PM-11249] Update cipher revision date when an attachment is added or deleted (#4873)

* update the cipher revision date when an attachment is added or deleted

* store the updated cipher in the DB when an attachment is altered

* return cipher from delete attachment endpoint
This commit is contained in:
Nick Krantz 2025-01-28 09:49:51 -06:00 committed by GitHub
parent a9a12301af
commit 4e1e514e83
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 35 additions and 7 deletions

View File

@ -1097,7 +1097,7 @@ public class CiphersController : Controller
[HttpDelete("{id}/attachment/{attachmentId}")] [HttpDelete("{id}/attachment/{attachmentId}")]
[HttpPost("{id}/attachment/{attachmentId}/delete")] [HttpPost("{id}/attachment/{attachmentId}/delete")]
public async Task DeleteAttachment(Guid id, string attachmentId) public async Task<DeleteAttachmentResponseData> DeleteAttachment(Guid id, string attachmentId)
{ {
var userId = _userService.GetProperUserId(User).Value; var userId = _userService.GetProperUserId(User).Value;
var cipher = await GetByIdAsync(id, userId); var cipher = await GetByIdAsync(id, userId);
@ -1106,7 +1106,7 @@ public class CiphersController : Controller
throw new NotFoundException(); throw new NotFoundException();
} }
await _cipherService.DeleteAttachmentAsync(cipher, attachmentId, userId, false); return await _cipherService.DeleteAttachmentAsync(cipher, attachmentId, userId, false);
} }
[HttpDelete("{id}/attachment/{attachmentId}/admin")] [HttpDelete("{id}/attachment/{attachmentId}/admin")]

View File

@ -0,0 +1,13 @@
using Bit.Core.Vault.Entities;
namespace Bit.Core.Vault.Models.Data;
public class DeleteAttachmentResponseData
{
public Cipher Cipher { get; set; }
public DeleteAttachmentResponseData(Cipher cipher)
{
Cipher = cipher;
}
}

View File

@ -17,7 +17,7 @@ public interface ICipherService
string attachmentId, Guid organizationShareId); string attachmentId, Guid organizationShareId);
Task DeleteAsync(Cipher cipher, Guid deletingUserId, bool orgAdmin = false); Task DeleteAsync(Cipher cipher, Guid deletingUserId, bool orgAdmin = false);
Task DeleteManyAsync(IEnumerable<Guid> cipherIds, Guid deletingUserId, Guid? organizationId = null, bool orgAdmin = false); Task DeleteManyAsync(IEnumerable<Guid> cipherIds, Guid deletingUserId, Guid? organizationId = null, bool orgAdmin = false);
Task DeleteAttachmentAsync(Cipher cipher, string attachmentId, Guid deletingUserId, bool orgAdmin = false); Task<DeleteAttachmentResponseData> DeleteAttachmentAsync(Cipher cipher, string attachmentId, Guid deletingUserId, bool orgAdmin = false);
Task PurgeAsync(Guid organizationId); Task PurgeAsync(Guid organizationId);
Task MoveManyAsync(IEnumerable<Guid> cipherIds, Guid? destinationFolderId, Guid movingUserId); Task MoveManyAsync(IEnumerable<Guid> cipherIds, Guid? destinationFolderId, Guid movingUserId);
Task SaveFolderAsync(Folder folder); Task SaveFolderAsync(Folder folder);

View File

@ -210,6 +210,11 @@ public class CipherService : ICipherService
AttachmentData = JsonSerializer.Serialize(data) AttachmentData = JsonSerializer.Serialize(data)
}); });
cipher.AddAttachment(attachmentId, data); cipher.AddAttachment(attachmentId, data);
// Update the revision date when an attachment is added
cipher.RevisionDate = DateTime.UtcNow;
await _cipherRepository.ReplaceAsync((CipherDetails)cipher);
await _pushService.PushSyncCipherUpdateAsync(cipher, null); await _pushService.PushSyncCipherUpdateAsync(cipher, null);
return (attachmentId, uploadUrl); return (attachmentId, uploadUrl);
@ -259,6 +264,10 @@ public class CipherService : ICipherService
throw; throw;
} }
// Update the revision date when an attachment is added
cipher.RevisionDate = DateTime.UtcNow;
await _cipherRepository.ReplaceAsync((CipherDetails)cipher);
// push // push
await _pushService.PushSyncCipherUpdateAsync(cipher, null); await _pushService.PushSyncCipherUpdateAsync(cipher, null);
} }
@ -441,7 +450,7 @@ public class CipherService : ICipherService
await _pushService.PushSyncCiphersAsync(deletingUserId); await _pushService.PushSyncCiphersAsync(deletingUserId);
} }
public async Task DeleteAttachmentAsync(Cipher cipher, string attachmentId, Guid deletingUserId, public async Task<DeleteAttachmentResponseData> DeleteAttachmentAsync(Cipher cipher, string attachmentId, Guid deletingUserId,
bool orgAdmin = false) bool orgAdmin = false)
{ {
if (!orgAdmin && !(await UserCanEditAsync(cipher, deletingUserId))) if (!orgAdmin && !(await UserCanEditAsync(cipher, deletingUserId)))
@ -454,7 +463,7 @@ public class CipherService : ICipherService
throw new NotFoundException(); throw new NotFoundException();
} }
await DeleteAttachmentAsync(cipher, cipher.GetAttachments()[attachmentId]); return await DeleteAttachmentAsync(cipher, cipher.GetAttachments()[attachmentId]);
} }
public async Task PurgeAsync(Guid organizationId) public async Task PurgeAsync(Guid organizationId)
@ -834,11 +843,11 @@ public class CipherService : ICipherService
} }
} }
private async Task DeleteAttachmentAsync(Cipher cipher, CipherAttachment.MetaData attachmentData) private async Task<DeleteAttachmentResponseData> DeleteAttachmentAsync(Cipher cipher, CipherAttachment.MetaData attachmentData)
{ {
if (attachmentData == null || string.IsNullOrWhiteSpace(attachmentData.AttachmentId)) if (attachmentData == null || string.IsNullOrWhiteSpace(attachmentData.AttachmentId))
{ {
return; return null;
} }
await _cipherRepository.DeleteAttachmentAsync(cipher.Id, attachmentData.AttachmentId); await _cipherRepository.DeleteAttachmentAsync(cipher.Id, attachmentData.AttachmentId);
@ -846,8 +855,14 @@ public class CipherService : ICipherService
await _attachmentStorageService.DeleteAttachmentAsync(cipher.Id, attachmentData); await _attachmentStorageService.DeleteAttachmentAsync(cipher.Id, attachmentData);
await _eventService.LogCipherEventAsync(cipher, Bit.Core.Enums.EventType.Cipher_AttachmentDeleted); await _eventService.LogCipherEventAsync(cipher, Bit.Core.Enums.EventType.Cipher_AttachmentDeleted);
// Update the revision date when an attachment is deleted
cipher.RevisionDate = DateTime.UtcNow;
await _cipherRepository.ReplaceAsync((CipherDetails)cipher);
// push // push
await _pushService.PushSyncCipherUpdateAsync(cipher, null); await _pushService.PushSyncCipherUpdateAsync(cipher, null);
return new DeleteAttachmentResponseData(cipher);
} }
private async Task ValidateCipherEditForAttachmentAsync(Cipher cipher, Guid savingUserId, bool orgAdmin, private async Task ValidateCipherEditForAttachmentAsync(Cipher cipher, Guid savingUserId, bool orgAdmin,