using FluentAssertions; using Microsoft.AspNetCore.TestHost; using Microsoft.Extensions.DependencyInjection; using System; using System.Collections.Generic; using System.Net.Http; using System.Text.Json; using System.Text.Json.Serialization; using System.Threading.Tasks; using Timeline.Models.Converters; using Timeline.Models.Http; using Timeline.Services; using Timeline.Tests.Helpers; using Xunit; namespace Timeline.Tests.IntegratedTests { public abstract class IntegratedTestBase : IAsyncLifetime { protected TestApplication TestApp { get; } public IReadOnlyList UserInfos { get; private set; } private readonly int _userCount; public IntegratedTestBase() : this(1) { } public IntegratedTestBase(int userCount) { if (userCount < 0) throw new ArgumentOutOfRangeException(nameof(userCount), userCount, "User count can't be negative."); _userCount = userCount; TestApp = new TestApplication(); } protected virtual Task OnInitializeAsync() { return Task.CompletedTask; } protected virtual Task OnDisposeAsync() { return Task.CompletedTask; } protected virtual void OnDispose() { } public async Task InitializeAsync() { await TestApp.InitializeAsync(); using (var scope = TestApp.Host.Services.CreateScope()) { var users = new List<(string username, string password, string nickname)>() { ("admin", "adminpw", "administrator") }; for (int i = 1; i <= _userCount; i++) { users.Add(($"user{i}", $"user{i}pw", $"imuser{i}")); } var userInfoList = new List(); var userService = scope.ServiceProvider.GetRequiredService(); foreach (var user in users) { var (username, password, nickname) = user; var u = await userService.CreateUser(username, password); await userService.ModifyUser(u.Id, new ModifyUserParams() { Nickname = nickname }); } using var client = await CreateDefaultClient(); var options = new JsonSerializerOptions { PropertyNamingPolicy = JsonNamingPolicy.CamelCase }; options.Converters.Add(new JsonStringEnumConverter()); options.Converters.Add(new JsonDateTimeConverter()); foreach (var user in users) { var s = await client.GetStringAsync($"users/{user.username}"); userInfoList.Add(JsonSerializer.Deserialize(s, options)); } UserInfos = userInfoList; } await OnInitializeAsync(); } public async Task DisposeAsync() { await OnDisposeAsync(); OnDispose(); await TestApp.DisposeAsync(); } public Task CreateDefaultClient(bool setApiBase = true) { var client = TestApp.Host.GetTestServer().CreateClient(); if (setApiBase) { client.BaseAddress = new Uri(client.BaseAddress, "api/"); } return Task.FromResult(client); } public async Task CreateClientWithCredential(string username, string password, bool setApiBase = true) { var client = TestApp.Host.GetTestServer().CreateClient(); if (setApiBase) { client.BaseAddress = new Uri(client.BaseAddress, "api/"); } var response = await client.PostAsJsonAsync("token/create", new CreateTokenRequest { Username = username, Password = password }); var token = response.Should().HaveStatusCode(200) .And.HaveJsonBody().Which.Token; client.DefaultRequestHeaders.Add("Authorization", "Bearer " + token); return client; } public Task CreateClientAs(int userNumber, bool setApiBase = true) { if (userNumber < 0) return CreateDefaultClient(setApiBase); if (userNumber == 0) return CreateClientWithCredential("admin", "adminpw", setApiBase); else return CreateClientWithCredential($"user{userNumber}", $"user{userNumber}pw", setApiBase); } public Task CreateClientAsAdministrator(bool setApiBase = true) { return CreateClientAs(0, setApiBase); } public Task CreateClientAsUser(bool setApiBase = true) { return CreateClientAs(1, setApiBase); } } }