2020-02-20 20:36:31 +01:00
|
|
|
|
using System;
|
2019-07-06 05:35:54 +02:00
|
|
|
|
using System.Collections.Generic;
|
2022-01-21 15:36:25 +01:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text.Json;
|
|
|
|
|
using System.Threading.Tasks;
|
2022-01-11 10:40:51 +01:00
|
|
|
|
using Bit.Core.Entities;
|
2022-04-01 22:08:47 +02:00
|
|
|
|
using Bit.Core.Enums;
|
|
|
|
|
using Bit.Core.Models;
|
2022-01-21 15:36:25 +01:00
|
|
|
|
using Bit.Core.Models.Business;
|
2019-07-06 05:35:54 +02:00
|
|
|
|
using Bit.Core.Services;
|
2022-01-21 15:36:25 +01:00
|
|
|
|
using Bit.Test.Common.AutoFixture;
|
|
|
|
|
using Bit.Test.Common.AutoFixture.Attributes;
|
|
|
|
|
using Bit.Test.Common.Helpers;
|
2019-07-06 05:35:54 +02:00
|
|
|
|
using Microsoft.AspNetCore.Identity;
|
|
|
|
|
using NSubstitute;
|
2022-04-01 22:08:47 +02:00
|
|
|
|
using NSubstitute.ReceivedExtensions;
|
2019-07-06 05:35:54 +02:00
|
|
|
|
using Xunit;
|
|
|
|
|
|
|
|
|
|
namespace Bit.Core.Test.Services
|
|
|
|
|
{
|
|
|
|
|
public class UserServiceTests
|
|
|
|
|
{
|
2022-01-21 15:36:25 +01:00
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task UpdateLicenseAsync_Success(SutProvider<UserService> sutProvider,
|
|
|
|
|
User user, UserLicense userLicense)
|
|
|
|
|
{
|
|
|
|
|
using var tempDir = new TempDirectory();
|
2019-07-06 05:35:54 +02:00
|
|
|
|
|
2022-01-21 15:36:25 +01:00
|
|
|
|
var now = DateTime.UtcNow;
|
|
|
|
|
userLicense.Issued = now.AddDays(-10);
|
|
|
|
|
userLicense.Expires = now.AddDays(10);
|
|
|
|
|
userLicense.Version = 1;
|
|
|
|
|
userLicense.Premium = true;
|
2019-07-06 05:35:54 +02:00
|
|
|
|
|
2022-01-21 15:36:25 +01:00
|
|
|
|
user.EmailVerified = true;
|
|
|
|
|
user.Email = userLicense.Email;
|
2019-07-06 05:35:54 +02:00
|
|
|
|
|
2022-01-21 15:36:25 +01:00
|
|
|
|
sutProvider.GetDependency<Settings.GlobalSettings>().SelfHosted = true;
|
|
|
|
|
sutProvider.GetDependency<Settings.GlobalSettings>().LicenseDirectory = tempDir.Directory;
|
|
|
|
|
sutProvider.GetDependency<ILicensingService>()
|
|
|
|
|
.VerifyLicense(userLicense)
|
|
|
|
|
.Returns(true);
|
2019-07-06 05:35:54 +02:00
|
|
|
|
|
2022-01-21 15:36:25 +01:00
|
|
|
|
await sutProvider.Sut.UpdateLicenseAsync(user, userLicense);
|
|
|
|
|
|
|
|
|
|
var filePath = Path.Combine(tempDir.Directory, "user", $"{user.Id}.json");
|
|
|
|
|
Assert.True(File.Exists(filePath));
|
|
|
|
|
var document = JsonDocument.Parse(File.OpenRead(filePath));
|
|
|
|
|
var root = document.RootElement;
|
|
|
|
|
Assert.Equal(JsonValueKind.Object, root.ValueKind);
|
|
|
|
|
// Sort of a lazy way to test that it is indented but not sure of a better way
|
|
|
|
|
Assert.Contains('\n', root.GetRawText());
|
|
|
|
|
AssertHelper.AssertJsonProperty(root, "LicenseKey", JsonValueKind.String);
|
|
|
|
|
AssertHelper.AssertJsonProperty(root, "Id", JsonValueKind.String);
|
|
|
|
|
AssertHelper.AssertJsonProperty(root, "Premium", JsonValueKind.True);
|
|
|
|
|
var versionProp = AssertHelper.AssertJsonProperty(root, "Version", JsonValueKind.Number);
|
|
|
|
|
Assert.Equal(1, versionProp.GetInt32());
|
2019-07-06 05:35:54 +02:00
|
|
|
|
}
|
2022-04-01 22:08:47 +02:00
|
|
|
|
|
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task SendTwoFactorEmailAsync_Success(SutProvider<UserService> sutProvider, User user)
|
|
|
|
|
{
|
|
|
|
|
var email = user.Email.ToLowerInvariant();
|
|
|
|
|
var token = "thisisatokentocompare";
|
|
|
|
|
|
|
|
|
|
var userTwoFactorTokenProvider = Substitute.For<IUserTwoFactorTokenProvider<User>>();
|
|
|
|
|
userTwoFactorTokenProvider
|
|
|
|
|
.CanGenerateTwoFactorTokenAsync(Arg.Any<UserManager<User>>(), user)
|
|
|
|
|
.Returns(Task.FromResult(true));
|
|
|
|
|
userTwoFactorTokenProvider
|
|
|
|
|
.GenerateAsync("2faEmail:" + email, Arg.Any<UserManager<User>>(), user)
|
|
|
|
|
.Returns(Task.FromResult(token));
|
|
|
|
|
|
|
|
|
|
sutProvider.Sut.RegisterTokenProvider("Email", userTwoFactorTokenProvider);
|
|
|
|
|
|
|
|
|
|
user.SetTwoFactorProviders(new Dictionary<TwoFactorProviderType, TwoFactorProvider>
|
|
|
|
|
{
|
|
|
|
|
[TwoFactorProviderType.Email] = new TwoFactorProvider
|
|
|
|
|
{
|
|
|
|
|
MetaData = new Dictionary<string, object> { ["Email"] = email },
|
|
|
|
|
Enabled = true
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
await sutProvider.Sut.SendTwoFactorEmailAsync(user);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<IMailService>()
|
|
|
|
|
.Received(1)
|
|
|
|
|
.SendTwoFactorEmailAsync(email, token);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task SendTwoFactorEmailBecauseNewDeviceLoginAsync_Success(SutProvider<UserService> sutProvider, User user)
|
|
|
|
|
{
|
|
|
|
|
var email = user.Email.ToLowerInvariant();
|
|
|
|
|
var token = "thisisatokentocompare";
|
|
|
|
|
|
|
|
|
|
var userTwoFactorTokenProvider = Substitute.For<IUserTwoFactorTokenProvider<User>>();
|
|
|
|
|
userTwoFactorTokenProvider
|
|
|
|
|
.CanGenerateTwoFactorTokenAsync(Arg.Any<UserManager<User>>(), user)
|
|
|
|
|
.Returns(Task.FromResult(true));
|
|
|
|
|
userTwoFactorTokenProvider
|
|
|
|
|
.GenerateAsync("2faEmail:" + email, Arg.Any<UserManager<User>>(), user)
|
|
|
|
|
.Returns(Task.FromResult(token));
|
|
|
|
|
|
|
|
|
|
sutProvider.Sut.RegisterTokenProvider("Email", userTwoFactorTokenProvider);
|
|
|
|
|
|
|
|
|
|
user.SetTwoFactorProviders(new Dictionary<TwoFactorProviderType, TwoFactorProvider>
|
|
|
|
|
{
|
|
|
|
|
[TwoFactorProviderType.Email] = new TwoFactorProvider
|
|
|
|
|
{
|
|
|
|
|
MetaData = new Dictionary<string, object> { ["Email"] = email },
|
|
|
|
|
Enabled = true
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
await sutProvider.Sut.SendTwoFactorEmailAsync(user, true);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<IMailService>()
|
|
|
|
|
.Received(1)
|
|
|
|
|
.SendNewDeviceLoginTwoFactorEmailAsync(email, token);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task SendTwoFactorEmailAsync_ExceptionBecauseNoProviderOnUser(SutProvider<UserService> sutProvider, User user)
|
|
|
|
|
{
|
|
|
|
|
user.TwoFactorProviders = null;
|
|
|
|
|
|
|
|
|
|
await Assert.ThrowsAsync<ArgumentNullException>("No email.", () => sutProvider.Sut.SendTwoFactorEmailAsync(user));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task SendTwoFactorEmailAsync_ExceptionBecauseNoProviderMetadataOnUser(SutProvider<UserService> sutProvider, User user)
|
|
|
|
|
{
|
|
|
|
|
user.SetTwoFactorProviders(new Dictionary<TwoFactorProviderType, TwoFactorProvider>
|
|
|
|
|
{
|
|
|
|
|
[TwoFactorProviderType.Email] = new TwoFactorProvider
|
|
|
|
|
{
|
|
|
|
|
MetaData = null,
|
|
|
|
|
Enabled = true
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await Assert.ThrowsAsync<ArgumentNullException>("No email.", () => sutProvider.Sut.SendTwoFactorEmailAsync(user));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task SendTwoFactorEmailAsync_ExceptionBecauseNoProviderEmailMetadataOnUser(SutProvider<UserService> sutProvider, User user)
|
|
|
|
|
{
|
|
|
|
|
user.SetTwoFactorProviders(new Dictionary<TwoFactorProviderType, TwoFactorProvider>
|
|
|
|
|
{
|
|
|
|
|
[TwoFactorProviderType.Email] = new TwoFactorProvider
|
|
|
|
|
{
|
|
|
|
|
MetaData = new Dictionary<string, object> { ["qweqwe"] = user.Email.ToLowerInvariant() },
|
|
|
|
|
Enabled = true
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await Assert.ThrowsAsync<ArgumentNullException>("No email.", () => sutProvider.Sut.SendTwoFactorEmailAsync(user));
|
|
|
|
|
}
|
2019-07-06 05:35:54 +02:00
|
|
|
|
}
|
|
|
|
|
}
|