2023-09-20 12:02:32 +02:00
|
|
|
|
using CodexContractsPlugin;
|
|
|
|
|
using CodexPlugin;
|
|
|
|
|
using Core;
|
|
|
|
|
using GethPlugin;
|
2023-06-22 14:37:37 +02:00
|
|
|
|
using KubernetesWorkflow;
|
2023-06-26 15:37:16 +02:00
|
|
|
|
using Logging;
|
2023-09-20 12:02:32 +02:00
|
|
|
|
using MetricsPlugin;
|
2023-06-22 10:17:12 +02:00
|
|
|
|
|
|
|
|
|
namespace CodexNetDeployer
|
|
|
|
|
{
|
|
|
|
|
public class Deployer
|
|
|
|
|
{
|
|
|
|
|
private readonly Configuration config;
|
2023-08-24 11:32:32 +02:00
|
|
|
|
private readonly PeerConnectivityChecker peerConnectivityChecker;
|
2023-09-20 12:02:32 +02:00
|
|
|
|
private readonly EntryPoint entryPoint;
|
2023-09-28 11:31:09 +02:00
|
|
|
|
private readonly LocalCodexBuilder localCodexBuilder;
|
2023-06-22 10:17:12 +02:00
|
|
|
|
|
|
|
|
|
public Deployer(Configuration config)
|
|
|
|
|
{
|
|
|
|
|
this.config = config;
|
2023-08-24 11:32:32 +02:00
|
|
|
|
peerConnectivityChecker = new PeerConnectivityChecker();
|
2023-09-28 11:31:09 +02:00
|
|
|
|
localCodexBuilder = new LocalCodexBuilder(new ConsoleLog(), config.CodexLocalRepoPath);
|
2023-09-20 13:56:01 +02:00
|
|
|
|
|
|
|
|
|
ProjectPlugin.Load<CodexPlugin.CodexPlugin>();
|
|
|
|
|
ProjectPlugin.Load<CodexContractsPlugin.CodexContractsPlugin>();
|
|
|
|
|
ProjectPlugin.Load<GethPlugin.GethPlugin>();
|
|
|
|
|
ProjectPlugin.Load<MetricsPlugin.MetricsPlugin>();
|
2023-09-20 12:02:32 +02:00
|
|
|
|
entryPoint = CreateEntryPoint(new NullLog());
|
2023-06-22 10:17:12 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
public void AnnouncePlugins()
|
2023-06-22 10:33:21 +02:00
|
|
|
|
{
|
2023-09-20 12:02:32 +02:00
|
|
|
|
var ep = CreateEntryPoint(new ConsoleLog());
|
2023-06-22 10:33:21 +02:00
|
|
|
|
|
2023-09-28 11:31:09 +02:00
|
|
|
|
localCodexBuilder.Intialize();
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Using plugins:" + Environment.NewLine);
|
|
|
|
|
var metadata = ep.GetPluginMetadata();
|
2023-09-21 08:49:09 +02:00
|
|
|
|
var longestKey = metadata.Keys.Max(k => k.Length);
|
2023-09-20 12:02:32 +02:00
|
|
|
|
foreach (var entry in metadata)
|
|
|
|
|
{
|
2023-09-21 08:49:09 +02:00
|
|
|
|
Console.Write(entry.Key);
|
|
|
|
|
Console.CursorLeft = longestKey + 5;
|
|
|
|
|
Console.WriteLine($"= {entry.Value}");
|
2023-09-20 12:02:32 +02:00
|
|
|
|
}
|
|
|
|
|
Log("");
|
|
|
|
|
}
|
2023-06-22 10:33:21 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
public CodexDeployment Deploy()
|
|
|
|
|
{
|
2023-09-28 11:31:09 +02:00
|
|
|
|
localCodexBuilder.Build();
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Initializing...");
|
|
|
|
|
var ci = entryPoint.CreateInterface();
|
2023-06-22 10:33:21 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Deploying Geth instance...");
|
2023-09-21 14:48:41 +02:00
|
|
|
|
var gethDeployment = ci.DeployGeth(s => s.IsMiner().WithName("geth"));
|
2023-09-20 12:02:32 +02:00
|
|
|
|
var gethNode = ci.WrapGethDeployment(gethDeployment);
|
2023-06-22 14:37:37 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Geth started. Deploying Codex contracts...");
|
|
|
|
|
var contractsDeployment = ci.DeployCodexContracts(gethNode);
|
|
|
|
|
var contracts = ci.WrapCodexContractsDeployment(contractsDeployment);
|
|
|
|
|
Log("Codex contracts deployed.");
|
2023-06-30 09:54:13 +02:00
|
|
|
|
|
2023-06-22 14:37:37 +02:00
|
|
|
|
Log("Starting Codex nodes...");
|
2023-09-20 12:02:32 +02:00
|
|
|
|
var codexStarter = new CodexNodeStarter(config, ci, gethNode, contracts, config.NumberOfValidators!.Value);
|
2023-08-24 11:32:32 +02:00
|
|
|
|
var startResults = new List<CodexNodeStartResult>();
|
2023-06-22 14:37:37 +02:00
|
|
|
|
for (var i = 0; i < config.NumberOfCodexNodes; i++)
|
2023-06-22 10:33:21 +02:00
|
|
|
|
{
|
2023-08-24 11:32:32 +02:00
|
|
|
|
var result = codexStarter.Start(i);
|
|
|
|
|
if (result != null) startResults.Add(result);
|
2023-06-22 10:33:21 +02:00
|
|
|
|
}
|
2023-06-23 08:44:27 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Codex nodes started.");
|
|
|
|
|
var metricsService = StartMetricsService(ci, startResults);
|
2023-07-11 10:59:41 +02:00
|
|
|
|
|
2023-08-24 11:32:32 +02:00
|
|
|
|
CheckPeerConnectivity(startResults);
|
2023-08-28 11:53:59 +02:00
|
|
|
|
CheckContainerRestarts(startResults);
|
2023-08-24 11:32:32 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
var codexContainers = startResults.Select(s => s.CodexNode.Container).ToArray();
|
|
|
|
|
return new CodexDeployment(codexContainers, gethDeployment, metricsService, CreateMetadata());
|
2023-06-22 10:33:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
private EntryPoint CreateEntryPoint(ILog log)
|
2023-06-22 10:17:12 +02:00
|
|
|
|
{
|
2023-06-27 08:29:39 +02:00
|
|
|
|
var kubeConfig = GetKubeConfig(config.KubeConfigFile);
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
var configuration = new KubernetesWorkflow.Configuration(
|
|
|
|
|
kubeConfig,
|
2023-10-05 08:58:05 +02:00
|
|
|
|
operationTimeout: TimeSpan.FromSeconds(300),
|
|
|
|
|
retryDelay: TimeSpan.FromSeconds(10),
|
2023-09-20 12:02:32 +02:00
|
|
|
|
kubernetesNamespace: config.KubeNamespace);
|
|
|
|
|
|
2023-09-26 14:32:28 +02:00
|
|
|
|
var result = new EntryPoint(log, configuration, string.Empty);
|
|
|
|
|
configuration.Hooks = new K8sHook(config.TestsTypePodLabel, result.GetPluginMetadata());
|
2023-09-22 11:10:22 +02:00
|
|
|
|
|
2023-09-26 14:32:28 +02:00
|
|
|
|
return result;
|
2023-06-22 10:33:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
private RunningContainer? StartMetricsService(CoreInterface ci, List<CodexNodeStartResult> startResults)
|
2023-07-11 10:59:41 +02:00
|
|
|
|
{
|
2023-09-20 12:02:32 +02:00
|
|
|
|
if (!config.Metrics) return null;
|
2023-07-11 12:21:48 +02:00
|
|
|
|
|
2023-07-11 10:59:41 +02:00
|
|
|
|
Log("Starting metrics service...");
|
2023-08-13 09:07:23 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
var runningContainer = ci.DeployMetricsCollector(startResults.Select(r => r.CodexNode).ToArray());
|
|
|
|
|
|
|
|
|
|
Log("Metrics service started.");
|
2023-08-13 09:07:23 +02:00
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
return runningContainer;
|
2023-07-11 10:59:41 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 08:29:39 +02:00
|
|
|
|
private string? GetKubeConfig(string kubeConfigFile)
|
|
|
|
|
{
|
|
|
|
|
if (string.IsNullOrEmpty(kubeConfigFile) || kubeConfigFile.ToLowerInvariant() == "null") return null;
|
|
|
|
|
return kubeConfigFile;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-24 11:32:32 +02:00
|
|
|
|
private void CheckPeerConnectivity(List<CodexNodeStartResult> codexContainers)
|
|
|
|
|
{
|
|
|
|
|
if (!config.CheckPeerConnection) return;
|
|
|
|
|
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Starting peer connectivity check for deployed nodes...");
|
2023-08-24 11:32:32 +02:00
|
|
|
|
peerConnectivityChecker.CheckConnectivity(codexContainers);
|
|
|
|
|
Log("Check passed.");
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-28 11:53:59 +02:00
|
|
|
|
private void CheckContainerRestarts(List<CodexNodeStartResult> startResults)
|
|
|
|
|
{
|
|
|
|
|
var crashes = new List<RunningContainer>();
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Starting container crash check...");
|
2023-08-28 11:53:59 +02:00
|
|
|
|
foreach (var startResult in startResults)
|
|
|
|
|
{
|
2023-09-20 12:55:09 +02:00
|
|
|
|
var watcher = startResult.CodexNode.CrashWatcher;
|
2023-09-20 12:02:32 +02:00
|
|
|
|
if (watcher == null) throw new Exception("Expected each CodexNode container to be created with a crash-watcher.");
|
|
|
|
|
if (watcher.HasContainerCrashed()) crashes.Add(startResult.CodexNode.Container);
|
2023-08-28 11:53:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!crashes.Any())
|
|
|
|
|
{
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log("Check passed.");
|
2023-08-28 11:53:59 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2023-09-20 12:02:32 +02:00
|
|
|
|
Log($"Check failed. The following containers have crashed: {string.Join(",", crashes.Select(c => c.Name))}");
|
2023-08-28 11:53:59 +02:00
|
|
|
|
throw new Exception("Deployment failed: One or more containers crashed.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-23 09:08:18 +02:00
|
|
|
|
private DeploymentMetadata CreateMetadata()
|
|
|
|
|
{
|
|
|
|
|
return new DeploymentMetadata(
|
|
|
|
|
kubeNamespace: config.KubeNamespace,
|
|
|
|
|
numberOfCodexNodes: config.NumberOfCodexNodes!.Value,
|
|
|
|
|
numberOfValidators: config.NumberOfValidators!.Value,
|
|
|
|
|
storageQuotaMB: config.StorageQuota!.Value,
|
2023-06-28 08:48:46 +02:00
|
|
|
|
codexLogLevel: config.CodexLogLevel,
|
|
|
|
|
initialTestTokens: config.InitialTestTokens,
|
|
|
|
|
minPrice: config.MinPrice,
|
|
|
|
|
maxCollateral: config.MaxCollateral,
|
2023-08-23 08:29:16 +02:00
|
|
|
|
maxDuration: config.MaxDuration,
|
|
|
|
|
blockTTL: config.BlockTTL,
|
|
|
|
|
blockMI: config.BlockMI,
|
|
|
|
|
blockMN: config.BlockMN);
|
2023-06-23 09:08:18 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-06-22 10:33:21 +02:00
|
|
|
|
private void Log(string msg)
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine(msg);
|
2023-06-22 10:17:12 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|