using Adaptation._Tests.Shared.PasteSpecialXml.EAF.XML.API.CellInstance;
using Adaptation._Tests.Shared.PasteSpecialXml.EAF.XML.API.ConfigurationData;
using Adaptation._Tests.Shared.PasteSpecialXml.EAF.XML.API.EquipmentDictionary;
using Adaptation._Tests.Shared.PasteSpecialXml.EAF.XML.API.EquipmentType;
using Adaptation.Eaf.Management.ConfigurationData.CellAutomation;
using Adaptation.Ifx.Eaf.Common.Configuration;
using Adaptation.Ifx.Eaf.EquipmentConnector.File.Configuration;
using Adaptation.Shared;
using Adaptation.Shared.Methods;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Text.Json;
using System.Text.Json.Serialization;
using System.Xml;
using System.Xml.Linq;
using System.Xml.Serialization;

namespace Adaptation._Tests.Shared;

public class AdaptationTesting : ISMTP
{

    protected readonly string _DummyRoot;
    protected readonly string _Environment;
    protected readonly string _HostNameAndPort;
    protected readonly bool _HasWaitForProperty;
    protected readonly TestContext _TestContext;
    protected readonly bool _SkipEquipmentDictionary;
    protected readonly string _TestContextPropertiesAsJson;
    protected readonly Dictionary<string, CellInstanceVersion> _CellInstanceVersions;
    protected readonly Dictionary<string, EquipmentTypeVersion> _EquipmentTypeVersions;
    protected readonly Dictionary<string, string> _ParameterizedModelObjectDefinitionTypes;
    protected readonly Dictionary<string, EquipmentDictionaryVersion> _EquipmentDictionaryVersions;
    protected readonly Dictionary<string, FileConnectorConfiguration> _FileConnectorConfigurations;
    protected readonly Dictionary<string, IList<ModelObjectParameterDefinition>> _ModelObjectParameters;
    protected readonly Dictionary<string, List<Tuple<string, string>>> _EquipmentDictionaryEventDescriptions;

    public string DummyRoot => _DummyRoot;
    public string Environment => _Environment;
    public TestContext TestContext => _TestContext;
    public string HostNameAndPort => _HostNameAndPort;
    public bool HasWaitForProperty => _HasWaitForProperty;
    public bool SkipEquipmentDictionary => _SkipEquipmentDictionary;
    public string TestContextPropertiesAsJson => _TestContextPropertiesAsJson;
    public Dictionary<string, CellInstanceVersion> CellInstanceVersions => _CellInstanceVersions;
    public Dictionary<string, EquipmentTypeVersion> EquipmentTypeVersions => _EquipmentTypeVersions;
    public Dictionary<string, IList<ModelObjectParameterDefinition>> ModelObjectParameters => _ModelObjectParameters;
    public Dictionary<string, EquipmentDictionaryVersion> EquipmentDictionaryVersions => _EquipmentDictionaryVersions;
    public Dictionary<string, FileConnectorConfiguration> FileConnectorConfigurations => _FileConnectorConfigurations;
    public Dictionary<string, string> ParameterizedModelObjectDefinitionTypes => _ParameterizedModelObjectDefinitionTypes;
    public Dictionary<string, List<Tuple<string, string>>> EquipmentDictionaryEventDescriptions => _EquipmentDictionaryEventDescriptions;

    void ISMTP.SendLowPriorityEmailMessage(string subject, string body) => throw new NotImplementedException();

    void ISMTP.SendHighPriorityEmailMessage(string subject, string body) => throw new NotImplementedException();

    void ISMTP.SendNormalPriorityEmailMessage(string subject, string body) => throw new NotImplementedException();

    public AdaptationTesting(string dummyRoot, TestContext testContext, bool skipEquipmentDictionary, string testContextPropertiesAsJson, bool hasWaitForProperty)
    {

        string environment = GetEnvironment(testContext);
        string hostNameAndPort = GetHostNameAndPort(environment);
        _DummyRoot = dummyRoot;
        _TestContext = testContext;
        _Environment = environment;
        _HostNameAndPort = hostNameAndPort;
        _HasWaitForProperty = hasWaitForProperty;
        _SkipEquipmentDictionary = skipEquipmentDictionary;
        _TestContextPropertiesAsJson = testContextPropertiesAsJson;
        _CellInstanceVersions = new Dictionary<string, CellInstanceVersion>();
        _EquipmentTypeVersions = new Dictionary<string, EquipmentTypeVersion>();
        _EquipmentTypeVersions = new Dictionary<string, EquipmentTypeVersion>();
        _ParameterizedModelObjectDefinitionTypes = new Dictionary<string, string>();
        _ParameterizedModelObjectDefinitionTypes = new Dictionary<string, string>();
        _EquipmentDictionaryVersions = new Dictionary<string, EquipmentDictionaryVersion>();
        _FileConnectorConfigurations = new Dictionary<string, FileConnectorConfiguration>();
        _ModelObjectParameters = new Dictionary<string, IList<ModelObjectParameterDefinition>>();
        _EquipmentDictionaryEventDescriptions = new Dictionary<string, List<Tuple<string, string>>>();
    }

    protected static string GetEnvironment(TestContext testContext)
    {
        string result = testContext.TestName.Split('_')[0];
        return result;
    }

    protected static string GetHostNameAndPort(string environment)
    {
        string result;
        result = environment switch
        {
            "LocalHost" => "localhost:9003",
            "Development" => "eaf-dev.mes.infineon.com:9003",
            "Staging" => "eaf-staging.mes.infineon.com:9003",
            "Production" => "eaf-prod.mes.infineon.com:9003",
            _ => throw new Exception(),
        };
        return result;
    }

    public static string GetTestResultsDirectory(string testContextTestResultsDirectory, bool hasWaitForProperty)
    {
        string result = string.Empty;
        string testResults = "05_TestResults";
        string checkDirectory = testContextTestResultsDirectory;
        if (hasWaitForProperty && (string.IsNullOrEmpty(checkDirectory) || !checkDirectory.Contains(testResults)))
            throw new Exception($"A:{checkDirectory}; B:{testResults};");
        else if (!hasWaitForProperty && (string.IsNullOrEmpty(checkDirectory) || !checkDirectory.Contains(testResults)))
            result = testContextTestResultsDirectory;
        else
        {
            string rootDirectory = Path.GetPathRoot(checkDirectory);
            for (int i = 0; i < int.MaxValue; i++)
            {
                checkDirectory = Path.GetDirectoryName(checkDirectory);
                if (string.IsNullOrEmpty(checkDirectory) || checkDirectory == rootDirectory)
                    break;
                if (checkDirectory.EndsWith(testResults) && Directory.Exists(checkDirectory))
                {
                    result = checkDirectory;
                    break;
                }
            }
        }
        if (string.IsNullOrEmpty(result))
            throw new Exception();
        return result;
    }

    private string GetTestResultsDirectory(bool hasWaitForProperty)
    {
        string result = GetTestResultsDirectory(_TestContext.TestResultsDirectory, hasWaitForProperty);
        return result;
    }

    protected static string GetCellInstanceConnectionName(string cellInstanceConnectionName)
    {
        string result;
        if (string.IsNullOrEmpty(cellInstanceConnectionName) || cellInstanceConnectionName[cellInstanceConnectionName.Length - 1] != '_')
            result = cellInstanceConnectionName;
        else
        {
            bool check = false;
            List<char> chars = new();
            StringBuilder stringBuilder = new();
            for (int i = cellInstanceConnectionName.Length - 1; i > -1; i--)
            {
                if (!check && cellInstanceConnectionName[i] != '_')
                    check = true;
                else if (!check && cellInstanceConnectionName[i] == '_')
                    chars.Add('-');
                if (check)
                    chars.Add(cellInstanceConnectionName[i]);
            }
            for (int i = chars.Count - 1; i > -1; i--)
                _ = stringBuilder.Append(chars[i]);
            result = stringBuilder.ToString();
        }
        return result;
    }

    private static string GetMethodBaseNameWithActualCICN(string methodBaseName, string cellInstanceName, string cellInstanceConnectionNameFromMethodBaseName, string cellInstanceConnectionName, string ticks)
    {
        string results;
        if (string.IsNullOrEmpty(cellInstanceConnectionNameFromMethodBaseName) || string.IsNullOrEmpty(cellInstanceConnectionName))
            results = methodBaseName;
        else if (cellInstanceConnectionNameFromMethodBaseName.Length != cellInstanceConnectionName.Length)
            throw new Exception();
        else
        {
            string[] segments = methodBaseName.Split(new string[] { cellInstanceName }, StringSplitOptions.None);
            if (segments.Length == 2)
                results = methodBaseName.Replace(cellInstanceConnectionNameFromMethodBaseName, cellInstanceConnectionName);
            else if (segments.Length != 3)
                throw new Exception();
            else if (string.IsNullOrEmpty(ticks))
                results = string.Concat(segments[0], cellInstanceName, segments[1], cellInstanceConnectionName);
            else if (!segments[2].Contains(ticks))
                throw new Exception();
            else
                results = string.Concat(segments[0], cellInstanceName, segments[1], cellInstanceConnectionName, ticks, segments[2].Split(new string[] { ticks }, StringSplitOptions.None)[1]);
        }
        if (methodBaseName.Length != results.Length)
            throw new Exception();
        return results;
    }

    public static MethodBaseName GetMethodBaseName(string dummyRoot, string environment, bool hasWaitForProperty, string methodBaseName, string testResultsDirectory)
    {
        MethodBaseName result;
        string comment;
        string fileFullName;
        string dummyDirectory;
        string withActualCICN;
        string separator = "__";
        string textFileDirectory;
        string connectionNameAndTicks;
        string cellInstanceConnectionName;
        string ticks = DateTime.Now.Ticks.ToString();
        string cellInstanceConnectionNameFromMethodBaseName;
        string[] segments = methodBaseName.Split(new string[] { separator }, StringSplitOptions.None);
        if (segments[0] != environment)
            throw new Exception();
        string rawVersionName = segments[1];
        string rawCellInstanceName = segments[2];
        string cellInstanceVersionName = segments[1].Replace('_', '.');
        string cellInstanceName = segments[2].Replace('_', '-').Replace("_EQPT", "-EQPT");
        string before = string.Concat(environment, separator, rawVersionName, separator, cellInstanceName, separator);
        string after = methodBaseName.Substring(before.Length);
        string versionDirectory = Path.Combine(testResultsDirectory, environment, cellInstanceName, cellInstanceVersionName);
        if (!Directory.Exists(versionDirectory))
            _ = Directory.CreateDirectory(versionDirectory);
        comment = segments[segments.Length - 1];
        if (after.Length < ticks.Length || after == comment)
        {
            ticks = string.Empty;
            cellInstanceConnectionNameFromMethodBaseName = string.Empty;
        }
        else
        {
            connectionNameAndTicks = after.Substring(0, after.Length - 2 - comment.Length);
            if (connectionNameAndTicks.Length - ticks.Length < 1)
            {
                ticks = string.Empty;
                cellInstanceConnectionNameFromMethodBaseName = string.Empty;
            }
            else
            {
                cellInstanceConnectionNameFromMethodBaseName = connectionNameAndTicks.Substring(0, connectionNameAndTicks.Length - ticks.Length);
                ticks = connectionNameAndTicks.Substring(cellInstanceConnectionNameFromMethodBaseName.Length);
            }
        }
        if (string.IsNullOrEmpty(ticks) || string.IsNullOrEmpty(cellInstanceConnectionNameFromMethodBaseName) || !long.TryParse(ticks, out _))
        {
            ticks = string.Empty;
            comment = string.Empty;
            cellInstanceConnectionNameFromMethodBaseName = after;
        }
        cellInstanceConnectionName = GetCellInstanceConnectionName(cellInstanceConnectionNameFromMethodBaseName);
        withActualCICN = GetMethodBaseNameWithActualCICN(methodBaseName, rawCellInstanceName, cellInstanceConnectionNameFromMethodBaseName, cellInstanceConnectionName, ticks);
        if (hasWaitForProperty)
            dummyDirectory = string.Empty;
        else if (string.IsNullOrEmpty(ticks))
            dummyDirectory = string.Empty;
        else
        {
            if (string.IsNullOrEmpty(dummyRoot))
                throw new Exception($"{nameof(dummyRoot)} is empty!");
            if (!withActualCICN.Contains(ticks))
                throw new Exception($"{withActualCICN} doesn't contain {ticks}!");
            segments = withActualCICN.Split(new string[] { ticks }, StringSplitOptions.None);
            dummyDirectory = Path.Combine(dummyRoot, cellInstanceName, ticks, string.Join(null, segments));
            if (!Directory.Exists(dummyDirectory))
                _ = Directory.CreateDirectory(dummyDirectory);
        }
        if (string.IsNullOrEmpty(ticks))
        {
            textFileDirectory = string.Empty;
            fileFullName = Path.Combine(versionDirectory, withActualCICN, $"{cellInstanceConnectionNameFromMethodBaseName}.json");
        }
        else
        {
            segments = withActualCICN.Split(new string[] { ticks }, StringSplitOptions.None);
            textFileDirectory = Path.Combine(versionDirectory, segments[0], string.Concat(ticks, segments[1]));
            fileFullName = Path.Combine(versionDirectory, segments[0], $"{cellInstanceConnectionNameFromMethodBaseName}.json");
        }
        result = new(after, before, cellInstanceConnectionName, cellInstanceConnectionNameFromMethodBaseName, cellInstanceName, cellInstanceVersionName, comment, dummyDirectory, fileFullName, rawCellInstanceName, rawVersionName, separator, testResultsDirectory, textFileDirectory, ticks, versionDirectory, withActualCICN);
        return result;
    }

    private MethodBaseName GetMethodBaseName(MethodBase methodBase)
    {
        MethodBaseName result;
        string testResultsDirectory = GetTestResultsDirectory(_HasWaitForProperty);
        result = GetMethodBaseName(_DummyRoot, _Environment, _HasWaitForProperty, methodBase.Name, testResultsDirectory);
        return result;
    }

    private string[] GetTextFiles(MethodBaseName mbn)
    {
        string[] results;
        if (string.IsNullOrEmpty(mbn.TextFileDirectory))
            results = Array.Empty<string>();
        else if (!Directory.Exists(mbn.TextFileDirectory))
        {
            results = Array.Empty<string>();
            if (!_HasWaitForProperty)
                _ = Directory.CreateDirectory(mbn.TextFileDirectory);
            else
            {
                string renameDirectory = Path.Combine(Path.GetDirectoryName(mbn.TextFileDirectory), $"_Rename - {Path.GetFileName(mbn.TextFileDirectory)}");
                _ = Directory.CreateDirectory(renameDirectory);
                _ = Process.Start("explorer.exe", renameDirectory);
                File.WriteAllText(Path.Combine(renameDirectory, $"{nameof(FileConnectorConfiguration.SourceFileFilter)}.txt"), string.Empty);
                File.WriteAllText(Path.Combine(renameDirectory, $"{nameof(FileConnectorConfiguration.SourceFileLocation)}.txt"), string.Empty);
            }
        }
        else
        {
            results = Directory.GetFiles(mbn.TextFileDirectory, "*.txt", SearchOption.TopDirectoryOnly);
            if (!string.IsNullOrEmpty(mbn.Ticks) && _HasWaitForProperty && results.Length == 0)
            {
                _ = Process.Start("explorer.exe", mbn.TextFileDirectory);
                File.WriteAllText(Path.Combine(mbn.TextFileDirectory, "_ Why.why"), string.Empty);
            }
        }
        return results;
    }

    protected static Stream ToStream(string @this)
    {
        MemoryStream memoryStream = new();
        StreamWriter streamWriter = new(memoryStream);
        streamWriter.Write(@this);
        streamWriter.Flush();
        memoryStream.Position = 0;
        return memoryStream;
    }

    internal static T ParseXML<T>(string @this, bool throwExceptions) where T : class
    {
        object result = null;
        try
        {
            Stream stream = ToStream(@this.Trim());
            XmlReader xmlReader = XmlReader.Create(stream, new XmlReaderSettings() { ConformanceLevel = ConformanceLevel.Document });
            XmlSerializer xmlSerializer = new(typeof(T), typeof(T).GetNestedTypes());
            result = xmlSerializer.Deserialize(xmlReader);
            stream.Dispose();
        }
        catch (Exception)
        {
            if (throwExceptions)
                throw;
        }
        return result as T;
    }

    public static CellInstanceVersion GetCellInstanceVersion(string url)
    {
        CellInstanceVersion result;
        byte[] byteArray;
        ConfigurationData configurationData;
        string decodedCellInstanceConfigurationData;
        string xml;
        try
        {
            xml = XDocument.Load(url).ToString();
        }
        catch (Exception exception)
        {
            throw new Exception(string.Concat(url, System.Environment.NewLine, exception.Message));
        }
        configurationData = ParseXML<ConfigurationData>(xml, throwExceptions: true);
        byteArray = Convert.FromBase64String(configurationData.Data);
        decodedCellInstanceConfigurationData = Encoding.Unicode.GetString(byteArray);
        if (xml.Length <= 41)
            throw new Exception(string.Concat("xml isn't valid {", xml, "}"));
        decodedCellInstanceConfigurationData = decodedCellInstanceConfigurationData.Substring(41).Replace("i:type", "i___type");
        result = ParseXML<CellInstanceVersion>(decodedCellInstanceConfigurationData, throwExceptions: true);
        return result;
    }

    protected Tuple<string, CellInstanceVersion> GetCellInstanceVersionTuple(string cellInstanceName, string cellInstanceVersionName)
    {
        Tuple<string, CellInstanceVersion> result;
        CellInstanceVersion cellInstanceVersion;
        string cellInstanceServiceV2 = string.Concat("http://", _HostNameAndPort, "/CellInstanceServiceV2/", cellInstanceName, "/", cellInstanceVersionName, "/configuration");
        if (!_CellInstanceVersions.TryGetValue(cellInstanceServiceV2, out cellInstanceVersion))
        {
            cellInstanceVersion = GetCellInstanceVersion(cellInstanceServiceV2);
            _CellInstanceVersions.Add(cellInstanceServiceV2, cellInstanceVersion);
        }
        result = new Tuple<string, CellInstanceVersion>(cellInstanceServiceV2, cellInstanceVersion);
        return result;
    }

    protected static Dictionary<string, int[]> GetComponentModelComponentsIndexes(CellInstanceVersion cellInstanceVersion, string cellInstanceConnectionName)
    {
        Dictionary<string, int[]> results = new();
        ComponentsCellComponent componentsCellComponent;
        if (cellInstanceVersion.ComponentModel.Components is not null)
        {
            for (int i = 0; i < cellInstanceVersion.ComponentModel.Components.Length; i++)
            {
                componentsCellComponent = cellInstanceVersion.ComponentModel.Components[i];
                for (int j = 0; j < componentsCellComponent.Children.Length; j++)
                {
                    if (string.IsNullOrEmpty(componentsCellComponent.Children[j].Equipment.Name))
                        continue;
                    results.Add(componentsCellComponent.Children[j].Name, new int[] { i, j });
                }
            }
        }
        if (results.Count == 0 || (!string.IsNullOrEmpty(cellInstanceConnectionName) && !results.ContainsKey(cellInstanceConnectionName)))
            throw new Exception("Match not found (check test method name matches Mango)!");
        return results;
    }

    protected static int[] GetCellInstanceConnectionNameIndexes(string cellInstanceConnectionName, Dictionary<string, int[]> componentModelComponentsIndexes)
    {
        int[] result;
        if (string.IsNullOrEmpty(cellInstanceConnectionName))
            result = componentModelComponentsIndexes.ElementAt(0).Value;
        else
        {
            if (componentModelComponentsIndexes is null || !componentModelComponentsIndexes.TryGetValue(cellInstanceConnectionName, out result))
                throw new Exception();
        }
        return result;
    }

    protected string[] GetCSharpTextB(FileInfo fileInfo, string cellInstanceName, string cellInstanceVersionName, CellInstanceVersion cellInstanceVersion)
    {
        List<string> results = new();
        string check;
        string loopName;
        string equipmentTypeName;
        string methodName = string.Empty;
        string extractText = string.Empty;
        StringBuilder stringBuilder = new();
        string createSelfDescriptionText = string.Empty;
        List<string> componentsCellComponentCellComponentEquipmentNames = new();
        string cellInstanceNameWithoutHyphen = cellInstanceName.Replace('-', '_');
        ComponentsCellComponentCellComponent componentsCellComponentCellComponent;
        List<string> componentsCellComponentCellComponentEquipmentTypeNames = new();
        string cellInstanceVersionNameAsCode = cellInstanceVersionName.Replace('.', '_');
        List<string> componentsCellComponentCellComponentEquipmentDictionaryNames = new();
        const string sourceDirectoryCloaking = nameof(FileConnectorConfiguration.SourceDirectoryCloaking);
        for (int i = 1; i < 3; i++)
        {
            if (i == 2)
                loopName = "Extract";
            else if (i == 1)
                loopName = "CreateSelfDescription";
            else
                throw new Exception();
            _ = stringBuilder.
                AppendLine("using Adaptation.Shared.Methods;").
                AppendLine("using Microsoft.Extensions.Logging;").
                AppendLine("using Microsoft.VisualStudio.TestTools.UnitTesting;").
                AppendLine("using Adaptation._Tests.Shared;").
                AppendLine("using System;").
                AppendLine("using System.Collections.Generic;").
                AppendLine("using System.Diagnostics;").
                AppendLine("using System.IO;").
                AppendLine("using System.Reflection;").
                AppendLine("using System.Text.Json;").
                AppendLine("using System.Threading;");
            _ = stringBuilder.AppendLine().
                Append("namespace Adaptation._Tests.").Append(loopName).Append('.').Append(_Environment).Append('.').Append(cellInstanceVersionNameAsCode).AppendLine(";").
                AppendLine().
                AppendLine("[TestClass]");
            if (i == 2)
                _ = stringBuilder.
                    Append("public class ").AppendLine(cellInstanceNameWithoutHyphen).
                    AppendLine("{").
                    AppendLine().
                    AppendLine("#pragma warning disable CA2254").
                    AppendLine("#pragma warning disable IDE0060").
                    AppendLine().
                    Append("private static CreateSelfDescription.").Append(_Environment).Append('.').Append(cellInstanceVersionNameAsCode).Append('.').Append(cellInstanceNameWithoutHyphen).Append(" _").Append(cellInstanceNameWithoutHyphen).AppendLine(";");
            else if (i == 1)
                _ = stringBuilder.
                    Append("public class ").Append(cellInstanceNameWithoutHyphen).AppendLine(" : EAFLoggingUnitTesting").
                    AppendLine("{").
                    AppendLine().
                    AppendLine("#pragma warning disable CA2254").
                    AppendLine("#pragma warning disable IDE0060").
                    AppendLine().
                    AppendLine("internal static string DummyRoot { get; private set; }").
                    Append("internal static ").Append(cellInstanceNameWithoutHyphen).AppendLine(" EAFLoggingUnitTesting { get; private set; }");
            else
                throw new Exception();
            if (i == 2)
                _ = stringBuilder.AppendLine();
            else if (i == 1)
                _ = stringBuilder.
                    AppendLine().
                    Append("static ").Append(cellInstanceNameWithoutHyphen).AppendLine("() => DummyRoot = @\"\\\\mesfs.infineon.com\\EC_Characterization_Si\\Dummy\";").
                    AppendLine().
                    Append("public ").Append(cellInstanceNameWithoutHyphen).AppendLine("() : base(DummyRoot, testContext: null, declaringType: null, skipEquipmentDictionary: false)").
                    AppendLine("{").
                    AppendLine("if (EAFLoggingUnitTesting is null)").
                    AppendLine("throw new Exception();").
                    AppendLine("}").
                    AppendLine().
                    Append("public ").Append(cellInstanceNameWithoutHyphen).AppendLine("(TestContext testContext) : base(DummyRoot, testContext, new StackFrame().GetMethod().DeclaringType, skipEquipmentDictionary: false)").
                    AppendLine("{").
                    AppendLine("}").
                    AppendLine();
            else
                throw new Exception();
            _ = stringBuilder.
                AppendLine("[ClassInitialize]").
                AppendLine("public static void ClassInitialize(TestContext testContext)").
                AppendLine("{");
            if (i == 2)
                _ = stringBuilder.
                    Append("CreateSelfDescription.").Append(_Environment).Append('.').Append(cellInstanceVersionNameAsCode).Append('.').Append(cellInstanceNameWithoutHyphen).AppendLine(".ClassInitialize(testContext);").
                    Append('_').Append(cellInstanceNameWithoutHyphen).Append(" = CreateSelfDescription.").Append(_Environment).Append('.').Append(cellInstanceVersionNameAsCode).Append('.').Append(cellInstanceNameWithoutHyphen).AppendLine(".EAFLoggingUnitTesting;").
                    AppendLine("}");
            else if (i == 1)
                _ = stringBuilder.
                    AppendLine("if (EAFLoggingUnitTesting is null)").
                    Append("EAFLoggingUnitTesting = new ").Append(cellInstanceNameWithoutHyphen).AppendLine("(testContext);").
                    AppendLine("EAFLoggingUnitTesting.Logger.LogInformation(string.Concat(testContext.TestName, \" - ClassInitialize\"));").
                    AppendLine("string[] fileNameAndText = EAFLoggingUnitTesting.AdaptationTesting.GetCSharpText(testContext.TestName);").
                    AppendLine("File.WriteAllText(fileNameAndText[0], fileNameAndText[1]);").
                    AppendLine("File.WriteAllText(fileNameAndText[2], fileNameAndText[3]);").
                    AppendLine("}");
            else
                throw new Exception();
            if (i == 2)
                _ = stringBuilder.AppendLine();
            else if (i == 1)
                _ = stringBuilder.
                    AppendLine().
                    AppendLine("[ClassCleanup()]").
                    AppendLine("public static void ClassCleanup()").
                    AppendLine("{").
                    AppendLine("if (EAFLoggingUnitTesting.Logger is not null)").
                    AppendLine("EAFLoggingUnitTesting.Logger.LogInformation(\"Cleanup\");").
                    AppendLine("if (EAFLoggingUnitTesting is not null)").
                    AppendLine("EAFLoggingUnitTesting.Dispose();").
                    AppendLine("}").
                    AppendLine();
            else
                throw new Exception();
            foreach (ComponentsCellComponent componentsCellComponent in cellInstanceVersion.ComponentModel.Components)
            {
                if (componentsCellComponent.Children.Length != 1)
                    continue;
                componentsCellComponentCellComponent = componentsCellComponent.Children[0];
                if (componentsCellComponentCellComponent.Equipment.EquipmentDictionaries?.CellEquipmentDictionaryReference?.DictionaryName is not null)
                    componentsCellComponentCellComponentEquipmentDictionaryNames.Add(componentsCellComponentCellComponent.Equipment.EquipmentDictionaries?.CellEquipmentDictionaryReference?.DictionaryName);
                componentsCellComponentCellComponentEquipmentNames.Add(componentsCellComponentCellComponent.Equipment.Name);
                componentsCellComponentCellComponentEquipmentTypeNames.Add(componentsCellComponentCellComponent.Equipment.EquipmentType.Name);
                methodName = $"{_Environment}__{cellInstanceVersionNameAsCode}__{cellInstanceNameWithoutHyphen}__{componentsCellComponentCellComponent.Equipment.Name.Replace('-', '_')}";
                if (componentsCellComponentCellComponent?.Equipment?.ConnectionSettings?.Setting is null)
                    check = string.Empty;
                else
                    check = (from l in componentsCellComponentCellComponent.Equipment.ConnectionSettings.Setting where l.Name == sourceDirectoryCloaking select l.Value).FirstOrDefault();
                if (string.IsNullOrEmpty(check))
                    check = componentsCellComponentCellComponent.Equipment.SourceFileFilter;
                if (i == 2)
                {
                    _ = stringBuilder.
                        AppendLine("#if true").
                        AppendLine("[Ignore]").
                        AppendLine("#endif").
                        AppendLine("[TestMethod]").
                        Append("public void ").Append(methodName).Append("() => ").Append('_').Append(cellInstanceNameWithoutHyphen).Append('.').Append(methodName).AppendLine("();").AppendLine();
                }
                else if (i == 1)
                {
                    if (componentsCellComponentCellComponent.Equipment.EquipmentType.Version != cellInstanceVersionName)
                        throw new Exception($"Versions should match! {componentsCellComponentCellComponent.Equipment.EquipmentType.Version} != {cellInstanceVersionName}");
                    equipmentTypeName = componentsCellComponentCellComponent.Equipment.EquipmentType.Name;
                    _ = stringBuilder.
                        AppendLine("#if true").
                        AppendLine("[Ignore]").
                        AppendLine("#endif").
                        AppendLine("[TestMethod]").
                        Append("public void ").Append(methodName).AppendLine("()").
                        AppendLine("{").
                        Append("string check = \"").Append(check.Split('\\').Last()).AppendLine("\";").
                        AppendLine("MethodBase methodBase = new StackFrame().GetMethod();").
                        AppendLine("EAFLoggingUnitTesting.Logger.LogInformation(string.Concat(methodBase.Name, \" - Getting configuration\"));").
                        AppendLine("_ = AdaptationTesting.GetWriteConfigurationGetFileRead(methodBase, check, EAFLoggingUnitTesting.AdaptationTesting);").
                        AppendLine("EAFLoggingUnitTesting.Logger.LogInformation(string.Concat(methodBase.Name, \" - Exit\"));").
                        AppendLine("}").
                        AppendLine();
                }
                else
                    throw new Exception();
            }
            _ = stringBuilder.
                AppendLine("}").
                AppendLine();
            if (i == 2)
                extractText = stringBuilder.ToString().Trim();
            else if (i == 1)
                createSelfDescriptionText = stringBuilder.ToString().Trim();
            else
                throw new Exception();
            _ = stringBuilder.Clear();
        }
        if (componentsCellComponentCellComponentEquipmentDictionaryNames.Count != 0 && string.IsNullOrEmpty(cellInstanceVersion.FrozenBy))
        {
            if (!cellInstanceVersion.CellCommunicatingRule.EndsWith(".Communicating") || !(from l in componentsCellComponentCellComponentEquipmentNames where l == cellInstanceVersion.CellCommunicatingRule.Split('.')[0] select true).Any())
                throw new Exception($"{methodName} - CellCommunicatingRule not correct in Mango!");
            if (!cellInstanceVersion.CellNotCommunicatingRule.EndsWith(".NotCommunicating") || !(from l in componentsCellComponentCellComponentEquipmentNames where l == cellInstanceVersion.CellNotCommunicatingRule.Split('.')[0] select true).Any())
                throw new Exception($"{methodName} - CellNotCommunicatingRule not correct in Mango!");
        }
        string versionLevelDirectory = Path.GetDirectoryName(fileInfo.DirectoryName);
        results.Add(Path.Combine(versionLevelDirectory, $"{cellInstanceName}-0-CreateSelfDescription.txt"));
        results.Add(createSelfDescriptionText);
        results.Add(Path.Combine(versionLevelDirectory, $"{cellInstanceName}-1-Extract.txt"));
        results.Add(extractText);
        return results.ToArray();
    }

    protected static FileConnectorConfiguration GetFileConnectorConfiguration(string json, ComponentsCellComponentCellComponent componentsCellComponentCellComponent)
    {
        FileConnectorConfiguration result;
        const string sourceDirectoryCloaking = nameof(FileConnectorConfiguration.SourceDirectoryCloaking);
        JsonSerializerOptions jsonSerializerOptions = new()
        { Converters = { new JsonStringEnumConverter(JsonNamingPolicy.CamelCase) } };
        json = json.Replace(string.Concat("\"", nameof(FileConnectorConfiguration.ConnectionSettings), "\":"), string.Concat("\"Ignore", nameof(FileConnectorConfiguration.ConnectionSettings), "\":"));
        result = JsonSerializer.Deserialize<FileConnectorConfiguration>(json, jsonSerializerOptions);
        if (string.IsNullOrEmpty(result.SourceFileFilter))
            result.SourceFileFilter = string.Empty;
        result.ErrorTargetFileLocation ??= string.Empty;
        result.SourceFileLocation ??= string.Empty;
        result.TargetFileLocation ??= string.Empty;
        result.FolderAgeCheckIndividualSubFolders ??= false;
        result.ConnectionSettings = new List<ConnectionSetting>();
        result.SourceFileFilters = result.SourceFileFilter.Split('|').ToList();
        if (componentsCellComponentCellComponent.Equipment?.ConnectionSettings is not null)
        {
            foreach (Setting setting in componentsCellComponentCellComponent.Equipment.ConnectionSettings.Setting)
                result.ConnectionSettings.Add(new ConnectionSetting(null, null) { Name = setting.Name, Value = setting.Value });
        }
        IEnumerable<ConnectionSetting> sourceDirectoryCloakingCollection = from l in result.ConnectionSettings where l.Name == sourceDirectoryCloaking select l;
        if (sourceDirectoryCloakingCollection.Any())
            result.SourceDirectoryCloaking = sourceDirectoryCloakingCollection.First().Value;
        else
        {
            result.SourceDirectoryCloaking = string.Empty;
            result.ConnectionSettings.Add(new ConnectionSetting(null, null) { Name = sourceDirectoryCloaking, Value = string.Empty });
        }
        return result;
    }

    protected Tuple<string, FileConnectorConfiguration> GetFileConnectorConfigurationTuple(Tuple<string, CellInstanceVersion> cellInstanceVersionTuple, string cellInstanceConnectionName)
    {
        Tuple<string, FileConnectorConfiguration> result;
        FileConnectorConfiguration fileConnectorConfiguration;
        string cellInstanceServiceV2With = string.Concat(cellInstanceVersionTuple.Item1, '/', cellInstanceConnectionName);
        if (!_FileConnectorConfigurations.TryGetValue(cellInstanceServiceV2With, out fileConnectorConfiguration))
        {
            Dictionary<string, int[]> componentModelComponentsIndexes = GetComponentModelComponentsIndexes(cellInstanceVersionTuple.Item2, cellInstanceConnectionName);
            int[] cellInstanceConnectionNameIndexes = GetCellInstanceConnectionNameIndexes(cellInstanceConnectionName, componentModelComponentsIndexes);
            ComponentsCellComponentCellComponent componentsCellComponentCellComponent = cellInstanceVersionTuple.Item2.ComponentModel.Components[cellInstanceConnectionNameIndexes[0]].Children[cellInstanceConnectionNameIndexes[1]];
            string json = JsonSerializer.Serialize(componentsCellComponentCellComponent.Equipment, new JsonSerializerOptions { WriteIndented = true });
            fileConnectorConfiguration = GetFileConnectorConfiguration(json, componentsCellComponentCellComponent);
            _FileConnectorConfigurations.Add(cellInstanceServiceV2With, fileConnectorConfiguration);
        }
        result = new Tuple<string, FileConnectorConfiguration>(cellInstanceServiceV2With, fileConnectorConfiguration);
        return result;
    }

    protected static EquipmentTypeVersion GetEquipmentTypeVersion(string url)
    {
        EquipmentTypeVersion result;
        byte[] byteArray;
        ConfigurationData configurationData;
        string decodedCellInstanceConfigurationData;
        string xml;
        try
        {
            xml = XDocument.Load(url).ToString();
        }
        catch (Exception exception)
        {
            throw new Exception(string.Concat(url, System.Environment.NewLine, exception.Message));
        }
        configurationData = ParseXML<ConfigurationData>(xml, throwExceptions: true);
        byteArray = Convert.FromBase64String(configurationData.Data);
        decodedCellInstanceConfigurationData = Encoding.Unicode.GetString(byteArray);
        if (xml.Length <= 41)
            throw new Exception(string.Concat("xml isn't valid {", xml, "}"));
        decodedCellInstanceConfigurationData = decodedCellInstanceConfigurationData.Substring(41).Replace("i:type", "i___type");
        result = ParseXML<EquipmentTypeVersion>(decodedCellInstanceConfigurationData, throwExceptions: true);
        return result;
    }

    protected Tuple<string, string, string, EquipmentTypeVersion> GetEquipmentTypeVersionTuple(CellInstanceVersion cellInstanceVersion, string cellInstanceConnectionName)
    {
        Tuple<string, string, string, EquipmentTypeVersion> result;
        EquipmentTypeVersion equipmentTypeVersion;
        Dictionary<string, int[]> componentModelComponentsIndexes = GetComponentModelComponentsIndexes(cellInstanceVersion, cellInstanceConnectionName);
        int[] cellInstanceConnectionNameIndexes = GetCellInstanceConnectionNameIndexes(cellInstanceConnectionName, componentModelComponentsIndexes);
        ComponentsCellComponentCellComponent componentsCellComponentCellComponent = cellInstanceVersion.ComponentModel.Components[cellInstanceConnectionNameIndexes[0]].Children[cellInstanceConnectionNameIndexes[1]];
        string equipmentTypeServiceV2 = string.Concat("http://", _HostNameAndPort, "/EquipmentTypeServiceV2/", componentsCellComponentCellComponent.Equipment.EquipmentType.Name, "/", componentsCellComponentCellComponent.Equipment.EquipmentType.Version, "/configuration");
        if (!_EquipmentTypeVersions.TryGetValue(equipmentTypeServiceV2, out equipmentTypeVersion))
        {
            equipmentTypeVersion = GetEquipmentTypeVersion(equipmentTypeServiceV2);
            _EquipmentTypeVersions.Add(equipmentTypeServiceV2, equipmentTypeVersion);
        }
        result = new Tuple<string, string, string, EquipmentTypeVersion>(equipmentTypeServiceV2, componentsCellComponentCellComponent.Equipment.EquipmentType.Name, componentsCellComponentCellComponent.Equipment.EquipmentType.Version, equipmentTypeVersion);
        return result;
    }

    protected Tuple<string, string> GetParameterizedModelObjectDefinitionTypeTuple(Tuple<string, string, string, EquipmentTypeVersion> equipmentTypeVersionTuple)
    {
        Tuple<string, string> result;
        string parameterizedModelObjectDefinitionType;
        if (_FileConnectorConfigurations.ContainsKey(equipmentTypeVersionTuple.Item1))
            parameterizedModelObjectDefinitionType = _ParameterizedModelObjectDefinitionTypes[equipmentTypeVersionTuple.Item1];
        else
            parameterizedModelObjectDefinitionType = equipmentTypeVersionTuple.Item4.FileHandlerObjectTypes.ParameterizedModelObjectDefinition.Type;
        result = new Tuple<string, string>(equipmentTypeVersionTuple.Item1, parameterizedModelObjectDefinitionType);
        return result;
    }

    protected IList<ModelObjectParameterDefinition> GetModelObjectParameters(string json)
    {
        IList<ModelObjectParameterDefinition> results;
        JsonSerializerOptions jsonSerializerOptions = new()
        { Converters = { new JsonStringEnumConverter(JsonNamingPolicy.CamelCase) } };
        JsonElement jsonElement = JsonSerializer.Deserialize<JsonElement>(json);
        string parameters = "Parameters";
        string fileHandlerObjectTypes = nameof(FileHandlerObjectTypes);
        string parameterizedModelObjectDefinition = nameof(ParameterizedModelObjectDefinition);
        if (!json.Contains(fileHandlerObjectTypes) || !json.Contains(parameterizedModelObjectDefinition))
            throw new Exception();
        jsonElement = jsonElement.GetProperty(fileHandlerObjectTypes);
        if (jsonElement.ValueKind != JsonValueKind.Object)
            throw new Exception();
        jsonElement = jsonElement.GetProperty(parameterizedModelObjectDefinition);
        if (jsonElement.ValueKind != JsonValueKind.Object)
            throw new Exception();
        jsonElement = jsonElement.GetProperty(parameters);
        if (jsonElement.ValueKind != JsonValueKind.Array)
            throw new Exception();
        results = JsonSerializer.Deserialize<IList<ModelObjectParameterDefinition>>(jsonElement.ToString(), jsonSerializerOptions);
        return results;
    }

    protected Tuple<string, IList<ModelObjectParameterDefinition>> GetModelObjectParameters(Tuple<string, string, string, EquipmentTypeVersion> equipmentTypeVersionTuple)
    {
        Tuple<string, IList<ModelObjectParameterDefinition>> result;
        IList<ModelObjectParameterDefinition> modelObjectParameters;
        if (_FileConnectorConfigurations.ContainsKey(equipmentTypeVersionTuple.Item1))
            modelObjectParameters = _ModelObjectParameters[equipmentTypeVersionTuple.Item1];
        else
        {
            string json = JsonSerializer.Serialize(equipmentTypeVersionTuple.Item4, new JsonSerializerOptions { WriteIndented = true });
            modelObjectParameters = GetModelObjectParameters(json);
        }
        result = new Tuple<string, IList<ModelObjectParameterDefinition>>(equipmentTypeVersionTuple.Item1, modelObjectParameters);
        return result;
    }

    protected string[] GetEquipmentDictionaryStrings(ComponentsCellComponentCellComponentEquipment componentsCellComponentCellComponentEquipment, EquipmentTypeVersion equipmentTypeVersion)
    {
        string[] results;
        string equipmentDictionaryName;
        string equipmentDictionaryVersionName;
        if (_SkipEquipmentDictionary || equipmentTypeVersion?.EventActionSequences is null || equipmentTypeVersion.EventActionSequences.Length == 0 || !(from l in equipmentTypeVersion.EventActionSequences where l.HandledEvent.StartsWith("Equipment.FileRead") select 1).Any())
        {
            equipmentDictionaryName = string.Empty;
            equipmentDictionaryVersionName = string.Empty;
        }
        else
        {
            if (componentsCellComponentCellComponentEquipment?.EquipmentDictionaries.CellEquipmentDictionaryReference is null)
            {
                // equipmentDictionaryName = equipmentTypeVersion.Dictionaries.EquipmentTypeDictionaryReference.DictionaryName;
                // equipmentDictionaryVersionName = equipmentTypeVersion.Dictionaries.EquipmentTypeDictionaryReference.DictionaryVersion;
                equipmentDictionaryName = string.Empty;
                equipmentDictionaryVersionName = string.Empty;
            }
            else
            {
                equipmentDictionaryName = componentsCellComponentCellComponentEquipment.EquipmentDictionaries.CellEquipmentDictionaryReference.DictionaryName;
                equipmentDictionaryVersionName = componentsCellComponentCellComponentEquipment.EquipmentDictionaries.CellEquipmentDictionaryReference.DictionaryVersion;
            }
        }
        results = new string[] { equipmentDictionaryName, equipmentDictionaryVersionName };
        return results;
    }

    protected static EquipmentDictionaryVersion GetEquipmentDictionaryVersion(string url)
    {
        EquipmentDictionaryVersion result;
        string xml;
        try
        {
            xml = XDocument.Load(url).ToString();
        }
        catch (Exception exception)
        {
            throw new Exception(string.Concat(url, System.Environment.NewLine, exception.Message));
        }
        ConfigurationData configurationData = ParseXML<ConfigurationData>(xml, throwExceptions: true);
        byte[] byteArray = Convert.FromBase64String(configurationData.Data);
        string decodedCellInstanceConfigurationData = Encoding.Unicode.GetString(byteArray);
        if (xml.Length <= 41)
            throw new Exception(string.Concat("xml isn't valid {", xml, "}"));
        decodedCellInstanceConfigurationData = decodedCellInstanceConfigurationData.Substring(41).Replace("i:type", "i___type");
        result = ParseXML<EquipmentDictionaryVersion>(decodedCellInstanceConfigurationData, throwExceptions: true);
        return result;
    }

    protected Tuple<string, string, string, EquipmentDictionaryVersion> GetEquipmentDictionaryVersionTuple(CellInstanceVersion cellInstanceVersion, string cellInstanceConnectionName, EquipmentTypeVersion equipmentTypeVersion)
    {
        Tuple<string, string, string, EquipmentDictionaryVersion> result;
        string equipmentDictionaryName;
        string equipmentDictionaryVersionName;
        EquipmentDictionaryVersion equipmentDictionaryVersion;
        Dictionary<string, int[]> componentModelComponentsIndexes = GetComponentModelComponentsIndexes(cellInstanceVersion, cellInstanceConnectionName);
        int[] cellInstanceConnectionNameIndexes = GetCellInstanceConnectionNameIndexes(cellInstanceConnectionName, componentModelComponentsIndexes);
        ComponentsCellComponentCellComponent componentsCellComponentCellComponent = cellInstanceVersion.ComponentModel.Components[cellInstanceConnectionNameIndexes[0]].Children[cellInstanceConnectionNameIndexes[1]];
        string[] segments = GetEquipmentDictionaryStrings(componentsCellComponentCellComponent.Equipment, equipmentTypeVersion);
        if (_SkipEquipmentDictionary || segments is null || segments.Length != 2 || string.IsNullOrEmpty(segments[0]) || string.IsNullOrEmpty(segments[1]))
        {
            equipmentDictionaryName = string.Empty;
            equipmentDictionaryVersionName = string.Empty;
        }
        else
        {
            equipmentDictionaryName = segments[0];
            equipmentDictionaryVersionName = segments[1];
        }
        string equipmentDictionaryServiceV2 = string.Concat("http://", _HostNameAndPort, "/EquipmentDictionaryServiceV2/", equipmentDictionaryName, "/", equipmentDictionaryVersionName, "/configuration");
        if (string.IsNullOrEmpty(equipmentDictionaryName) || string.IsNullOrEmpty(equipmentDictionaryVersionName))
            equipmentDictionaryVersion = null;
        else
        {
            if (!_EquipmentDictionaryVersions.TryGetValue(equipmentDictionaryServiceV2, out equipmentDictionaryVersion))
            {
                equipmentDictionaryVersion = GetEquipmentDictionaryVersion(equipmentDictionaryServiceV2);
                _EquipmentDictionaryVersions.Add(equipmentDictionaryServiceV2, equipmentDictionaryVersion);
            }
        }
        result = new Tuple<string, string, string, EquipmentDictionaryVersion>(equipmentDictionaryServiceV2, equipmentDictionaryName, equipmentDictionaryVersionName, equipmentDictionaryVersion);
        return result;
    }

    protected Tuple<string, List<Tuple<string, string>>> GetEquipmentDictionaryIsAlwaysEnabledEventsTuple(Tuple<string, string, string, EquipmentDictionaryVersion> equipmentDictionaryVersionTuple)
    {
        Tuple<string, List<Tuple<string, string>>> result;
        List<Tuple<string, string>> results;
        List<Tuple<string, string>> collection;
        if (_SkipEquipmentDictionary)
            results = new List<Tuple<string, string>>();
        else if (string.IsNullOrEmpty(equipmentDictionaryVersionTuple.Item1))
            throw new Exception();
        else if (equipmentDictionaryVersionTuple?.Item4?.Events?.Event is null)
            results = new List<Tuple<string, string>>();
        else if (_EquipmentDictionaryEventDescriptions.TryGetValue(equipmentDictionaryVersionTuple.Item1, out collection))
            results = collection;
        else
        {
            results = new List<Tuple<string, string>>();
            foreach (EquipmentDictionaryVersionEventsEvent equipmentDictionaryVersionEventsEvent in equipmentDictionaryVersionTuple.Item4.Events.Event)
            {
                if (string.IsNullOrEmpty(equipmentDictionaryVersionEventsEvent.Description))
                    continue;
                if (!equipmentDictionaryVersionEventsEvent.IsAlwaysEnabled)
                    continue;
                results.Add(new Tuple<string, string>(equipmentDictionaryVersionEventsEvent.Name, equipmentDictionaryVersionEventsEvent.Description));
            }
        }
        result = new Tuple<string, List<Tuple<string, string>>>(equipmentDictionaryVersionTuple.Item1, results);
        return result;
    }

    protected Dictionary<string, object> GetKeyValuePairs(string cellInstanceName, string cellInstanceVersionName, string cellInstanceConnectionName, FileConnectorConfiguration fileConnectorConfiguration, string equipmentTypeName, string parameterizedModelObjectDefinitionType, IList<ModelObjectParameterDefinition> modelObjectParameters, string equipmentDictionaryName, List<Tuple<string, string>> equipmentDictionaryIsAlwaysEnabledEvents, int edaConnectionPortNumber)
    {
        Dictionary<string, object> results = new()
        {
            { nameof(System.Environment), _Environment },
            { nameof(HostNameAndPort), _HostNameAndPort },
            { nameof(cellInstanceName), cellInstanceName },
            { nameof(equipmentTypeName), equipmentTypeName },
            { nameof(cellInstanceVersionName), cellInstanceVersionName },
            { nameof(edaConnectionPortNumber), edaConnectionPortNumber },
            { nameof(equipmentDictionaryName), equipmentDictionaryName },
            { nameof(cellInstanceConnectionName), cellInstanceConnectionName },
            { nameof(FileConnectorConfiguration), fileConnectorConfiguration },
            { nameof(IList<ModelObjectParameterDefinition>), modelObjectParameters },
            { nameof(parameterizedModelObjectDefinitionType), parameterizedModelObjectDefinitionType },
            { nameof(equipmentDictionaryIsAlwaysEnabledEvents), equipmentDictionaryIsAlwaysEnabledEvents }
        };
        return results;
    }

    public string[] GetCSharpText(string testName)
    {
        string[] results;
        string testResultsDirectory = GetTestResultsDirectory(_HasWaitForProperty);
        MethodBaseName mbn = GetMethodBaseName(_DummyRoot, _Environment, _HasWaitForProperty, testName, testResultsDirectory);
        FileInfo fileInfo = new(mbn.FileFullName);
        if (!string.IsNullOrEmpty(mbn.CellInstanceConnectionName) && !Directory.Exists(fileInfo.DirectoryName))
            _ = Directory.CreateDirectory(fileInfo.Directory.FullName);
        Tuple<string, CellInstanceVersion> cellInstanceVersionTuple = GetCellInstanceVersionTuple(mbn.CellInstanceName, mbn.CellInstanceVersionName);
        results = GetCSharpTextB(fileInfo, mbn.CellInstanceName, mbn.CellInstanceVersionName, cellInstanceVersionTuple.Item2);
        return results;
    }

    public string[] GetConfiguration(MethodBase methodBase)
    {
        string[] results;
        MethodBaseName mbn = GetMethodBaseName(methodBase);
        FileInfo fileInfo = new(mbn.FileFullName);
        if (!string.IsNullOrEmpty(mbn.CellInstanceConnectionName) && !Directory.Exists(fileInfo.DirectoryName))
            _ = Directory.CreateDirectory(fileInfo.Directory.FullName);
        Tuple<string, CellInstanceVersion> cellInstanceVersionTuple = GetCellInstanceVersionTuple(mbn.CellInstanceName, mbn.CellInstanceVersionName);
        Tuple<string, FileConnectorConfiguration> fileConnectorConfigurationTuple = GetFileConnectorConfigurationTuple(cellInstanceVersionTuple, mbn.CellInstanceConnectionName);
        if (string.IsNullOrEmpty(mbn.Ticks) && fileConnectorConfigurationTuple.Item2?.FileScanningIntervalInSeconds is not null)
        {
            string fileScanningIntervalInSecondsLine;
            string versionDirectory = Path.GetDirectoryName(fileInfo.DirectoryName);
            if (fileConnectorConfigurationTuple.Item2.FileScanningIntervalInSeconds.Value < 0)
                fileScanningIntervalInSecondsLine = $"-\t{fileConnectorConfigurationTuple.Item2.FileScanningIntervalInSeconds.Value:0000}\t{Path.GetFileName(fileInfo.DirectoryName)}";
            else
                fileScanningIntervalInSecondsLine = $"+\t{fileConnectorConfigurationTuple.Item2.FileScanningIntervalInSeconds.Value:+0000}\t{Path.GetFileName(fileInfo.DirectoryName)}";
            File.AppendAllLines(Path.Combine(versionDirectory, "FileScanningIntervalInSeconds.txt"), new string[] { fileScanningIntervalInSecondsLine });
        }
        Tuple<string, string, string, EquipmentTypeVersion> equipmentTypeVersionTuple = GetEquipmentTypeVersionTuple(cellInstanceVersionTuple.Item2, mbn.CellInstanceConnectionName);
        Tuple<string, string> parameterizedModelObjectDefinitionTypeTuple = GetParameterizedModelObjectDefinitionTypeTuple(equipmentTypeVersionTuple);
        Tuple<string, IList<ModelObjectParameterDefinition>> modelObjectParametersTuple = GetModelObjectParameters(equipmentTypeVersionTuple);
        Tuple<string, string, string, EquipmentDictionaryVersion> equipmentDictionaryVersionTuple = GetEquipmentDictionaryVersionTuple(cellInstanceVersionTuple.Item2, mbn.CellInstanceConnectionName, equipmentTypeVersionTuple.Item4);
        Tuple<string, List<Tuple<string, string>>> equipmentDictionaryIsAlwaysEnabledEventsTuple = GetEquipmentDictionaryIsAlwaysEnabledEventsTuple(equipmentDictionaryVersionTuple);
        Dictionary<string, object> objects = GetKeyValuePairs(mbn.CellInstanceName, mbn.CellInstanceVersionName, mbn.CellInstanceConnectionName, fileConnectorConfigurationTuple.Item2, equipmentTypeVersionTuple.Item2, parameterizedModelObjectDefinitionTypeTuple.Item2, modelObjectParametersTuple.Item2, equipmentDictionaryVersionTuple.Item2, equipmentDictionaryIsAlwaysEnabledEventsTuple.Item2, cellInstanceVersionTuple.Item2.EdaConnection.PortNumber);
        string json = JsonSerializer.Serialize(objects, new JsonSerializerOptions { WriteIndented = true });
        results = new string[] { fileInfo.FullName, json };
        return results;
    }

    public IFileRead Get(MethodBase methodBase, string sourceFileLocation, string sourceFileFilter, bool useCyclicalForDescription)
    {
        IFileRead result;
        MethodBaseName mbn = GetMethodBaseName(methodBase);
        FileInfo fileInfo = new(mbn.FileFullName);
        Dictionary<string, string> fileParameter = new();
        if (!string.IsNullOrEmpty(mbn.CellInstanceConnectionName) && !Directory.Exists(fileInfo.DirectoryName))
            _ = Directory.CreateDirectory(fileInfo.Directory.FullName);
        Dictionary<string, List<long>> dummyRuns = new();
        Dictionary<long, List<string>> staticRuns = new();
        Tuple<string, CellInstanceVersion> cellInstanceVersionTuple = GetCellInstanceVersionTuple(mbn.CellInstanceName, mbn.CellInstanceVersionName);
        Tuple<string, FileConnectorConfiguration> fileConnectorConfigurationTuple = GetFileConnectorConfigurationTuple(cellInstanceVersionTuple, mbn.CellInstanceConnectionName);
        Tuple<string, string, string, EquipmentTypeVersion> equipmentTypeVersionTuple = GetEquipmentTypeVersionTuple(cellInstanceVersionTuple.Item2, mbn.CellInstanceConnectionName);
        Tuple<string, string> parameterizedModelObjectDefinitionTypeTuple = GetParameterizedModelObjectDefinitionTypeTuple(equipmentTypeVersionTuple);
        Tuple<string, IList<ModelObjectParameterDefinition>> modelObjectParametersTuple = GetModelObjectParameters(equipmentTypeVersionTuple);
        Tuple<string, string, string, EquipmentDictionaryVersion> equipmentDictionaryVersionTuple = GetEquipmentDictionaryVersionTuple(cellInstanceVersionTuple.Item2, mbn.CellInstanceConnectionName, equipmentTypeVersionTuple.Item4);
        _ = GetEquipmentDictionaryIsAlwaysEnabledEventsTuple(equipmentDictionaryVersionTuple);
        if (!string.IsNullOrEmpty(sourceFileLocation) && sourceFileLocation != fileConnectorConfigurationTuple.Item2.SourceFileLocation)
            fileConnectorConfigurationTuple.Item2.SourceFileLocation = sourceFileLocation;
        if (!string.IsNullOrEmpty(sourceFileFilter) && sourceFileFilter != fileConnectorConfigurationTuple.Item2.SourceFileFilter)
        {
            fileConnectorConfigurationTuple.Item2.SourceFileFilter = sourceFileFilter;
            fileConnectorConfigurationTuple.Item2.SourceFileFilters = sourceFileFilter.Split('|').ToList();
        }
        if (_TestContext.FullyQualifiedTestClassName.Contains(nameof(Extract)))
        {
            try
            {
                if (!string.IsNullOrEmpty(fileConnectorConfigurationTuple.Item2.ErrorTargetFileLocation))
                {
                    if (!Directory.Exists(fileConnectorConfigurationTuple.Item2.ErrorTargetFileLocation))
                        _ = Directory.CreateDirectory(fileConnectorConfigurationTuple.Item2.ErrorTargetFileLocation);
                }
                if (!string.IsNullOrEmpty(fileConnectorConfigurationTuple.Item2.SourceFileLocation))
                {
                    if (!Directory.Exists(fileConnectorConfigurationTuple.Item2.SourceFileLocation))
                        _ = Directory.CreateDirectory(fileConnectorConfigurationTuple.Item2.SourceFileLocation);
                }
                if (!string.IsNullOrEmpty(fileConnectorConfigurationTuple.Item2.TargetFileLocation))
                {
                    if (!Directory.Exists(fileConnectorConfigurationTuple.Item2.TargetFileLocation))
                        _ = Directory.CreateDirectory(fileConnectorConfigurationTuple.Item2.TargetFileLocation);
                }
                if (!string.IsNullOrEmpty(fileConnectorConfigurationTuple.Item2.AlternateTargetFolder))
                {
                    if (!Directory.Exists(fileConnectorConfigurationTuple.Item2.AlternateTargetFolder.Split('|')[0]))
                        _ = Directory.CreateDirectory(fileConnectorConfigurationTuple.Item2.AlternateTargetFolder.Split('|')[0]);
                }
            }
            catch (IOException ex)
            {
                if (!ex.Message.Contains("SMB1"))
                    throw;
            }
        }
        result = FileHandlers.CellInstanceConnectionName.Get(this, fileParameter, mbn.CellInstanceName, mbn.CellInstanceConnectionName, fileConnectorConfigurationTuple.Item2, equipmentTypeVersionTuple.Item2, parameterizedModelObjectDefinitionTypeTuple.Item2, modelObjectParametersTuple.Item2, equipmentDictionaryVersionTuple.Item2, dummyRuns, staticRuns, useCyclicalForDescription, connectionCount: cellInstanceVersionTuple.Item2.EquipmentConnections.Length);
        return result;
    }

    public string[] GetVariables(MethodBase methodBase, string check, bool validatePDSF = true)
    {
        string[] results;
        string[] lines;
        string ipdsfFile;
        string[] segments;
        string fileNameWithoutExtension;
        string searchPattern = "*.ipdsf";
        string methodBaseNameWithActualCICN;
        string sourceFileFilter = string.Empty;
        string sourceFileLocation = string.Empty;
        MethodBaseName mbn = GetMethodBaseName(methodBase);
        string[] textFiles = GetTextFiles(mbn);
        if (textFiles.Length == 0)
        {
            if (_HasWaitForProperty)
                throw new Exception("Set text file!");
            sourceFileLocation = mbn.DummyDirectory;
        }
        foreach (string textFile in textFiles)
        {
            lines = File.ReadAllLines(textFile);
            if (lines.Length != 1)
                continue;
            fileNameWithoutExtension = Path.GetFileNameWithoutExtension(textFile);
            if (fileNameWithoutExtension == nameof(FileConnectorConfiguration.SourceFileFilter))
                sourceFileFilter = lines[0];
            else if (fileNameWithoutExtension == nameof(FileConnectorConfiguration.SourceFileLocation))
            {
                methodBaseNameWithActualCICN = GetMethodBaseNameWithActualCICN(methodBase.Name, mbn.RawCellInstanceName, mbn.CellInstanceConnectionNameFromMethodBaseName, mbn.CellInstanceConnectionName, mbn.Ticks);
                segments = lines[0].Split(new string[] { mbn.Ticks }, StringSplitOptions.None);
                if (segments.Length > 2)
                    throw new Exception("Ticks should only appear once in source file location!");
                if (segments.Length != 2)
                    throw new Exception("Ticks missing from source file location!");
                if (segments[1].Contains(mbn.Ticks))
                    throw new Exception("From source file location path should not contain ticks!");
                if (!segments[1].EndsWith(methodBaseNameWithActualCICN.Replace(mbn.Ticks, string.Empty)))
                    throw new Exception("Method name missing from source file location!");
                sourceFileLocation = lines[0];
            }
        }
        FileInfo fileInfo = new(mbn.FileFullName);
        if (!Directory.Exists(fileInfo.Directory.FullName))
            _ = Directory.CreateDirectory(fileInfo.Directory.FullName);
        if (!fileInfo.Exists)
            throw new Exception();
        string json = File.ReadAllText(fileInfo.FullName);
        if (!json.Contains(check))
            throw new Exception();
        if (!json.Contains(nameof(FileConnectorConfiguration)))
            throw new Exception();
        JsonElement jsonElement = JsonSerializer.Deserialize<JsonElement>(json);
        JsonElement fileConnectorConfigurationJsonElement = jsonElement.GetProperty(nameof(FileConnectorConfiguration));
        if (fileConnectorConfigurationJsonElement.ValueKind != JsonValueKind.Object)
            throw new Exception();
        JsonSerializerOptions jsonSerializerOptions = new()
        { Converters = { new JsonStringEnumConverter(JsonNamingPolicy.CamelCase) } };
        FileConnectorConfiguration fileConnectorConfiguration = JsonSerializer.Deserialize<FileConnectorConfiguration>(fileConnectorConfigurationJsonElement.ToString(), jsonSerializerOptions);
        if (!string.IsNullOrEmpty(sourceFileFilter))
            fileConnectorConfiguration.SourceFileFilter = sourceFileFilter;
        if (!string.IsNullOrEmpty(sourceFileLocation))
            fileConnectorConfiguration.SourceFileLocation = sourceFileLocation;
        if (!validatePDSF)
            ipdsfFile = string.Empty;
        else
        {
            if (string.IsNullOrEmpty(sourceFileLocation))
                ipdsfFile = searchPattern;
            else
            {
                string ipdsfDirectory = Path.Combine(sourceFileLocation, "ipdsf");
                if (!Directory.Exists(ipdsfDirectory))
                    ipdsfFile = searchPattern;
                else
                {
                    string[] files = Directory.GetFiles(ipdsfDirectory, searchPattern, SearchOption.TopDirectoryOnly);
                    if (files.Length != 0)
                        ipdsfFile = files[0];
                    else
                        ipdsfFile = searchPattern;
                }
            }
            if (ipdsfFile == searchPattern)
                throw new Exception();
        }
        results = new string[] { fileInfo.FullName, json, fileConnectorConfiguration.SourceFileLocation, fileConnectorConfiguration.SourceFileFilter, ipdsfFile, mbn.TextFileDirectory };
        if (string.IsNullOrEmpty(results[0]))
            throw new Exception();
        if (string.IsNullOrEmpty(results[1]))
            throw new Exception();
        if (string.IsNullOrEmpty(results[2]))
            throw new Exception();
        if (string.IsNullOrEmpty(results[3]))
            throw new Exception();
        if (validatePDSF && string.IsNullOrEmpty(results[4]))
            throw new Exception();
        if (string.IsNullOrEmpty(results[5]))
            throw new Exception();
        return results;
    }

    internal static Tuple<string, string[], string[]> GetLogisticsColumnsAndBody(string fileFullName)
    {
        Tuple<string, string[], string[]> results;
        results = ProcessDataStandardFormat.GetLogisticsColumnsAndBody(fileFullName);
        Assert.IsFalse(string.IsNullOrEmpty(results.Item1));
        Assert.IsTrue(results.Item2.Length > 0, "Column check");
        Assert.IsTrue(results.Item3.Length > 0, "Body check");
        return results;
    }

    internal static Tuple<string, string[], string[]> GetLogisticsColumnsAndBody(string searchDirectory, string searchPattern)
    {
        Tuple<string, string[], string[]> results;
        if (searchPattern.Length > 3 && !searchPattern.Contains('*') && File.Exists(searchPattern))
            results = GetLogisticsColumnsAndBody(searchPattern);
        else
        {
            string[] pdsfFiles;
            pdsfFiles = Directory.GetFiles(searchDirectory, searchPattern, SearchOption.TopDirectoryOnly);
            if (pdsfFiles.Length == 0)
                _ = Process.Start("explorer.exe", searchDirectory);
            Assert.IsTrue(pdsfFiles.Length != 0, "GetFiles check");
            results = GetLogisticsColumnsAndBody(pdsfFiles[0]);
        }
        Assert.IsFalse(string.IsNullOrEmpty(results.Item1));
        Assert.IsTrue(results.Item2.Length > 0, "Column check");
        Assert.IsTrue(results.Item3.Length > 0, "Body check");
        return results;
    }

    internal static Tuple<string, string[], string[]> GetLogisticsColumnsAndBody(IFileRead fileRead, Logistics logistics, Tuple<string, Test[], JsonElement[], List<FileInfo>> extractResult, Tuple<string, string[], string[]> pdsf)
    {
        Tuple<string, string[], string[]> results;
        string text = ProcessDataStandardFormat.GetPDSFText(fileRead, logistics, extractResult.Item3, logisticsText: pdsf.Item1);
        string[] lines = text.Split(new string[] { System.Environment.NewLine }, StringSplitOptions.RemoveEmptyEntries);
        results = ProcessDataStandardFormat.GetLogisticsColumnsAndBody(logistics.ReportFullPath, lines);
        Assert.IsFalse(string.IsNullOrEmpty(results.Item1));
        Assert.IsTrue(results.Item2.Length > 0, "Column check");
        Assert.IsTrue(results.Item3.Length > 0, "Body check");
        return results;
    }

    internal static string[] GetItem2(Tuple<string, string[], string[]> pdsf, Tuple<string, string[], string[]> pdsfNew)
    {
        JsonSerializerOptions jsonSerializerOptions = new() { WriteIndented = true };
        string jsonOld = JsonSerializer.Serialize(pdsf.Item2, pdsf.Item2.GetType(), jsonSerializerOptions);
        string jsonNew = JsonSerializer.Serialize(pdsfNew.Item2, pdsfNew.Item2.GetType(), jsonSerializerOptions);
        return new string[] { jsonOld, jsonNew };
    }

    internal static string[] GetItem3(Tuple<string, string[], string[]> pdsf, Tuple<string, string[], string[]> pdsfNew)
    {
        string joinOld = string.Join(System.Environment.NewLine, from l in pdsf.Item3 select string.Join('\t', from t in l.Split('\t') where !t.Contains(@"\\") select t));
        string joinNew = string.Join(System.Environment.NewLine, from l in pdsfNew.Item3 select string.Join('\t', from t in l.Split('\t') where !t.Contains(@"\\") select t));
        return new string[] { joinOld, joinNew };
    }

    internal static void UpdatePassDirectory(string searchDirectory)
    {
        DateTime dateTime = DateTime.Now;
        try
        { Directory.SetLastWriteTime(searchDirectory, dateTime); }
        catch (Exception) { }
        string ticksDirectory = Path.GetDirectoryName(searchDirectory);
        try
        { Directory.SetLastWriteTime(ticksDirectory, dateTime); }
        catch (Exception) { }
        string[] directories = Directory.GetDirectories(searchDirectory, "*", SearchOption.TopDirectoryOnly);
        foreach (string directory in directories)
        {
            try
            { Directory.SetLastWriteTime(directory, dateTime); }
            catch (Exception) { }
        }
    }

    internal static string GetFileName(MethodBase methodBase)
    {
        string result;
        string connectionName;
        string separator = "__";
        string connectionNameAndTicks;
        string[] segments = methodBase.Name.Split(new string[] { separator }, StringSplitOptions.None);
        string environment = segments[0];
        string rawVersionName = segments[1];
        string equipmentTypeDirectory = segments[2];
        string ticks = DateTime.Now.Ticks.ToString();
        string comment = segments[segments.Length - 1];
        string versionName = segments[1].Replace('_', '.');
        string before = string.Concat(environment, separator, rawVersionName, separator, equipmentTypeDirectory, separator);
        string after = methodBase.Name.Substring(before.Length);
        if (after.Length < ticks.Length)
        {
            connectionName = after;
        }
        else
        {
            connectionNameAndTicks = after.Substring(0, after.Length - 2 - comment.Length);
            connectionName = connectionNameAndTicks.Substring(0, connectionNameAndTicks.Length - ticks.Length);
            ticks = connectionNameAndTicks.Substring(connectionName.Length);
        }
        result = Path.Combine(environment, equipmentTypeDirectory, versionName, $"{environment}__{rawVersionName}__{equipmentTypeDirectory}__{connectionName}", ticks, $"{connectionName.Replace('_', '-')}.json");
        if (result.Contains('/'))
            result = string.Concat('/', result);
        else
            result = string.Concat('\\', result);
        return result;
    }

    internal static void CompareSaveTSV(string textFileDirectory, string[] join)
    {
        if (join[0] != join[1])
        {
            _ = Process.Start("explorer.exe", textFileDirectory);
            File.WriteAllText(Path.Combine(textFileDirectory, "0.tsv"), join[0]);
            File.WriteAllText(Path.Combine(textFileDirectory, "1.tsv"), join[1]);
        }
    }

    internal static void CompareSaveJSON(string textFileDirectory, string[] json)
    {
        if (json[0] != json[1])
        {
            _ = Process.Start("explorer.exe", textFileDirectory);
            File.WriteAllText(Path.Combine(textFileDirectory, "0.json"), json[0]);
            File.WriteAllText(Path.Combine(textFileDirectory, "1.json"), json[1]);
        }
    }

    internal static void CompareSave(string textFileDirectory, Tuple<string, string[], string[]> pdsf, Tuple<string, string[], string[]> pdsfNew)
    {
        if (pdsf.Item1 != pdsfNew.Item1)
        {
            _ = Process.Start("explorer.exe", textFileDirectory);
            File.WriteAllText(Path.Combine(textFileDirectory, "0.dat"), pdsf.Item1);
            File.WriteAllText(Path.Combine(textFileDirectory, "1.dat"), pdsfNew.Item1);
        }
    }

    internal static IFileRead GetWriteConfigurationGetFileRead(MethodBase methodBase, string check, AdaptationTesting adaptationTesting)
    {
        IFileRead result;
        string[] fileNameAndJson = adaptationTesting.GetConfiguration(methodBase);
        Assert.IsTrue(fileNameAndJson[1].Contains(check));
        File.WriteAllText(fileNameAndJson[0], fileNameAndJson[1]);
        result = adaptationTesting.Get(methodBase, sourceFileLocation: string.Empty, sourceFileFilter: string.Empty, useCyclicalForDescription: false);
        Assert.IsFalse(string.IsNullOrEmpty(result.CellInstanceConnectionName));
        return result;
    }

    internal static string ReExtractCompareUpdatePassDirectory(string[] variables, IFileRead fileRead, Logistics logistics, bool validatePDSF = true)
    {
        string result;
        Tuple<string, Test[], JsonElement[], List<FileInfo>> extractResult = fileRead.ReExtract();
        if (extractResult is null)
            throw new Exception($"Using pattern {variables[4]} no file was found <{variables[2]}>");
        if (!fileRead.IsDuplicator)
        {
            Assert.IsFalse(string.IsNullOrEmpty(extractResult?.Item1));
            Assert.IsNotNull(extractResult.Item3);
            Assert.IsNotNull(extractResult.Item4);
            if (!validatePDSF)
                _ = GetLogisticsColumnsAndBody(fileRead, logistics, extractResult, new(string.Empty, Array.Empty<string>(), Array.Empty<string>()));
            else
            {
                Assert.IsTrue(extractResult.Item3.Length > 0, "extractResult Array Length check!");
                Tuple<string, string[], string[]> pdsf = GetLogisticsColumnsAndBody(variables[2], variables[4]);
                Tuple<string, string[], string[]> pdsfNew = GetLogisticsColumnsAndBody(fileRead, logistics, extractResult, pdsf);
                CompareSave(variables[5], pdsf, pdsfNew);
                Assert.IsTrue(pdsf.Item1 == pdsfNew.Item1, "Item1 check!");
                string[] json = GetItem2(pdsf, pdsfNew);
                CompareSaveJSON(variables[5], json);
                Assert.IsTrue(json[0] == json[1], "Item2 check!");
                string[] join = GetItem3(pdsf, pdsfNew);
                CompareSaveTSV(variables[5], join);
                Assert.IsTrue(join[0] == join[1], "Item3 (Join) check!");
            }
            UpdatePassDirectory(variables[2]);
        }
        result = extractResult.Item1;
        return result;
    }

}
// namespace Adaptation._Tests.Helpers { public class AdaptationTesting { } }
// 2022-08-05 -> AdaptationTesting