using Microsoft.AspNetCore.Mvc.Testing;
using Microsoft.Extensions.DependencyInjection;
using OI.Metrology.Shared.DataModels;
using OI.Metrology.Shared.Models.Stateless;
using Serilog;
using System.Text;

namespace OI.Metrology.Tests;

[TestClass]
public class UnitTestExportController
{

#pragma warning disable CS8618

    private static ILogger _Logger;
    private static string _ControllerName;
    private static TestContext _TestContext;
    private static WebApplicationFactory<Server.Program> _WebApplicationFactory;

#pragma warning restore

    [ClassInitialize]
    public static void ClassInitAsync(TestContext testContext)
    {
        _TestContext = testContext;
        _Logger = Log.ForContext<UnitTestExportController>();
        _WebApplicationFactory = new WebApplicationFactory<Server.Program>();
        _ControllerName = nameof(Server.ApiControllers.ExportController)[..^10];
    }

    [TestMethod]
    public void TestControllerName()
    {
        _Logger.Information("Starting Web Application");
        Assert.AreEqual(IExportController<string>.GetRouteName(), _ControllerName);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    private static HeaderCommon GetHeaderCommon() =>
        new() { PSN = "5008", Reactor = "61", RDS = "579487", ID = 1678209360 };

    private static StringContent GetStringContent() =>
        new(System.Text.Json.JsonSerializer.Serialize(GetHeaderCommon()), Encoding.UTF8, "application/json");

    [TestMethod]
    public void GetExport()
    {
        _Logger.Information("Starting Web Application");
        IServiceProvider serviceProvider = _WebApplicationFactory.Services.CreateScope().ServiceProvider;
        IExportRepository exportRepository = serviceProvider.GetRequiredService<IExportRepository>();
        string result = exportRepository.GetExport(GetHeaderCommon());
        Assert.IsNotNull(result);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task GetExportApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        string? result = await httpClient.GetStringAsync($"api/{_ControllerName}/export");
        File.WriteAllText(Path.Combine(AppContext.BaseDirectory, $"{_ControllerName}-{nameof(GetExport)}.txt"), result);
        Assert.IsNotNull(result);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task PostExportApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        HttpResponseMessage httpResponseMessage = await httpClient.PostAsync($"api/{_ControllerName}/export", GetStringContent());
        Assert.IsTrue(httpResponseMessage.StatusCode == System.Net.HttpStatusCode.OK);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public void GetHeaders()
    {
        _Logger.Information("Starting Web Application");
        IServiceProvider serviceProvider = _WebApplicationFactory.Services.CreateScope().ServiceProvider;
        IExportRepository exportRepository = serviceProvider.GetRequiredService<IExportRepository>();
        Result<HeaderCommon[]> result = exportRepository.GetHeaders(GetHeaderCommon());
        Assert.IsNotNull(result?.Results);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task GetHeadersApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        string? json = await httpClient.GetStringAsync($"api/{_ControllerName}/headers");
        File.WriteAllText(Path.Combine(AppContext.BaseDirectory, $"{_ControllerName}-{nameof(GetHeaders)}.json"), json);
        Result<HeaderCommon[]>? result = System.Text.Json.JsonSerializer.Deserialize<Result<HeaderCommon[]>>(json);
        Assert.IsNotNull(result?.Results);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task PostHeadersApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        HttpResponseMessage httpResponseMessage = await httpClient.PostAsync($"api/{_ControllerName}/headers", GetStringContent());
        Assert.IsTrue(httpResponseMessage.StatusCode == System.Net.HttpStatusCode.OK);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public void GetLogistics()
    {
        _Logger.Information("Starting Web Application");
        IServiceProvider serviceProvider = _WebApplicationFactory.Services.CreateScope().ServiceProvider;
        IExportRepository exportRepository = serviceProvider.GetRequiredService<IExportRepository>();
        Result<HeaderCommon[]> result = exportRepository.GetLogistics(GetHeaderCommon());
        Assert.IsNotNull(result?.Results);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task GetLogisticsApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        string? json = await httpClient.GetStringAsync($"api/{_ControllerName}/logistics");
        File.WriteAllText(Path.Combine(AppContext.BaseDirectory, $"{_ControllerName}-{nameof(GetLogistics)}.json"), json);
        Result<HeaderCommon[]>? result = System.Text.Json.JsonSerializer.Deserialize<Result<HeaderCommon[]>>(json);
        Assert.IsNotNull(result?.Results);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task PostLogisticsApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        HttpResponseMessage httpResponseMessage = await httpClient.PostAsync($"api/{_ControllerName}/logistics", GetStringContent());
        Assert.IsTrue(httpResponseMessage.StatusCode == System.Net.HttpStatusCode.OK);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public void GetProcessDataStandardFormat()
    {
        _Logger.Information("Starting Web Application");
        IServiceProvider serviceProvider = _WebApplicationFactory.Services.CreateScope().ServiceProvider;
        IExportRepository exportRepository = serviceProvider.GetRequiredService<IExportRepository>();
        string result = exportRepository.GetProcessDataStandardFormat(GetHeaderCommon());
        Assert.IsNotNull(result);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task GetProcessDataStandardFormatApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        string? result = await httpClient.GetStringAsync($"api/{_ControllerName}/processDataStandardFormat");
        File.WriteAllText(Path.Combine(AppContext.BaseDirectory, $"{_ControllerName}-{nameof(GetProcessDataStandardFormat)}.pdsf"), result);
        Assert.IsNotNull(result);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

    [TestMethod]
    public async Task PostProcessDataStandardFormatApi()
    {
        HttpClient httpClient = _WebApplicationFactory.CreateClient();
        _Logger.Information("Starting Web Application");
        HttpResponseMessage httpResponseMessage = await httpClient.PostAsync($"api/{_ControllerName}/processDataStandardFormat", GetStringContent());
        Assert.IsTrue(httpResponseMessage.StatusCode == System.Net.HttpStatusCode.OK);
        _Logger.Information($"{_TestContext?.TestName} completed");
    }

}