2
0
mirror of synced 2025-01-12 01:24:23 +00:00

Merge branch 'feature/pluralize-codexnodes-in-pod'

This commit is contained in:
benbierens 2023-03-23 12:37:03 +01:00
commit 6fc3ea3fca
No known key found for this signature in database
GPG Key ID: FE44815D96D0A1AA
21 changed files with 908 additions and 530 deletions

View File

@ -1,61 +0,0 @@
using k8s.Models;
namespace CodexDistTestCore
{
public class ActiveNode
{
public ActiveNode(OfflineCodexNode origin, int port, int orderNumber)
{
Origin = origin;
SelectorName = orderNumber.ToString().PadLeft(6, '0');
Port = port;
}
public OfflineCodexNode Origin { get; }
public string SelectorName { get; }
public int Port { get; }
public V1Deployment? Deployment { get; set; }
public V1Service? Service { get; set; }
public List<string> ActivePodNames { get; } = new List<string>();
public V1ObjectMeta GetServiceMetadata()
{
return new V1ObjectMeta
{
Name = "codex-test-entrypoint-" + SelectorName,
NamespaceProperty = K8sManager.K8sNamespace
};
}
public V1ObjectMeta GetDeploymentMetadata()
{
return new V1ObjectMeta
{
Name = "codex-test-node-" + SelectorName,
NamespaceProperty = K8sManager.K8sNamespace
};
}
public Dictionary<string, string> GetSelector()
{
return new Dictionary<string, string> { { "codex-test-node", "dist-test-" + SelectorName } };
}
public string GetContainerPortName()
{
//Caution, was: "codex-api-port" + SelectorName
//but string length causes 'UnprocessableEntity' exception in k8s.
return "api-" + SelectorName;
}
public string GetContainerName()
{
return "codex-test-node";
}
public string Describe()
{
return $"CodexNode{SelectorName}-Port:{Port}-{Origin.Describe()}";
}
}
}

View File

@ -14,10 +14,10 @@ namespace CodexDistTestCore
return "b20483"; return "b20483";
} }
public List<V1EnvVar> CreateEnvironmentVariables(OfflineCodexNode node) public List<V1EnvVar> CreateEnvironmentVariables(OfflineCodexNodes node, CodexNodeContainer environment)
{ {
var formatter = new EnvFormatter(); var formatter = new EnvFormatter();
formatter.Create(node); formatter.Create(node, environment);
return formatter.Result; return formatter.Result;
} }
@ -25,8 +25,13 @@ namespace CodexDistTestCore
{ {
public List<V1EnvVar> Result { get; } = new List<V1EnvVar>(); public List<V1EnvVar> Result { get; } = new List<V1EnvVar>();
public void Create(OfflineCodexNode node) public void Create(OfflineCodexNodes node, CodexNodeContainer environment)
{ {
AddVar("API_PORT", environment.ApiPort.ToString());
AddVar("DATA_DIR", environment.DataDir);
AddVar("DISC_PORT", environment.DiscoveryPort.ToString());
AddVar("LISTEN_ADDRS", $"/ip4/0.0.0.0/tcp/{environment.ListenPort}");
if (node.BootstrapNode != null) if (node.BootstrapNode != null)
{ {
var debugInfo = node.BootstrapNode.GetDebugInfo(); var debugInfo = node.BootstrapNode.GetDebugInfo();

View File

@ -0,0 +1,75 @@
namespace CodexDistTestCore
{
public class CodexNodeContainer
{
public CodexNodeContainer(string name, int servicePort, string servicePortName, int apiPort, string containerPortName, int discoveryPort, int listenPort, string dataDir)
{
Name = name;
ServicePort = servicePort;
ServicePortName = servicePortName;
ApiPort = apiPort;
ContainerPortName = containerPortName;
DiscoveryPort = discoveryPort;
ListenPort = listenPort;
DataDir = dataDir;
}
public string Name { get; }
public int ServicePort { get; }
public string ServicePortName { get; }
public int ApiPort { get; }
public string ContainerPortName { get; }
public int DiscoveryPort { get; }
public int ListenPort { get; }
public string DataDir { get; }
}
public class CodexGroupNumberSource
{
private readonly NumberSource codexNodeGroupNumberSource = new NumberSource(0);
private readonly NumberSource groupContainerNameSource = new NumberSource(1);
private readonly NumberSource servicePortSource = new NumberSource(30001);
public int GetNextCodexNodeGroupNumber()
{
return codexNodeGroupNumberSource.GetNextNumber();
}
public string GetNextServicePortName()
{
return $"node{groupContainerNameSource.GetNextNumber()}";
}
public int GetNextServicePort()
{
return servicePortSource.GetNextNumber();
}
}
public class CodexNodeContainerFactory
{
private readonly NumberSource containerNameSource = new NumberSource(1);
private readonly NumberSource codexPortSource = new NumberSource(8080);
private readonly CodexGroupNumberSource groupContainerFactory;
public CodexNodeContainerFactory(CodexGroupNumberSource groupContainerFactory)
{
this.groupContainerFactory = groupContainerFactory;
}
public CodexNodeContainer CreateNext()
{
var n = containerNameSource.GetNextNumber();
return new CodexNodeContainer(
name: $"codex-node{n}",
servicePort: groupContainerFactory.GetNextServicePort(),
servicePortName: groupContainerFactory.GetNextServicePortName(),
apiPort: codexPortSource.GetNextNumber(),
containerPortName: $"api-{n}",
discoveryPort: codexPortSource.GetNextNumber(),
listenPort: codexPortSource.GetNextNumber(),
dataDir: $"datadir{n}"
);
}
}
}

View File

@ -0,0 +1,101 @@
using k8s.Models;
using System.Collections;
namespace CodexDistTestCore
{
public interface ICodexNodeGroup : IEnumerable<IOnlineCodexNode>
{
IOfflineCodexNodes BringOffline();
IOnlineCodexNode this[int index] { get; }
}
public class CodexNodeGroup : ICodexNodeGroup
{
private readonly IK8sManager k8SManager;
public CodexNodeGroup(int orderNumber, OfflineCodexNodes origin, IK8sManager k8SManager, OnlineCodexNode[] nodes)
{
OrderNumber = orderNumber;
Origin = origin;
this.k8SManager = k8SManager;
Nodes = nodes;
foreach (var n in nodes) n.Group = this;
}
public IOnlineCodexNode this[int index]
{
get
{
return Nodes[index];
}
}
public IOfflineCodexNodes BringOffline()
{
return k8SManager.BringOffline(this);
}
public int OrderNumber { get; }
public OfflineCodexNodes Origin { get; }
public OnlineCodexNode[] Nodes { get; }
public V1Deployment? Deployment { get; set; }
public V1Service? Service { get; set; }
public PodInfo? PodInfo { get; set; }
public CodexNodeContainer[] GetContainers()
{
return Nodes.Select(n => n.Container).ToArray();
}
public IEnumerator<IOnlineCodexNode> GetEnumerator()
{
return Nodes.Cast<IOnlineCodexNode>().GetEnumerator();
}
IEnumerator IEnumerable.GetEnumerator()
{
return Nodes.GetEnumerator();
}
public V1ObjectMeta GetServiceMetadata()
{
return new V1ObjectMeta
{
Name = "codex-test-entrypoint-" + OrderNumber,
NamespaceProperty = K8sOperations.K8sNamespace
};
}
public V1ObjectMeta GetDeploymentMetadata()
{
return new V1ObjectMeta
{
Name = "codex-test-node-" + OrderNumber,
NamespaceProperty = K8sOperations.K8sNamespace
};
}
public Dictionary<string, string> GetSelector()
{
return new Dictionary<string, string> { { "codex-test-node", "dist-test-" + OrderNumber } };
}
public string Describe()
{
return $"CodexNodeGroup#{OrderNumber}-{Origin.Describe()}";
}
}
public class PodInfo
{
public PodInfo(string name, string ip)
{
Name = name;
Ip = ip;
}
public string Name { get; }
public string Ip { get; }
}
}

View File

@ -5,6 +5,7 @@ namespace CodexDistTestCore
[SetUpFixture] [SetUpFixture]
public abstract class DistTest public abstract class DistTest
{ {
private TestLog log = null!;
private FileManager fileManager = null!; private FileManager fileManager = null!;
private K8sManager k8sManager = null!; private K8sManager k8sManager = null!;
@ -13,12 +14,23 @@ namespace CodexDistTestCore
{ {
// Previous test run may have been interrupted. // Previous test run may have been interrupted.
// Begin by cleaning everything up. // Begin by cleaning everything up.
fileManager = new FileManager(); log = new TestLog();
k8sManager = new K8sManager(fileManager); fileManager = new FileManager(log);
k8sManager = new K8sManager(log, fileManager);
try
{
k8sManager.DeleteAllResources(); k8sManager.DeleteAllResources();
fileManager.DeleteAllTestFiles(); fileManager.DeleteAllTestFiles();
} }
catch (Exception ex)
{
GlobalTestFailure.HasFailed = true;
log.Error($"Global setup cleanup failed with: {ex}");
throw;
}
log.Log("Global setup cleanup successful");
}
[SetUp] [SetUp]
public void SetUpDistTest() public void SetUpDistTest()
@ -29,9 +41,9 @@ namespace CodexDistTestCore
} }
else else
{ {
TestLog.BeginTest(); log = new TestLog();
fileManager = new FileManager(); fileManager = new FileManager(log);
k8sManager = new K8sManager(fileManager); k8sManager = new K8sManager(log, fileManager);
} }
} }
@ -40,13 +52,13 @@ namespace CodexDistTestCore
{ {
try try
{ {
TestLog.EndTest(k8sManager); log.EndTest(k8sManager);
k8sManager.DeleteAllResources(); k8sManager.DeleteAllResources();
fileManager.DeleteAllTestFiles(); fileManager.DeleteAllTestFiles();
} }
catch (Exception ex) catch (Exception ex)
{ {
TestLog.Error("Cleanup failed: " + ex.Message); log.Error("Cleanup failed: " + ex.Message);
GlobalTestFailure.HasFailed = true; GlobalTestFailure.HasFailed = true;
} }
} }
@ -56,9 +68,9 @@ namespace CodexDistTestCore
return fileManager.GenerateTestFile(size); return fileManager.GenerateTestFile(size);
} }
public IOfflineCodexNode SetupCodexNode() public IOfflineCodexNodes SetupCodexNodes(int numberOfNodes)
{ {
return new OfflineCodexNode(k8sManager); return new OfflineCodexNodes(k8sManager, numberOfNodes);
} }
} }

View File

@ -15,10 +15,12 @@ namespace CodexDistTestCore
private const string Folder = "TestDataFiles"; private const string Folder = "TestDataFiles";
private readonly Random random = new Random(); private readonly Random random = new Random();
private readonly List<TestFile> activeFiles = new List<TestFile>(); private readonly List<TestFile> activeFiles = new List<TestFile>();
private readonly TestLog log;
public FileManager() public FileManager(TestLog log)
{ {
if (!Directory.Exists(Folder)) Directory.CreateDirectory(Folder); if (!Directory.Exists(Folder)) Directory.CreateDirectory(Folder);
this.log = log;
} }
public TestFile CreateEmptyTestFile() public TestFile CreateEmptyTestFile()
@ -26,7 +28,6 @@ namespace CodexDistTestCore
var result = new TestFile(Path.Combine(Folder, Guid.NewGuid().ToString() + "_test.bin")); var result = new TestFile(Path.Combine(Folder, Guid.NewGuid().ToString() + "_test.bin"));
File.Create(result.Filename).Close(); File.Create(result.Filename).Close();
activeFiles.Add(result); activeFiles.Add(result);
TestLog.Log($"Created test file '{result.Filename}'.");
return result; return result;
} }
@ -34,7 +35,7 @@ namespace CodexDistTestCore
{ {
var result = CreateEmptyTestFile(); var result = CreateEmptyTestFile();
GenerateFileBytes(result, size); GenerateFileBytes(result, size);
TestLog.Log($"Generated {size.SizeInBytes} bytes of content for file '{result.Filename}'."); log.Log($"Generated {size.SizeInBytes} bytes of content for file '{result.Filename}'.");
return result; return result;
} }
@ -73,6 +74,12 @@ namespace CodexDistTestCore
public string Filename { get; } public string Filename { get; }
public long GetFileSize()
{
var info = new FileInfo(Filename);
return info.Length;
}
public void AssertIsEqual(TestFile? other) public void AssertIsEqual(TestFile? other)
{ {
if (other == null) Assert.Fail("TestFile is null."); if (other == null) Assert.Fail("TestFile is null.");

View File

@ -20,18 +20,22 @@ namespace CodexDistTestCore
if (!this.baseUrl.EndsWith("/")) this.baseUrl += "/"; if (!this.baseUrl.EndsWith("/")) this.baseUrl += "/";
} }
public T HttpGetJson<T>(string route) public string HttpGetString(string route)
{ {
return Retry(() => return Retry(() =>
{ {
using var client = GetClient(); using var client = GetClient();
var url = GetUrl() + route; var url = GetUrl() + route;
var result = Utils.Wait(client.GetAsync(url)); var result = Utils.Wait(client.GetAsync(url));
var json = Utils.Wait(result.Content.ReadAsStringAsync()); return Utils.Wait(result.Content.ReadAsStringAsync());
return JsonConvert.DeserializeObject<T>(json)!;
}); });
} }
public T HttpGetJson<T>(string route)
{
return JsonConvert.DeserializeObject<T>(HttpGetString(route))!;
}
public string HttpPostStream(string route, Stream stream) public string HttpPostStream(string route, Stream stream)
{ {
return Retry(() => return Retry(() =>
@ -75,6 +79,7 @@ namespace CodexDistTestCore
} }
catch (Exception exception) catch (Exception exception)
{ {
Timing.HttpCallRetryDelay();
retryCounter++; retryCounter++;
if (retryCounter > Timing.HttpCallRetryCount()) if (retryCounter > Timing.HttpCallRetryCount())
{ {

View File

@ -1,297 +1,86 @@
using k8s; namespace CodexDistTestCore
using k8s.Models;
using NUnit.Framework;
namespace CodexDistTestCore
{ {
public interface IK8sManager public interface IK8sManager
{ {
IOnlineCodexNode BringOnline(OfflineCodexNode node); ICodexNodeGroup BringOnline(OfflineCodexNodes node);
IOfflineCodexNode BringOffline(IOnlineCodexNode node); IOfflineCodexNodes BringOffline(ICodexNodeGroup node);
} }
public class K8sManager : IK8sManager public class K8sManager : IK8sManager
{ {
public const string K8sNamespace = "codex-test-namespace"; private readonly CodexGroupNumberSource codexGroupNumberSource = new CodexGroupNumberSource();
private readonly CodexDockerImage dockerImage = new CodexDockerImage(); private readonly List<CodexNodeGroup> onlineCodexNodes = new List<CodexNodeGroup>();
private readonly NumberSource numberSource = new NumberSource(); private readonly KnownK8sPods knownPods = new KnownK8sPods();
private readonly Dictionary<OnlineCodexNode, ActiveNode> activeNodes = new Dictionary<OnlineCodexNode, ActiveNode>(); private readonly TestLog log;
private readonly List<string> knownActivePodNames = new List<string>();
private readonly IFileManager fileManager; private readonly IFileManager fileManager;
public K8sManager(IFileManager fileManager) public K8sManager(TestLog log, IFileManager fileManager)
{ {
this.log = log;
this.fileManager = fileManager; this.fileManager = fileManager;
} }
public IOnlineCodexNode BringOnline(OfflineCodexNode node) public ICodexNodeGroup BringOnline(OfflineCodexNodes offline)
{ {
var client = CreateClient(); var online = CreateOnlineCodexNodes(offline);
EnsureTestNamespace(client); K8s(k => k.BringOnline(online, offline));
var activeNode = new ActiveNode(node, numberSource.GetFreePort(), numberSource.GetNodeOrderNumber()); log.Log($"{online.Describe()} online.");
var codexNode = new OnlineCodexNode(this, fileManager, activeNode.Port);
activeNodes.Add(codexNode, activeNode);
CreateDeployment(activeNode, client, node); return online;
CreateService(activeNode, client);
WaitUntilOnline(activeNode, client);
TestLog.Log($"{activeNode.Describe()} online.");
return codexNode;
} }
public IOfflineCodexNode BringOffline(IOnlineCodexNode node) public IOfflineCodexNodes BringOffline(ICodexNodeGroup node)
{ {
var client = CreateClient(); var online = GetAndRemoveActiveNodeFor(node);
var activeNode = GetAndRemoveActiveNodeFor(node); K8s(k => k.BringOffline(online));
var deploymentName = activeNode.Deployment.Name(); log.Log($"{online.Describe()} offline.");
BringOffline(activeNode, client);
WaitUntilOffline(deploymentName, client);
TestLog.Log($"{activeNode.Describe()} offline.");
return activeNode.Origin; return online.Origin;
} }
public void DeleteAllResources() public void DeleteAllResources()
{ {
var client = CreateClient(); K8s(k => k.DeleteAllResources());
DeleteNamespace(client);
WaitUntilZeroPods(client);
WaitUntilNamespaceDeleted(client);
} }
public void FetchAllPodsLogs(Action<string, string, Stream> onLog) public void FetchAllPodsLogs(IPodLogsHandler logHandler)
{ {
var client = CreateClient(); K8s(k => k.FetchAllPodsLogs(onlineCodexNodes.ToArray(), logHandler));
foreach (var node in activeNodes.Values)
{
var nodeDescription = node.Describe();
foreach (var podName in node.ActivePodNames)
{
var stream = client.ReadNamespacedPodLog(podName, K8sNamespace);
onLog(node.SelectorName, $"{nodeDescription}:{podName}", stream);
}
}
} }
private void BringOffline(ActiveNode activeNode, Kubernetes client) private CodexNodeGroup CreateOnlineCodexNodes(OfflineCodexNodes offline)
{ {
DeleteDeployment(activeNode, client); var containers = CreateContainers(offline.NumberOfNodes);
DeleteService(activeNode, client); var online = containers.Select(c => new OnlineCodexNode(log, fileManager, c)).ToArray();
var result = new CodexNodeGroup(codexGroupNumberSource.GetNextCodexNodeGroupNumber(), offline, this, online);
onlineCodexNodes.Add(result);
return result;
} }
#region Waiting private CodexNodeContainer[] CreateContainers(int number)
private void WaitUntilOnline(ActiveNode activeNode, Kubernetes client)
{ {
WaitUntil(() => var factory = new CodexNodeContainerFactory(codexGroupNumberSource);
{ var containers = new List<CodexNodeContainer>();
activeNode.Deployment = client.ReadNamespacedDeployment(activeNode.Deployment.Name(), K8sNamespace); for (var i = 0; i < number; i++) containers.Add(factory.CreateNext());
return activeNode.Deployment?.Status.AvailableReplicas != null && activeNode.Deployment.Status.AvailableReplicas > 0; return containers.ToArray();
});
AssignActivePodNames(activeNode, client);
} }
private void AssignActivePodNames(ActiveNode activeNode, Kubernetes client) private CodexNodeGroup GetAndRemoveActiveNodeFor(ICodexNodeGroup node)
{ {
var pods = client.ListNamespacedPod(K8sNamespace); var n = (CodexNodeGroup)node;
var podNames = pods.Items.Select(p => p.Name()); onlineCodexNodes.Remove(n);
foreach (var podName in podNames) return n;
{
if (!knownActivePodNames.Contains(podName))
{
knownActivePodNames.Add(podName);
activeNode.ActivePodNames.Add(podName);
}
}
} }
private void WaitUntilOffline(string deploymentName, Kubernetes client) private void K8s(Action<K8sOperations> action)
{ {
WaitUntil(() => var k8s = new K8sOperations(knownPods);
{ action(k8s);
var deployment = client.ReadNamespacedDeployment(deploymentName, K8sNamespace); k8s.Close();
return deployment == null || deployment.Status.AvailableReplicas == 0;
});
}
private void WaitUntilZeroPods(Kubernetes client)
{
WaitUntil(() => !client.ListNamespacedPod(K8sNamespace).Items.Any());
}
private void WaitUntilNamespaceDeleted(Kubernetes client)
{
WaitUntil(() => !IsTestNamespaceOnline(client));
}
private void WaitUntil(Func<bool> predicate)
{
var start = DateTime.UtcNow;
var state = predicate();
while (!state)
{
if (DateTime.UtcNow - start > Timing.K8sOperationTimeout())
{
Assert.Fail("K8s operation timed out.");
throw new TimeoutException();
}
Timing.WaitForK8sServiceDelay();
state = predicate();
}
}
#endregion
#region Service management
private void CreateService(ActiveNode node, Kubernetes client)
{
var serviceSpec = new V1Service
{
ApiVersion = "v1",
Metadata = node.GetServiceMetadata(),
Spec = new V1ServiceSpec
{
Type = "NodePort",
Selector = node.GetSelector(),
Ports = new List<V1ServicePort>
{
new V1ServicePort
{
Protocol = "TCP",
Port = 8080,
TargetPort = node.GetContainerPortName(),
NodePort = node.Port
}
}
}
};
node.Service = client.CreateNamespacedService(serviceSpec, K8sNamespace);
}
private void DeleteService(ActiveNode node, Kubernetes client)
{
if (node.Service == null) return;
client.DeleteNamespacedService(node.Service.Name(), K8sNamespace);
node.Service = null;
}
#endregion
#region Deployment management
private void CreateDeployment(ActiveNode node, Kubernetes client, OfflineCodexNode codexNode)
{
var deploymentSpec = new V1Deployment
{
ApiVersion = "apps/v1",
Metadata = node.GetDeploymentMetadata(),
Spec = new V1DeploymentSpec
{
Replicas = 1,
Selector = new V1LabelSelector
{
MatchLabels = node.GetSelector()
},
Template = new V1PodTemplateSpec
{
Metadata = new V1ObjectMeta
{
Labels = node.GetSelector()
},
Spec = new V1PodSpec
{
Containers = new List<V1Container>
{
new V1Container
{
Name = node.GetContainerName(),
Image = dockerImage.GetImageTag(),
Ports = new List<V1ContainerPort>
{
new V1ContainerPort
{
ContainerPort = 8080,
Name = node.GetContainerPortName()
}
},
Env = dockerImage.CreateEnvironmentVariables(codexNode)
}
}
}
}
}
};
node.Deployment = client.CreateNamespacedDeployment(deploymentSpec, K8sNamespace);
}
private void DeleteDeployment(ActiveNode node, Kubernetes client)
{
if (node.Deployment == null) return;
client.DeleteNamespacedDeployment(node.Deployment.Name(), K8sNamespace);
node.Deployment = null;
}
#endregion
#region Namespace management
private void EnsureTestNamespace(Kubernetes client)
{
if (IsTestNamespaceOnline(client)) return;
var namespaceSpec = new V1Namespace
{
ApiVersion = "v1",
Metadata = new V1ObjectMeta
{
Name = K8sNamespace,
Labels = new Dictionary<string, string> { { "name", K8sNamespace } }
}
};
client.CreateNamespace(namespaceSpec);
}
private void DeleteNamespace(Kubernetes client)
{
if (IsTestNamespaceOnline(client))
{
client.DeleteNamespace(K8sNamespace, null, null, gracePeriodSeconds: 0);
}
}
#endregion
private static Kubernetes CreateClient()
{
// todo: If the default KubeConfig file does not suffice, change it here:
var config = KubernetesClientConfiguration.BuildConfigFromConfigFile();
return new Kubernetes(config);
}
private static bool IsTestNamespaceOnline(Kubernetes client)
{
return client.ListNamespace().Items.Any(n => n.Metadata.Name == K8sNamespace);
}
private ActiveNode GetAndRemoveActiveNodeFor(IOnlineCodexNode node)
{
var n = (OnlineCodexNode)node;
var activeNode = activeNodes[n];
activeNodes.Remove(n);
return activeNode;
} }
} }
} }

View File

@ -0,0 +1,286 @@
using k8s;
using k8s.Models;
using NUnit.Framework;
namespace CodexDistTestCore
{
public class K8sOperations
{
public const string K8sNamespace = "codex-test-namespace";
private readonly CodexDockerImage dockerImage = new CodexDockerImage();
private readonly Kubernetes client;
private readonly KnownK8sPods knownPods;
public K8sOperations(KnownK8sPods knownPods)
{
this.knownPods = knownPods;
// todo: If the default KubeConfig file does not suffice, change it here:
var config = KubernetesClientConfiguration.BuildConfigFromConfigFile();
client = new Kubernetes(config);
}
public void Close()
{
client.Dispose();
}
public void BringOnline(CodexNodeGroup online, OfflineCodexNodes offline)
{
EnsureTestNamespace();
CreateDeployment(online, offline);
CreateService(online);
WaitUntilOnline(online);
FetchPodInfo(online);
}
public void BringOffline(CodexNodeGroup online)
{
var deploymentName = online.Deployment.Name();
DeleteDeployment(online);
DeleteService(online);
WaitUntilOffline(deploymentName);
}
public void DeleteAllResources()
{
DeleteNamespace();
WaitUntilZeroPods();
WaitUntilNamespaceDeleted();
}
public void FetchAllPodsLogs(CodexNodeGroup[] onlines, IPodLogsHandler logHandler)
{
var logNumberSource = new NumberSource(0);
foreach (var online in onlines)
{
foreach (var node in online)
{
WritePodLogs(online, node, logHandler, logNumberSource);
}
}
}
private void WritePodLogs(CodexNodeGroup online, IOnlineCodexNode node, IPodLogsHandler logHandler, NumberSource logNumberSource)
{
var n = (OnlineCodexNode)node;
var nodeDescription = $"{online.Describe()} contains {n.GetName()}";
var stream = client.ReadNamespacedPodLog(online.PodInfo!.Name, K8sNamespace, n.Container.Name);
logHandler.Log(logNumberSource.GetNextNumber(), nodeDescription, stream);
}
private void FetchPodInfo(CodexNodeGroup online)
{
var pods = client.ListNamespacedPod(K8sNamespace).Items;
var newPods = pods.Where(p => !knownPods.Contains(p.Name())).ToArray();
Assert.That(newPods.Length, Is.EqualTo(1), "Expected only 1 pod to be created. Test infra failure.");
var newPod = newPods.Single();
online.PodInfo = new PodInfo(newPod.Name(), newPod.Status.PodIP);
Assert.That(!string.IsNullOrEmpty(online.PodInfo.Name), "Invalid pod name received. Test infra failure.");
Assert.That(!string.IsNullOrEmpty(online.PodInfo.Ip), "Invalid pod IP received. Test infra failure.");
knownPods.Add(newPod.Name());
}
#region Waiting
private void WaitUntilOnline(CodexNodeGroup online)
{
WaitUntil(() =>
{
online.Deployment = client.ReadNamespacedDeployment(online.Deployment.Name(), K8sNamespace);
return online.Deployment?.Status.AvailableReplicas != null && online.Deployment.Status.AvailableReplicas > 0;
});
}
private void WaitUntilOffline(string deploymentName)
{
WaitUntil(() =>
{
var deployment = client.ReadNamespacedDeployment(deploymentName, K8sNamespace);
return deployment == null || deployment.Status.AvailableReplicas == 0;
});
}
private void WaitUntilZeroPods()
{
WaitUntil(() => !client.ListNamespacedPod(K8sNamespace).Items.Any());
}
private void WaitUntilNamespaceDeleted()
{
WaitUntil(() => !IsTestNamespaceOnline());
}
private void WaitUntil(Func<bool> predicate)
{
var start = DateTime.UtcNow;
var state = predicate();
while (!state)
{
if (DateTime.UtcNow - start > Timing.K8sOperationTimeout())
{
Assert.Fail("K8s operation timed out.");
throw new TimeoutException();
}
Timing.WaitForK8sServiceDelay();
state = predicate();
}
}
#endregion
#region Service management
private void CreateService(CodexNodeGroup online)
{
var serviceSpec = new V1Service
{
ApiVersion = "v1",
Metadata = online.GetServiceMetadata(),
Spec = new V1ServiceSpec
{
Type = "NodePort",
Selector = online.GetSelector(),
Ports = CreateServicePorts(online)
}
};
online.Service = client.CreateNamespacedService(serviceSpec, K8sNamespace);
}
private List<V1ServicePort> CreateServicePorts(CodexNodeGroup online)
{
var result = new List<V1ServicePort>();
var containers = online.GetContainers();
foreach (var container in containers)
{
result.Add(new V1ServicePort
{
Name = container.ServicePortName,
Protocol = "TCP",
Port = container.ApiPort,
TargetPort = container.ContainerPortName,
NodePort = container.ServicePort
});
}
return result;
}
private void DeleteService(CodexNodeGroup online)
{
if (online.Service == null) return;
client.DeleteNamespacedService(online.Service.Name(), K8sNamespace);
online.Service = null;
}
#endregion
#region Deployment management
private void CreateDeployment(CodexNodeGroup online, OfflineCodexNodes offline)
{
var deploymentSpec = new V1Deployment
{
ApiVersion = "apps/v1",
Metadata = online.GetDeploymentMetadata(),
Spec = new V1DeploymentSpec
{
Replicas = 1,
Selector = new V1LabelSelector
{
MatchLabels = online.GetSelector()
},
Template = new V1PodTemplateSpec
{
Metadata = new V1ObjectMeta
{
Labels = online.GetSelector()
},
Spec = new V1PodSpec
{
Containers = CreateDeploymentContainers(online, offline)
}
}
}
};
online.Deployment = client.CreateNamespacedDeployment(deploymentSpec, K8sNamespace);
}
private List<V1Container> CreateDeploymentContainers(CodexNodeGroup online, OfflineCodexNodes offline)
{
var result = new List<V1Container>();
var containers = online.GetContainers();
foreach (var container in containers)
{
result.Add(new V1Container
{
Name = container.Name,
Image = dockerImage.GetImageTag(),
Ports = new List<V1ContainerPort>
{
new V1ContainerPort
{
ContainerPort = container.ApiPort,
Name = container.ContainerPortName
}
},
Env = dockerImage.CreateEnvironmentVariables(offline, container)
});
}
return result;
}
private void DeleteDeployment(CodexNodeGroup online)
{
if (online.Deployment == null) return;
client.DeleteNamespacedDeployment(online.Deployment.Name(), K8sNamespace);
online.Deployment = null;
}
#endregion
#region Namespace management
private void EnsureTestNamespace()
{
if (IsTestNamespaceOnline()) return;
var namespaceSpec = new V1Namespace
{
ApiVersion = "v1",
Metadata = new V1ObjectMeta
{
Name = K8sNamespace,
Labels = new Dictionary<string, string> { { "name", K8sNamespace } }
}
};
client.CreateNamespace(namespaceSpec);
}
private void DeleteNamespace()
{
if (IsTestNamespaceOnline())
{
client.DeleteNamespace(K8sNamespace, null, null, gracePeriodSeconds: 0);
}
}
#endregion
private bool IsTestNamespaceOnline()
{
return client.ListNamespace().Items.Any(n => n.Metadata.Name == K8sNamespace);
}
}
}

View File

@ -0,0 +1,17 @@
namespace CodexDistTestCore
{
public class KnownK8sPods
{
private readonly List<string> knownActivePodNames = new List<string>();
public bool Contains(string name)
{
return knownActivePodNames.Contains(name);
}
public void Add(string name)
{
knownActivePodNames.Add(name);
}
}
}

View File

@ -2,27 +2,18 @@
{ {
public class NumberSource public class NumberSource
{ {
private int freePort; private int number;
private int nodeOrderNumber;
public NumberSource() public NumberSource(int start)
{ {
freePort = 30001; number = start;
nodeOrderNumber = 0;
} }
public int GetFreePort() public int GetNextNumber()
{ {
var port = freePort; var n = number;
freePort++; number++;
return port; return n;
}
public int GetNodeOrderNumber()
{
var number = nodeOrderNumber;
nodeOrderNumber++;
return number;
} }
} }
} }

View File

@ -1,65 +0,0 @@
namespace CodexDistTestCore
{
public interface IOfflineCodexNode
{
IOfflineCodexNode WithLogLevel(CodexLogLevel level);
IOfflineCodexNode WithBootstrapNode(IOnlineCodexNode node);
IOfflineCodexNode WithStorageQuota(ByteSize storageQuota);
IOnlineCodexNode BringOnline();
}
public enum CodexLogLevel
{
Trace,
Debug,
Info,
Warn,
Error
}
public class OfflineCodexNode : IOfflineCodexNode
{
private readonly IK8sManager k8SManager;
public CodexLogLevel? LogLevel { get; private set; }
public IOnlineCodexNode? BootstrapNode { get; private set; }
public ByteSize? StorageQuota { get; private set; }
public OfflineCodexNode(IK8sManager k8SManager)
{
this.k8SManager = k8SManager;
}
public IOnlineCodexNode BringOnline()
{
return k8SManager.BringOnline(this);
}
public IOfflineCodexNode WithBootstrapNode(IOnlineCodexNode node)
{
BootstrapNode = node;
return this;
}
public IOfflineCodexNode WithLogLevel(CodexLogLevel level)
{
LogLevel = level;
return this;
}
public IOfflineCodexNode WithStorageQuota(ByteSize storageQuota)
{
StorageQuota = storageQuota;
return this;
}
public string Describe()
{
var result = "";
if (LogLevel != null) result += $"LogLevel={LogLevel},";
if (BootstrapNode != null) result += "BootstrapNode=set,";
if (StorageQuota != null) result += $"StorageQuote={StorageQuota.SizeInBytes},";
return result;
}
}
}

View File

@ -0,0 +1,71 @@
namespace CodexDistTestCore
{
public interface IOfflineCodexNodes
{
IOfflineCodexNodes WithLogLevel(CodexLogLevel level);
IOfflineCodexNodes WithBootstrapNode(IOnlineCodexNode node);
IOfflineCodexNodes WithStorageQuota(ByteSize storageQuota);
ICodexNodeGroup BringOnline();
}
public enum CodexLogLevel
{
Trace,
Debug,
Info,
Warn,
Error
}
public class OfflineCodexNodes : IOfflineCodexNodes
{
private readonly IK8sManager k8SManager;
public int NumberOfNodes { get; }
public CodexLogLevel? LogLevel { get; private set; }
public IOnlineCodexNode? BootstrapNode { get; private set; }
public ByteSize? StorageQuota { get; private set; }
public OfflineCodexNodes(IK8sManager k8SManager, int numberOfNodes)
{
this.k8SManager = k8SManager;
NumberOfNodes = numberOfNodes;
}
public ICodexNodeGroup BringOnline()
{
return k8SManager.BringOnline(this);
}
public IOfflineCodexNodes WithBootstrapNode(IOnlineCodexNode node)
{
BootstrapNode = node;
return this;
}
public IOfflineCodexNodes WithLogLevel(CodexLogLevel level)
{
LogLevel = level;
return this;
}
public IOfflineCodexNodes WithStorageQuota(ByteSize storageQuota)
{
StorageQuota = storageQuota;
return this;
}
public string Describe()
{
var args = string.Join(',', DescribeArgs());
return $"{NumberOfNodes} CodexNodes with [{args}]";
}
private IEnumerable<string> DescribeArgs()
{
if (LogLevel != null) yield return ($"LogLevel={LogLevel}");
if (BootstrapNode != null) yield return ("BootstrapNode=set");
if (StorageQuota != null) yield return ($"StorageQuote={StorageQuota.SizeInBytes}");
}
}
}

View File

@ -7,55 +7,106 @@ namespace CodexDistTestCore
CodexDebugResponse GetDebugInfo(); CodexDebugResponse GetDebugInfo();
ContentId UploadFile(TestFile file); ContentId UploadFile(TestFile file);
TestFile? DownloadContent(ContentId contentId); TestFile? DownloadContent(ContentId contentId);
IOfflineCodexNode BringOffline(); void ConnectToPeer(IOnlineCodexNode node);
} }
public class OnlineCodexNode : IOnlineCodexNode public class OnlineCodexNode : IOnlineCodexNode
{ {
private readonly IK8sManager k8SManager; private const string SuccessfullyConnectedMessage = "Successfully connected to peer";
private readonly IFileManager fileManager; private const string UploadFailedMessage = "Unable to store block";
private readonly int port;
public OnlineCodexNode(IK8sManager k8SManager, IFileManager fileManager, int port) private readonly TestLog log;
private readonly IFileManager fileManager;
public OnlineCodexNode(TestLog log, IFileManager fileManager, CodexNodeContainer container)
{ {
this.k8SManager = k8SManager; this.log = log;
this.fileManager = fileManager; this.fileManager = fileManager;
this.port = port; Container = container;
} }
public IOfflineCodexNode BringOffline() public CodexNodeContainer Container { get; }
public CodexNodeGroup Group { get; internal set; } = null!;
public string GetName()
{ {
return k8SManager.BringOffline(this); return $"<{Container.Name}>";
} }
public CodexDebugResponse GetDebugInfo() public CodexDebugResponse GetDebugInfo()
{ {
return Http().HttpGetJson<CodexDebugResponse>("debug/info"); var response = Http().HttpGetJson<CodexDebugResponse>("debug/info");
Log($"Got DebugInfo with id: '{response.id}'.");
return response;
} }
public ContentId UploadFile(TestFile file) public ContentId UploadFile(TestFile file)
{ {
Log($"Uploading file of size {file.GetFileSize()}...");
using var fileStream = File.OpenRead(file.Filename); using var fileStream = File.OpenRead(file.Filename);
var response = Http().HttpPostStream("upload", fileStream); var response = Http().HttpPostStream("upload", fileStream);
if (response.StartsWith("Unable to store block")) if (response.StartsWith(UploadFailedMessage))
{ {
Assert.Fail("Node failed to store block."); Assert.Fail("Node failed to store block.");
} }
Log($"Uploaded file. Received contentId: '{response}'.");
return new ContentId(response); return new ContentId(response);
} }
public TestFile? DownloadContent(ContentId contentId) public TestFile? DownloadContent(ContentId contentId)
{ {
Log($"Downloading for contentId: '{contentId.Id}'...");
var file = fileManager.CreateEmptyTestFile(); var file = fileManager.CreateEmptyTestFile();
using var fileStream = File.OpenWrite(file.Filename); DownloadToFile(contentId.Id, file);
using var downloadStream = Http().HttpGetStream("download/" + contentId.Id); Log($"Downloaded file of size {file.GetFileSize()} to '{file.Filename}'.");
downloadStream.CopyTo(fileStream);
return file; return file;
} }
public void ConnectToPeer(IOnlineCodexNode node)
{
var peer = (OnlineCodexNode)node;
Log($"Connecting to peer {peer.GetName()}...");
var peerInfo = node.GetDebugInfo();
var peerId = peerInfo.id;
var peerMultiAddress = GetPeerMultiAddress(peer, peerInfo);
var response = Http().HttpGetString($"connect/{peerId}?addrs={peerMultiAddress}");
Assert.That(response, Is.EqualTo(SuccessfullyConnectedMessage), "Unable to connect codex nodes.");
Log($"Successfully connected to peer {peer.GetName()}.");
}
private string GetPeerMultiAddress(OnlineCodexNode peer, CodexDebugResponse peerInfo)
{
var multiAddress = peerInfo.addrs.First();
// Todo: Is there a case where First address in list is not the way?
if (Group == peer.Group)
{
return multiAddress;
}
// The peer we want to connect is in a different pod.
// We must replace the default IP with the pod IP in the multiAddress.
return multiAddress.Replace("0.0.0.0", peer.Group.PodInfo!.Ip);
}
private void DownloadToFile(string contentId, TestFile file)
{
using var fileStream = File.OpenWrite(file.Filename);
using var downloadStream = Http().HttpGetStream("download/" + contentId);
downloadStream.CopyTo(fileStream);
}
private Http Http() private Http Http()
{ {
return new Http(ip: "127.0.0.1", port: port, baseUrl: "/api/codex/v1"); return new Http(ip: "127.0.0.1", port: Container.ServicePort, baseUrl: "/api/codex/v1");
}
private void Log(string msg)
{
log.Log($"{GetName()}: {msg}");
} }
} }

View File

@ -0,0 +1,7 @@
namespace CodexDistTestCore
{
public interface IPodLogsHandler
{
void Log(int id, string podDescription, Stream log);
}
}

View File

@ -5,75 +5,94 @@ namespace CodexDistTestCore
public class TestLog public class TestLog
{ {
public const string LogRoot = "D:/CodexTestLogs"; public const string LogRoot = "D:/CodexTestLogs";
private readonly LogFile file;
private static LogFile? file = null; public TestLog()
public static void Log(string message)
{ {
file!.Write(message);
}
public static void Error(string message)
{
Log($"[ERROR] {message}");
}
public static void BeginTest()
{
if (file != null) throw new InvalidOperationException("Test is already started!");
var name = GetTestName(); var name = GetTestName();
file = new LogFile(name); file = new LogFile(name);
Log($"Begin: {name}"); Log($"Begin: {name}");
} }
public static void EndTest(K8sManager k8sManager) public void Log(string message)
{ {
if (file == null) throw new InvalidOperationException("No test is started!"); file.Write(message);
}
public void Error(string message)
{
Log($"[ERROR] {message}");
}
public void EndTest(K8sManager k8sManager)
{
var result = TestContext.CurrentContext.Result; var result = TestContext.CurrentContext.Result;
Log($"Finished: {GetTestName()} = {result.Outcome.Status}"); Log($"Finished: {GetTestName()} = {result.Outcome.Status}");
if (result.Outcome.Status == NUnit.Framework.Interfaces.TestStatus.Failed)
if (!string.IsNullOrEmpty(result.Message))
{ {
IncludeFullPodLogging(k8sManager); Log(result.Message);
} }
file = null; if (result.Outcome.Status == NUnit.Framework.Interfaces.TestStatus.Failed)
{
Log($"{result.StackTrace}");
var logWriter = new PodLogWriter(file);
logWriter.IncludeFullPodLogging(k8sManager);
}
} }
private static string GetTestName() private static string GetTestName()
{ {
var test = TestContext.CurrentContext.Test; var test = TestContext.CurrentContext.Test;
var className = test.ClassName!.Substring(test.ClassName.LastIndexOf('.') + 1); var className = test.ClassName!.Substring(test.ClassName.LastIndexOf('.') + 1);
return $"{className}.{test.MethodName}"; var args = FormatArguments(test);
return $"{className}.{test.MethodName}{args}";
} }
private static void LogRaw(string message, string filename) private static string FormatArguments(TestContext.TestAdapter test)
{ {
file!.WriteRaw(message, filename); if (test.Arguments == null || !test.Arguments.Any()) return "";
return $"[{string.Join(',', test.Arguments)}]";
}
} }
private static void IncludeFullPodLogging(K8sManager k8sManager) public class PodLogWriter : IPodLogsHandler
{ {
Log("Full pod logging:"); private readonly LogFile file;
k8sManager.FetchAllPodsLogs(WritePodLog);
public PodLogWriter(LogFile file)
{
this.file = file;
} }
private static void WritePodLog(string id, string nodeDescription, Stream stream) public void IncludeFullPodLogging(K8sManager k8sManager)
{ {
Log($"{nodeDescription} -->> {id}"); file.Write("Full pod logging:");
LogRaw(nodeDescription, id); k8sManager.FetchAllPodsLogs(this);
var reader = new StreamReader(stream); }
public void Log(int id, string podDescription, Stream log)
{
var logFile = id.ToString().PadLeft(6, '0');
file.Write($"{podDescription} -->> {logFile}");
LogRaw(podDescription, logFile);
var reader = new StreamReader(log);
var line = reader.ReadLine(); var line = reader.ReadLine();
while (line != null) while (line != null)
{ {
LogRaw(line, id); LogRaw(line, logFile);
line = reader.ReadLine(); line = reader.ReadLine();
} }
} }
private void LogRaw(string message, string filename)
{
file!.WriteRaw(message, filename);
}
} }
public class LogFile public class LogFile

View File

@ -25,9 +25,9 @@ namespace CodexDistTestCore
return GetTimes().HttpCallRetryCount(); return GetTimes().HttpCallRetryCount();
} }
public static void RetryDelay() public static void HttpCallRetryDelay()
{ {
Utils.Sleep(GetTimes().RetryDelay()); Utils.Sleep(GetTimes().HttpCallRetryDelay());
} }
public static void WaitForK8sServiceDelay() public static void WaitForK8sServiceDelay()
@ -52,7 +52,7 @@ namespace CodexDistTestCore
{ {
TimeSpan HttpCallTimeout(); TimeSpan HttpCallTimeout();
int HttpCallRetryCount(); int HttpCallRetryCount();
TimeSpan RetryDelay(); TimeSpan HttpCallRetryDelay();
TimeSpan WaitForK8sServiceDelay(); TimeSpan WaitForK8sServiceDelay();
TimeSpan K8sOperationTimeout(); TimeSpan K8sOperationTimeout();
} }
@ -69,7 +69,7 @@ namespace CodexDistTestCore
return 5; return 5;
} }
public TimeSpan RetryDelay() public TimeSpan HttpCallRetryDelay()
{ {
return TimeSpan.FromSeconds(3); return TimeSpan.FromSeconds(3);
} }
@ -97,7 +97,7 @@ namespace CodexDistTestCore
return 2; return 2;
} }
public TimeSpan RetryDelay() public TimeSpan HttpCallRetryDelay()
{ {
return TimeSpan.FromMinutes(5); return TimeSpan.FromMinutes(5);
} }

View File

@ -4,17 +4,17 @@ using NUnit.Framework;
namespace LongTests.BasicTests namespace LongTests.BasicTests
{ {
[TestFixture] [TestFixture]
public class SimpleTests : DistTest public class LargeFileTests : DistTest
{ {
[Test, UseLongTimeouts] [Test, UseLongTimeouts]
public void OneClientLargeFileTest() public void OneClientLargeFileTest()
{ {
var primary = SetupCodexNode() var primary = SetupCodexNodes(1)
.WithLogLevel(CodexLogLevel.Warn) .WithLogLevel(CodexLogLevel.Warn)
.WithStorageQuota(10.GB()) .WithStorageQuota(20.GB())
.BringOnline(); .BringOnline()[0];
var testFile = GenerateTestFile(1.GB()); var testFile = GenerateTestFile(10.GB());
var contentId = primary.UploadFile(testFile); var contentId = primary.UploadFile(testFile);

View File

@ -0,0 +1,53 @@
using CodexDistTestCore;
using NUnit.Framework;
namespace LongTests.BasicTests
{
public class TestInfraTests : DistTest
{
[Test, UseLongTimeouts]
public void TestInfraShouldHave1000AddressSpacesPerPod()
{
var group = SetupCodexNodes(1000).BringOnline();
var nodeIds = group.Select(n => n.GetDebugInfo().id).ToArray();
Assert.That(nodeIds.Length, Is.EqualTo(nodeIds.Distinct().Count()),
"Not all created nodes provided a unique id.");
}
[Test, UseLongTimeouts]
public void TestInfraSupportsManyConcurrentPods()
{
for (var i = 0; i < 20; i++)
{
var n = SetupCodexNodes(1).BringOnline()[0];
Assert.That(!string.IsNullOrEmpty(n.GetDebugInfo().id));
}
}
[Test, UseLongTimeouts]
public void DownloadConsistencyTest()
{
var primary = SetupCodexNodes(1)
.WithLogLevel(CodexLogLevel.Trace)
.WithStorageQuota(2.MB())
.BringOnline()[0];
var testFile = GenerateTestFile(1.MB());
var contentId = primary.UploadFile(testFile);
var files = new List<TestFile?>();
for (var i = 0; i < 100; i++)
{
files.Add(primary.DownloadContent(contentId));
}
Assert.That(files.All(f => f != null));
Assert.That(files.All(f => f!.GetFileSize() == testFile.GetFileSize()));
foreach (var file in files) file!.AssertIsEqual(testFile);
}
}
}

View File

@ -11,7 +11,7 @@ namespace Tests.BasicTests
{ {
var dockerImage = new CodexDockerImage(); var dockerImage = new CodexDockerImage();
var node = SetupCodexNode().BringOnline(); var node = SetupCodexNodes(1).BringOnline()[0];
var debugInfo = node.GetDebugInfo(); var debugInfo = node.GetDebugInfo();
@ -22,10 +22,10 @@ namespace Tests.BasicTests
[Test] [Test]
public void OneClientTest() public void OneClientTest()
{ {
var primary = SetupCodexNode() var primary = SetupCodexNodes(1)
.WithLogLevel(CodexLogLevel.Trace) .WithLogLevel(CodexLogLevel.Trace)
.WithStorageQuota(2.MB()) .WithStorageQuota(2.MB())
.BringOnline(); .BringOnline()[0];
var testFile = GenerateTestFile(1.MB()); var testFile = GenerateTestFile(1.MB());
@ -36,26 +36,45 @@ namespace Tests.BasicTests
testFile.AssertIsEqual(downloadedFile); testFile.AssertIsEqual(downloadedFile);
} }
//[Test] [Test]
//public void TwoClientTest() public void TwoClientOnePodTest()
//{ {
// var primary = SetupCodexNode() var group = SetupCodexNodes(2)
// .WithLogLevel(CodexLogLevel.Trace) .WithLogLevel(CodexLogLevel.Trace)
// .WithStorageQuota(1024 * 1024 * 2) .WithStorageQuota(2.MB())
// .BringOnline(); .BringOnline();
// var secondary = SetupCodexNode() var primary = group[0];
// .WithLogLevel(CodexLogLevel.Trace) var secondary = group[1];
// .WithBootstrapNode(primary)
// .BringOnline();
// var testFile = GenerateTestFile(1024 * 1024); PerformTwoClientTest(primary, secondary);
}
// var contentId = primary.UploadFile(testFile); [Test]
public void TwoClientTwoPodTest()
{
var primary = SetupCodexNodes(1)
.WithStorageQuota(2.MB())
.BringOnline()[0];
// var downloadedFile = secondary.DownloadContent(contentId); var secondary = SetupCodexNodes(1)
.WithStorageQuota(2.MB())
.BringOnline()[0];
// testFile.AssertIsEqual(downloadedFile); PerformTwoClientTest(primary, secondary);
//} }
private void PerformTwoClientTest(IOnlineCodexNode primary, IOnlineCodexNode secondary)
{
primary.ConnectToPeer(secondary);
var testFile = GenerateTestFile(1.MB());
var contentId = primary.UploadFile(testFile);
var downloadedFile = secondary.DownloadContent(contentId);
testFile.AssertIsEqual(downloadedFile);
}
} }
} }

View File

@ -1,28 +1,3 @@
# apiVersion: v1
# kind: Pod
# metadata:
# name: codex-pod
# spec:
# containers:
# - name: codex-node-container
# image: thatbenbierens/nim-codex:sha-c9a62de
# ports:
# - containerPort: 8080
# kind: PersistentVolumeClaim
# apiVersion: v1
# metadata:
# name: codex-volume
# spec:
# accessModes:
# - ReadWriteOnce
# resources:
# requests:
# storage: 1Gi
# storageClassName: rbd
# ---
apiVersion: v1 apiVersion: v1
kind: Namespace kind: Namespace
metadata: metadata:
@ -48,21 +23,35 @@ spec:
codex-node: dist-test codex-node: dist-test
spec: spec:
containers: containers:
- name: codex-node - name: codex-node1
image: thatbenbierens/nim-codex:sha-c9a62de image: thatbenbierens/nim-codex:sha-b204837
ports: ports:
- containerPort: 8080 - containerPort: 8080
name: codex-api-port name: api-1
env: env:
- name: LOG_LEVEL - name: API_PORT
value: WARN value: "8080"
# volumeMounts: - name: DATA_DIR
# - mountPath: /datadir value: datadir1
# name: codex-data - name: DISC_PORT
# volumes: value: "8081"
# - name: codex-data - name: LISTEN_ADDRS
# persistentVolumeClaim: value: "/ip4/0.0.0.0/tcp/8082"
# claimName: codex-volume - name: codex-node2
image: thatbenbierens/nim-codex:sha-b204837
ports:
- containerPort: 8083
name: api-2
env:
- name: API_PORT
value: "8083"
- name: DATA_DIR
value: datadir2
- name: DISC_PORT
value: "8084"
- name: LISTEN_ADDRS
value: "/ip4/0.0.0.0/tcp/8085"
--- ---
apiVersion: v1 apiVersion: v1
@ -75,7 +64,14 @@ spec:
selector: selector:
codex-node: dist-test codex-node: dist-test
ports: ports:
- protocol: TCP - name: "node1"
protocol: TCP
port: 8080 port: 8080
targetPort: codex-api-port targetPort: api-1
nodePort: 30001 nodePort: 30001
- name: "node2"
protocol: TCP
port: 8083
targetPort: api-2
nodePort: 30002