2023-11-13 11:56:02 +01:00
|
|
|
|
using DistTestCore;
|
|
|
|
|
using DistTestCore.Logs;
|
2023-06-28 15:11:20 +02:00
|
|
|
|
using Logging;
|
2023-10-23 14:17:52 +02:00
|
|
|
|
using Newtonsoft.Json;
|
2023-09-27 11:33:54 +02:00
|
|
|
|
using Utils;
|
2023-06-25 11:06:47 +02:00
|
|
|
|
|
|
|
|
|
namespace ContinuousTests
|
2023-06-21 08:28:40 +02:00
|
|
|
|
{
|
2023-06-25 09:53:10 +02:00
|
|
|
|
public class ContinuousTestRunner
|
2023-06-21 08:28:40 +02:00
|
|
|
|
{
|
2023-09-20 13:33:58 +02:00
|
|
|
|
private readonly EntryPointFactory entryPointFactory = new EntryPointFactory();
|
2023-06-21 08:28:40 +02:00
|
|
|
|
private readonly ConfigLoader configLoader = new ConfigLoader();
|
2023-06-21 09:27:59 +02:00
|
|
|
|
private readonly TestFactory testFactory = new TestFactory();
|
2023-06-25 09:53:10 +02:00
|
|
|
|
private readonly Configuration config;
|
2023-06-28 16:19:37 +02:00
|
|
|
|
private readonly CancellationToken cancelToken;
|
2023-06-21 10:06:54 +02:00
|
|
|
|
|
2023-06-28 16:19:37 +02:00
|
|
|
|
public ContinuousTestRunner(string[] args, CancellationToken cancelToken)
|
2023-06-21 10:06:54 +02:00
|
|
|
|
{
|
2023-06-26 14:44:21 +02:00
|
|
|
|
config = configLoader.Load(args);
|
2023-06-28 16:19:37 +02:00
|
|
|
|
this.cancelToken = cancelToken;
|
2023-06-21 10:06:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-06-25 09:53:10 +02:00
|
|
|
|
public void Run()
|
2023-06-21 08:28:40 +02:00
|
|
|
|
{
|
2023-11-08 09:24:39 +01:00
|
|
|
|
var logConfig = new LogConfig(config.LogPath);
|
2023-09-28 09:39:15 +02:00
|
|
|
|
var startTime = DateTime.UtcNow;
|
|
|
|
|
|
2023-09-21 10:33:09 +02:00
|
|
|
|
var overviewLog = new LogSplitter(
|
2023-09-28 09:39:15 +02:00
|
|
|
|
new FixtureLog(logConfig, startTime, "Overview"),
|
2023-09-21 10:33:09 +02:00
|
|
|
|
new ConsoleLog()
|
|
|
|
|
);
|
2023-11-10 15:28:53 +01:00
|
|
|
|
var statusLog = new StatusLog(logConfig, startTime, "continuous-tests", "ContinuousTestRun");
|
2023-09-21 10:33:09 +02:00
|
|
|
|
|
|
|
|
|
overviewLog.Log("Initializing...");
|
2023-09-20 13:33:58 +02:00
|
|
|
|
|
|
|
|
|
var entryPoint = entryPointFactory.CreateEntryPoint(config.KubeConfigFile, config.DataPath, config.CodexDeployment.Metadata.KubeNamespace, overviewLog);
|
|
|
|
|
entryPoint.Announce();
|
|
|
|
|
|
2023-09-21 10:33:09 +02:00
|
|
|
|
overviewLog.Log("Initialized. Performing startup checks...");
|
|
|
|
|
|
2023-09-20 13:33:58 +02:00
|
|
|
|
var startupChecker = new StartupChecker(entryPoint, config, cancelToken);
|
2023-06-25 09:53:10 +02:00
|
|
|
|
startupChecker.Check();
|
2023-06-21 08:28:40 +02:00
|
|
|
|
|
2023-06-28 16:19:37 +02:00
|
|
|
|
var taskFactory = new TaskFactory();
|
2023-10-23 14:17:52 +02:00
|
|
|
|
overviewLog.Log("Startup checks passed. Configuration:");
|
|
|
|
|
overviewLog.Log(JsonConvert.SerializeObject(config, Formatting.Indented));
|
2023-11-13 11:56:02 +01:00
|
|
|
|
overviewLog.Log("Test framework revision: " + GitInfo.GetStatus());
|
2023-10-23 14:17:52 +02:00
|
|
|
|
overviewLog.Log("Continuous tests starting...");
|
2023-09-21 10:33:09 +02:00
|
|
|
|
overviewLog.Log("");
|
2023-06-25 09:53:10 +02:00
|
|
|
|
var allTests = testFactory.CreateTests();
|
2023-06-28 15:11:20 +02:00
|
|
|
|
|
|
|
|
|
ClearAllCustomNamespaces(allTests, overviewLog);
|
|
|
|
|
|
2023-10-03 09:53:02 +02:00
|
|
|
|
var filteredTests = FilterTests(allTests, overviewLog);
|
|
|
|
|
if (!filteredTests.Any())
|
|
|
|
|
{
|
|
|
|
|
overviewLog.Log("No tests selected.");
|
2023-10-31 15:33:00 +01:00
|
|
|
|
Cancellation.Cts.Cancel();
|
2023-10-03 09:53:02 +02:00
|
|
|
|
}
|
2023-10-31 15:33:00 +01:00
|
|
|
|
else
|
2023-06-21 08:28:40 +02:00
|
|
|
|
{
|
2023-11-09 11:35:45 +01:00
|
|
|
|
var testLoops = filteredTests.Select(t => new TestLoop(entryPointFactory, taskFactory, config, overviewLog, statusLog, t.GetType(), t.RunTestEvery, startupChecker, cancelToken)).ToArray();
|
2023-06-28 16:19:37 +02:00
|
|
|
|
|
2023-10-31 15:33:00 +01:00
|
|
|
|
foreach (var testLoop in testLoops)
|
|
|
|
|
{
|
|
|
|
|
if (cancelToken.IsCancellationRequested) break;
|
2023-06-27 10:16:59 +02:00
|
|
|
|
|
2023-10-31 15:33:00 +01:00
|
|
|
|
overviewLog.Log("Launching test-loop for " + testLoop.Name);
|
|
|
|
|
testLoop.Begin();
|
|
|
|
|
Thread.Sleep(TimeSpan.FromSeconds(5));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
overviewLog.Log("Finished launching test-loops.");
|
|
|
|
|
WaitUntilFinished(overviewLog, statusLog, startTime, testLoops);
|
|
|
|
|
overviewLog.Log("Stopping all test-loops...");
|
|
|
|
|
}
|
2023-06-28 16:19:37 +02:00
|
|
|
|
taskFactory.WaitAll();
|
2023-06-29 10:23:04 +02:00
|
|
|
|
overviewLog.Log("All tasks cancelled.");
|
2023-10-03 09:53:02 +02:00
|
|
|
|
|
|
|
|
|
PerformCleanup(overviewLog);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private ContinuousTest[] FilterTests(ContinuousTest[] allTests, ILog log)
|
|
|
|
|
{
|
|
|
|
|
log.Log($"Available tests: {string.Join(", ", allTests.Select(r => r.Name))}");
|
|
|
|
|
|
|
|
|
|
var result = allTests.ToArray();
|
|
|
|
|
var filters = config.Filter.Split(",", StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
|
if (filters.Any())
|
|
|
|
|
{
|
|
|
|
|
log.Log($"Applying filters: {string.Join(", ", filters)}");
|
|
|
|
|
result = allTests.Where(t => filters.Any(f => t.Name.Contains(f))).ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
log.Log($"Selected for running: {string.Join(", ", result.Select(r => r.Name))}");
|
|
|
|
|
return result;
|
2023-06-23 11:38:30 +02:00
|
|
|
|
}
|
2023-06-28 15:11:20 +02:00
|
|
|
|
|
2023-09-28 09:39:15 +02:00
|
|
|
|
private void WaitUntilFinished(LogSplitter overviewLog, StatusLog statusLog, DateTime startTime, TestLoop[] testLoops)
|
2023-09-27 11:33:54 +02:00
|
|
|
|
{
|
2023-11-14 12:56:47 +01:00
|
|
|
|
var testDuration = (DateTime.UtcNow - startTime).TotalSeconds.ToString();
|
2023-09-28 09:39:15 +02:00
|
|
|
|
var testData = FormatTestRuns(testLoops);
|
2023-10-23 10:32:11 +02:00
|
|
|
|
overviewLog.Log("Total duration: " + testDuration);
|
2023-09-28 09:39:15 +02:00
|
|
|
|
|
2023-11-17 14:51:32 +01:00
|
|
|
|
if (!string.IsNullOrEmpty(config.TargetDurationSeconds))
|
2023-09-27 11:33:54 +02:00
|
|
|
|
{
|
2023-11-17 14:51:32 +01:00
|
|
|
|
var targetDuration = Time.ParseTimespan(config.TargetDurationSeconds);
|
2023-10-02 09:24:01 +02:00
|
|
|
|
var wasCancelled = cancelToken.WaitHandle.WaitOne(targetDuration);
|
|
|
|
|
if (!wasCancelled)
|
|
|
|
|
{
|
|
|
|
|
Cancellation.Cts.Cancel();
|
|
|
|
|
overviewLog.Log($"Congratulations! The targer duration has been reached! ({Time.FormatDuration(targetDuration)})");
|
|
|
|
|
statusLog.ConcludeTest("Passed", testDuration, testData);
|
2023-11-13 11:17:26 +01:00
|
|
|
|
Environment.ExitCode = 0;
|
2023-10-02 09:24:01 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2023-09-27 11:33:54 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
cancelToken.WaitHandle.WaitOne();
|
2023-09-28 09:39:15 +02:00
|
|
|
|
}
|
2023-10-02 09:24:01 +02:00
|
|
|
|
statusLog.ConcludeTest("Failed", testDuration, testData);
|
2023-11-13 11:17:26 +01:00
|
|
|
|
Environment.ExitCode = 1;
|
2023-09-28 09:39:15 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Dictionary<string, string> FormatTestRuns(TestLoop[] testLoops)
|
|
|
|
|
{
|
|
|
|
|
var result = new Dictionary<string, string>();
|
|
|
|
|
foreach (var testLoop in testLoops)
|
|
|
|
|
{
|
2023-11-09 11:35:45 +01:00
|
|
|
|
result.Add("testname", testLoop.Name);
|
|
|
|
|
result.Add($"summary", $"passes: {testLoop.NumberOfPasses} - failures: {testLoop.NumberOfFailures}");
|
2023-09-27 11:33:54 +02:00
|
|
|
|
}
|
2023-09-28 09:39:15 +02:00
|
|
|
|
return result;
|
2023-09-27 11:33:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-21 10:33:09 +02:00
|
|
|
|
private void ClearAllCustomNamespaces(ContinuousTest[] allTests, ILog log)
|
2023-06-28 15:11:20 +02:00
|
|
|
|
{
|
|
|
|
|
foreach (var test in allTests) ClearAllCustomNamespaces(test, log);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-21 10:33:09 +02:00
|
|
|
|
private void ClearAllCustomNamespaces(ContinuousTest test, ILog log)
|
2023-06-28 15:11:20 +02:00
|
|
|
|
{
|
|
|
|
|
if (string.IsNullOrEmpty(test.CustomK8sNamespace)) return;
|
|
|
|
|
|
|
|
|
|
log.Log($"Clearing namespace '{test.CustomK8sNamespace}'...");
|
2023-09-20 13:33:58 +02:00
|
|
|
|
|
|
|
|
|
var entryPoint = entryPointFactory.CreateEntryPoint(config.KubeConfigFile, config.DataPath, test.CustomK8sNamespace, log);
|
|
|
|
|
entryPoint.Tools.CreateWorkflow().DeleteNamespacesStartingWith(test.CustomK8sNamespace);
|
2023-06-28 15:11:20 +02:00
|
|
|
|
}
|
2023-10-03 09:53:02 +02:00
|
|
|
|
|
|
|
|
|
private void PerformCleanup(ILog log)
|
|
|
|
|
{
|
|
|
|
|
if (!config.Cleanup) return;
|
|
|
|
|
log.Log("Cleaning up test namespace...");
|
|
|
|
|
|
|
|
|
|
var entryPoint = entryPointFactory.CreateEntryPoint(config.KubeConfigFile, config.DataPath, config.CodexDeployment.Metadata.KubeNamespace, log);
|
|
|
|
|
entryPoint.Decommission(deleteKubernetesResources: true, deleteTrackedFiles: true);
|
|
|
|
|
log.Log("Cleanup finished.");
|
|
|
|
|
}
|
2023-06-21 10:06:54 +02:00
|
|
|
|
}
|
2023-06-21 08:28:40 +02:00
|
|
|
|
}
|