2
0
mirror of synced 2025-01-12 17:44:08 +00:00

173 lines
6.4 KiB
C#
Raw Normal View History

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;
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;
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-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);
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
{
var result = codexStarter.Start(i);
if (result != null) startResults.Add(result);
2023-06-22 10:33:21 +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
CheckPeerConnectivity(startResults);
CheckContainerRestarts(startResults);
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,
operationTimeout: TimeSpan.FromSeconds(300),
retryDelay: TimeSpan.FromSeconds(10),
2023-09-20 12:02:32 +02:00
kubernetesNamespace: config.KubeNamespace);
var result = new EntryPoint(log, configuration, string.Empty);
configuration.Hooks = new K8sHook(config.TestsTypePodLabel, result.GetPluginMetadata());
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 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;
}
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...");
peerConnectivityChecker.CheckConnectivity(codexContainers);
Log("Check passed.");
}
private void CheckContainerRestarts(List<CodexNodeStartResult> startResults)
{
var crashes = new List<RunningContainer>();
2023-09-20 12:02:32 +02:00
Log("Starting container crash check...");
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);
}
if (!crashes.Any())
{
2023-09-20 12:02:32 +02:00
Log("Check passed.");
}
else
{
2023-09-20 12:02:32 +02:00
Log($"Check failed. The following containers have crashed: {string.Join(",", crashes.Select(c => c.Name))}");
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,
codexLogLevel: config.CodexLogLevel,
initialTestTokens: config.InitialTestTokens,
minPrice: config.MinPrice,
maxCollateral: config.MaxCollateral,
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
}
}
}