using System.Text; using System.Text.Json; using Bit.Core.Entities; using Bit.Core.Enums; using Bit.Core.Exceptions; using Bit.Core.Models.Data; using Bit.Core.Models.Data.Organizations.Policies; using Bit.Core.Repositories; using Bit.Core.Services; using Bit.Core.Test.AutoFixture.CurrentContextFixtures; using Bit.Core.Test.AutoFixture.SendFixtures; using Bit.Core.Test.Entities; using Bit.Test.Common.AutoFixture; using Bit.Test.Common.AutoFixture.Attributes; using Microsoft.AspNetCore.Identity; using NSubstitute; using Xunit; namespace Bit.Core.Test.Services; [SutProviderCustomize] [CurrentContextCustomize] [UserSendCustomize] public class SendServiceTests { private void SaveSendAsync_Setup(SendType sendType, bool disableSendPolicyAppliesToUser, SutProvider sutProvider, Send send) { send.Id = default; send.Type = sendType; sutProvider.GetDependency().GetCountByTypeApplicableToUserIdAsync( Arg.Any(), PolicyType.DisableSend).Returns(disableSendPolicyAppliesToUser ? 1 : 0); } // Disable Send policy check [Theory] [BitAutoData(SendType.File)] [BitAutoData(SendType.Text)] public async void SaveSendAsync_DisableSend_Applies_throws(SendType sendType, SutProvider sutProvider, Send send) { SaveSendAsync_Setup(sendType, disableSendPolicyAppliesToUser: true, sutProvider, send); await Assert.ThrowsAsync(() => sutProvider.Sut.SaveSendAsync(send)); } [Theory] [BitAutoData(SendType.File)] [BitAutoData(SendType.Text)] public async void SaveSendAsync_DisableSend_DoesntApply_success(SendType sendType, SutProvider sutProvider, Send send) { SaveSendAsync_Setup(sendType, disableSendPolicyAppliesToUser: false, sutProvider, send); await sutProvider.Sut.SaveSendAsync(send); await sutProvider.GetDependency().Received(1).CreateAsync(send); } // Send Options Policy - Disable Hide Email check private void SaveSendAsync_HideEmail_Setup(bool disableHideEmailAppliesToUser, SutProvider sutProvider, Send send, Policy policy) { send.HideEmail = true; var sendOptions = new SendOptionsPolicyData { DisableHideEmail = disableHideEmailAppliesToUser }; policy.Data = JsonSerializer.Serialize(sendOptions, new JsonSerializerOptions { PropertyNamingPolicy = JsonNamingPolicy.CamelCase, }); sutProvider.GetDependency().GetManyByTypeApplicableToUserIdAsync( Arg.Any(), PolicyType.SendOptions).Returns(new List { policy, }); } [Theory] [BitAutoData(SendType.File)] [BitAutoData(SendType.Text)] public async void SaveSendAsync_DisableHideEmail_Applies_throws(SendType sendType, SutProvider sutProvider, Send send, Policy policy) { SaveSendAsync_Setup(sendType, false, sutProvider, send); SaveSendAsync_HideEmail_Setup(true, sutProvider, send, policy); await Assert.ThrowsAsync(() => sutProvider.Sut.SaveSendAsync(send)); } [Theory] [BitAutoData(SendType.File)] [BitAutoData(SendType.Text)] public async void SaveSendAsync_DisableHideEmail_DoesntApply_success(SendType sendType, SutProvider sutProvider, Send send, Policy policy) { SaveSendAsync_Setup(sendType, false, sutProvider, send); SaveSendAsync_HideEmail_Setup(false, sutProvider, send, policy); await sutProvider.Sut.SaveSendAsync(send); await sutProvider.GetDependency().Received(1).CreateAsync(send); } [Theory] [BitAutoData] public async void SaveSendAsync_ExistingSend_Updates(SutProvider sutProvider, Send send) { send.Id = Guid.NewGuid(); var now = DateTime.UtcNow; await sutProvider.Sut.SaveSendAsync(send); Assert.True(send.RevisionDate - now < TimeSpan.FromSeconds(1)); await sutProvider.GetDependency() .Received(1) .UpsertAsync(send); await sutProvider.GetDependency() .Received(1) .PushSyncSendUpdateAsync(send); } [Theory] [BitAutoData] public async void SaveFileSendAsync_TextType_ThrowsBadRequest(SutProvider sutProvider, Send send) { send.Type = SendType.Text; var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 0) ); Assert.Contains("not of type \"file\"", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_EmptyFile_ThrowsBadRequest(SutProvider sutProvider, Send send) { send.Type = SendType.File; var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 0) ); Assert.Contains("no file data", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_UserCannotAccessPremium_ThrowsBadRequest(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(false); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 1) ); Assert.Contains("must have premium", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_UserHasUnconfirmedEmail_ThrowsBadRequest(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = false, }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 1) ); Assert.Contains("must confirm your email", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_UserCanAccessPremium_HasNoStorage_ThrowsBadRequest(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = true, Premium = true, MaxStorageGb = null, Storage = 0, }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 1) ); Assert.Contains("not enough storage", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_UserCanAccessPremium_StorageFull_ThrowsBadRequest(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = true, Premium = true, MaxStorageGb = 2, Storage = 2 * UserTests.Multiplier, }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 1) ); Assert.Contains("not enough storage", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_UserCanAccessPremium_IsNotPremium_IsSelfHosted_GiantFile_ThrowsBadRequest(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = true, Premium = false, }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); sutProvider.GetDependency() .SelfHosted = true; var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 11000 * UserTests.Multiplier) ); Assert.Contains("not enough storage", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_UserCanAccessPremium_IsNotPremium_IsNotSelfHosted_TwoGigabyteFile_ThrowsBadRequest(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = true, Premium = false, }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); sutProvider.GetDependency() .SelfHosted = false; var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 2 * UserTests.Multiplier) ); Assert.Contains("not enough storage", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_ThroughOrg_MaxStorageIsNull_ThrowsBadRequest(SutProvider sutProvider, Send send) { var org = new Organization { Id = Guid.NewGuid(), MaxStorageGb = null, }; send.UserId = null; send.OrganizationId = org.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(org.Id) .Returns(org); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 1) ); Assert.Contains("organization cannot use file sends", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_ThroughOrg_MaxStorageIsNull_TwoGBFile_ThrowsBadRequest(SutProvider sutProvider, Send send) { var org = new Organization { Id = Guid.NewGuid(), MaxStorageGb = null, }; send.UserId = null; send.OrganizationId = org.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(org.Id) .Returns(org); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 1) ); Assert.Contains("organization cannot use file sends", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_ThroughOrg_MaxStorageIsOneGB_TwoGBFile_ThrowsBadRequest(SutProvider sutProvider, Send send) { var org = new Organization { Id = Guid.NewGuid(), MaxStorageGb = 1, }; send.UserId = null; send.OrganizationId = org.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(org.Id) .Returns(org); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, null, 2 * UserTests.Multiplier) ); Assert.Contains("not enough storage", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void SaveFileSendAsync_HasEnouphStorage_Success(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = true, MaxStorageGb = 10, }; var data = new SendFileData { }; send.UserId = user.Id; send.Type = SendType.File; var testUrl = "https://test.com/"; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); sutProvider.GetDependency() .GetSendFileUploadUrlAsync(send, Arg.Any()) .Returns(testUrl); var utcNow = DateTime.UtcNow; var url = await sutProvider.Sut.SaveFileSendAsync(send, data, 1 * UserTests.Multiplier); Assert.Equal(testUrl, url); Assert.True(send.RevisionDate - utcNow < TimeSpan.FromSeconds(1)); await sutProvider.GetDependency() .Received(1) .GetSendFileUploadUrlAsync(send, Arg.Any()); await sutProvider.GetDependency() .Received(1) .UpsertAsync(send); await sutProvider.GetDependency() .Received(1) .PushSyncSendUpdateAsync(send); } [Theory] [BitAutoData] public async void SaveFileSendAsync_HasEnouphStorage_SendFileThrows_CleansUp(SutProvider sutProvider, Send send) { var user = new User { Id = Guid.NewGuid(), EmailVerified = true, MaxStorageGb = 10, }; var data = new SendFileData { }; send.UserId = user.Id; send.Type = SendType.File; sutProvider.GetDependency() .GetByIdAsync(user.Id) .Returns(user); sutProvider.GetDependency() .CanAccessPremium(user) .Returns(true); sutProvider.GetDependency() .GetSendFileUploadUrlAsync(send, Arg.Any()) .Returns(callInfo => throw new Exception("Problem")); var utcNow = DateTime.UtcNow; var exception = await Assert.ThrowsAsync(() => sutProvider.Sut.SaveFileSendAsync(send, data, 1 * UserTests.Multiplier) ); Assert.True(send.RevisionDate - utcNow < TimeSpan.FromSeconds(1)); Assert.Equal("Problem", exception.Message); await sutProvider.GetDependency() .Received(1) .GetSendFileUploadUrlAsync(send, Arg.Any()); await sutProvider.GetDependency() .Received(1) .UpsertAsync(send); await sutProvider.GetDependency() .Received(1) .PushSyncSendUpdateAsync(send); await sutProvider.GetDependency() .Received(1) .DeleteFileAsync(send, Arg.Any()); } [Theory] [BitAutoData] public async void UpdateFileToExistingSendAsync_SendNull_ThrowsBadRequest(SutProvider sutProvider) { var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.UploadFileToExistingSendAsync(new MemoryStream(), null) ); Assert.Contains("does not have file data", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void UpdateFileToExistingSendAsync_SendDataNull_ThrowsBadRequest(SutProvider sutProvider, Send send) { send.Data = null; var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.UploadFileToExistingSendAsync(new MemoryStream(), send) ); Assert.Contains("does not have file data", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void UpdateFileToExistingSendAsync_NotFileType_ThrowsBadRequest(SutProvider sutProvider, Send send) { var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.UploadFileToExistingSendAsync(new MemoryStream(), send) ); Assert.Contains("not a file type send", badRequest.Message, StringComparison.InvariantCultureIgnoreCase); } [Theory] [BitAutoData] public async void UpdateFileToExistingSendAsync_Success(SutProvider sutProvider, Send send) { var fileContents = "Test file content"; var sendFileData = new SendFileData { Id = "TEST", Size = fileContents.Length, Validated = false, }; send.Type = SendType.File; send.Data = JsonSerializer.Serialize(sendFileData); sutProvider.GetDependency() .ValidateFileAsync(send, sendFileData.Id, sendFileData.Size, Arg.Any()) .Returns((true, sendFileData.Size)); await sutProvider.Sut.UploadFileToExistingSendAsync(new MemoryStream(Encoding.UTF8.GetBytes(fileContents)), send); } [Theory] [BitAutoData] public async void UpdateFileToExistingSendAsync_InvalidSize(SutProvider sutProvider, Send send) { var fileContents = "Test file content"; var sendFileData = new SendFileData { Id = "TEST", Size = fileContents.Length, }; send.Type = SendType.File; send.Data = JsonSerializer.Serialize(sendFileData); sutProvider.GetDependency() .ValidateFileAsync(send, sendFileData.Id, sendFileData.Size, Arg.Any()) .Returns((false, sendFileData.Size)); var badRequest = await Assert.ThrowsAsync(() => sutProvider.Sut.UploadFileToExistingSendAsync(new MemoryStream(Encoding.UTF8.GetBytes(fileContents)), send) ); } [Theory] [BitAutoData] public void SendCanBeAccessed_Success(SutProvider sutProvider, Send send) { var now = DateTime.UtcNow; send.MaxAccessCount = 10; send.AccessCount = 5; send.ExpirationDate = now.AddYears(1); send.DeletionDate = now.AddYears(1); send.Disabled = false; sutProvider.GetDependency>() .VerifyHashedPassword(Arg.Any(), send.Password, "TEST") .Returns(PasswordVerificationResult.Success); var (grant, passwordRequiredError, passwordInvalidError) = sutProvider.Sut.SendCanBeAccessed(send, "TEST"); Assert.True(grant); Assert.False(passwordRequiredError); Assert.False(passwordInvalidError); } [Theory] [BitAutoData] public void SendCanBeAccessed_NullMaxAccess_Success(SutProvider sutProvider, Send send) { var now = DateTime.UtcNow; send.MaxAccessCount = null; send.AccessCount = 5; send.ExpirationDate = now.AddYears(1); send.DeletionDate = now.AddYears(1); send.Disabled = false; sutProvider.GetDependency>() .VerifyHashedPassword(Arg.Any(), send.Password, "TEST") .Returns(PasswordVerificationResult.Success); var (grant, passwordRequiredError, passwordInvalidError) = sutProvider.Sut.SendCanBeAccessed(send, "TEST"); Assert.True(grant); Assert.False(passwordRequiredError); Assert.False(passwordInvalidError); } [Theory] [BitAutoData] public void SendCanBeAccessed_NullSend_DoesNotGrantAccess(SutProvider sutProvider) { sutProvider.GetDependency>() .VerifyHashedPassword(Arg.Any(), "TEST", "TEST") .Returns(PasswordVerificationResult.Success); var (grant, passwordRequiredError, passwordInvalidError) = sutProvider.Sut.SendCanBeAccessed(null, "TEST"); Assert.False(grant); Assert.False(passwordRequiredError); Assert.False(passwordInvalidError); } [Theory] [BitAutoData] public void SendCanBeAccessed_NullPassword_PasswordRequiredErrorReturnsTrue(SutProvider sutProvider, Send send) { var now = DateTime.UtcNow; send.MaxAccessCount = null; send.AccessCount = 5; send.ExpirationDate = now.AddYears(1); send.DeletionDate = now.AddYears(1); send.Disabled = false; send.Password = "HASH"; sutProvider.GetDependency>() .VerifyHashedPassword(Arg.Any(), "TEST", "TEST") .Returns(PasswordVerificationResult.Success); var (grant, passwordRequiredError, passwordInvalidError) = sutProvider.Sut.SendCanBeAccessed(send, null); Assert.False(grant); Assert.True(passwordRequiredError); Assert.False(passwordInvalidError); } [Theory] [BitAutoData] public void SendCanBeAccessed_RehashNeeded_RehashesPassword(SutProvider sutProvider, Send send) { var now = DateTime.UtcNow; send.MaxAccessCount = null; send.AccessCount = 5; send.ExpirationDate = now.AddYears(1); send.DeletionDate = now.AddYears(1); send.Disabled = false; send.Password = "TEST"; sutProvider.GetDependency>() .VerifyHashedPassword(Arg.Any(), "TEST", "TEST") .Returns(PasswordVerificationResult.SuccessRehashNeeded); var (grant, passwordRequiredError, passwordInvalidError) = sutProvider.Sut.SendCanBeAccessed(send, "TEST"); sutProvider.GetDependency>() .Received(1) .HashPassword(Arg.Any(), "TEST"); Assert.True(grant); Assert.False(passwordRequiredError); Assert.False(passwordInvalidError); } [Theory] [BitAutoData] public void SendCanBeAccessed_VerifyFailed_PasswordInvalidReturnsTrue(SutProvider sutProvider, Send send) { var now = DateTime.UtcNow; send.MaxAccessCount = null; send.AccessCount = 5; send.ExpirationDate = now.AddYears(1); send.DeletionDate = now.AddYears(1); send.Disabled = false; send.Password = "TEST"; sutProvider.GetDependency>() .VerifyHashedPassword(Arg.Any(), "TEST", "TEST") .Returns(PasswordVerificationResult.Failed); var (grant, passwordRequiredError, passwordInvalidError) = sutProvider.Sut.SendCanBeAccessed(send, "TEST"); Assert.False(grant); Assert.False(passwordRequiredError); Assert.True(passwordInvalidError); } }