2023-04-25 11:31:15 +02:00
|
|
|
|
using Logging;
|
2023-10-23 15:49:14 +02:00
|
|
|
|
using Newtonsoft.Json;
|
2023-06-21 08:28:40 +02:00
|
|
|
|
using Utils;
|
2023-04-17 10:31:14 +02:00
|
|
|
|
|
|
|
|
|
namespace KubernetesWorkflow
|
2023-04-12 13:53:55 +02:00
|
|
|
|
{
|
2023-09-11 16:57:57 +02:00
|
|
|
|
public interface IStartupWorkflow
|
|
|
|
|
{
|
2023-09-25 08:47:19 +02:00
|
|
|
|
IKnownLocations GetAvailableLocations();
|
|
|
|
|
RunningContainers Start(int numberOfContainers, ContainerRecipeFactory recipeFactory, StartupConfig startupConfig);
|
|
|
|
|
RunningContainers Start(int numberOfContainers, ILocation location, ContainerRecipeFactory recipeFactory, StartupConfig startupConfig);
|
2023-09-20 12:55:09 +02:00
|
|
|
|
CrashWatcher CreateCrashWatcher(RunningContainer container);
|
2023-09-11 16:57:57 +02:00
|
|
|
|
void Stop(RunningContainers runningContainers);
|
2023-09-12 13:32:06 +02:00
|
|
|
|
void DownloadContainerLog(RunningContainer container, ILogHandler logHandler, int? tailLines = null);
|
2023-09-11 16:57:57 +02:00
|
|
|
|
string ExecuteCommand(RunningContainer container, string command, params string[] args);
|
2023-09-12 10:31:55 +02:00
|
|
|
|
void DeleteNamespace();
|
|
|
|
|
void DeleteNamespacesStartingWith(string namespacePrefix);
|
2023-09-11 16:57:57 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class StartupWorkflow : IStartupWorkflow
|
2023-04-12 13:53:55 +02:00
|
|
|
|
{
|
2023-09-12 10:31:55 +02:00
|
|
|
|
private readonly ILog log;
|
2023-04-12 15:11:36 +02:00
|
|
|
|
private readonly WorkflowNumberSource numberSource;
|
|
|
|
|
private readonly K8sCluster cluster;
|
|
|
|
|
private readonly KnownK8sPods knownK8SPods;
|
2023-09-12 10:31:55 +02:00
|
|
|
|
private readonly string k8sNamespace;
|
2023-04-12 13:53:55 +02:00
|
|
|
|
private readonly RecipeComponentFactory componentFactory = new RecipeComponentFactory();
|
2023-09-25 08:47:19 +02:00
|
|
|
|
private readonly LocationProvider locationProvider;
|
2023-04-12 13:53:55 +02:00
|
|
|
|
|
2023-09-12 10:31:55 +02:00
|
|
|
|
internal StartupWorkflow(ILog log, WorkflowNumberSource numberSource, K8sCluster cluster, KnownK8sPods knownK8SPods, string k8sNamespace)
|
2023-04-12 13:53:55 +02:00
|
|
|
|
{
|
2023-04-25 11:31:15 +02:00
|
|
|
|
this.log = log;
|
2023-04-12 15:11:36 +02:00
|
|
|
|
this.numberSource = numberSource;
|
|
|
|
|
this.cluster = cluster;
|
|
|
|
|
this.knownK8SPods = knownK8SPods;
|
2023-09-12 10:31:55 +02:00
|
|
|
|
this.k8sNamespace = k8sNamespace;
|
2023-09-25 08:47:19 +02:00
|
|
|
|
|
|
|
|
|
locationProvider = new LocationProvider(log, K8s);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IKnownLocations GetAvailableLocations()
|
|
|
|
|
{
|
|
|
|
|
return locationProvider.GetAvailableLocations();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public RunningContainers Start(int numberOfContainers, ContainerRecipeFactory recipeFactory, StartupConfig startupConfig)
|
|
|
|
|
{
|
|
|
|
|
return Start(numberOfContainers, KnownLocations.UnspecifiedLocation, recipeFactory, startupConfig);
|
2023-04-12 13:53:55 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-25 08:47:19 +02:00
|
|
|
|
public RunningContainers Start(int numberOfContainers, ILocation location, ContainerRecipeFactory recipeFactory, StartupConfig startupConfig)
|
2023-04-12 13:53:55 +02:00
|
|
|
|
{
|
2023-04-12 15:11:36 +02:00
|
|
|
|
return K8s(controller =>
|
|
|
|
|
{
|
|
|
|
|
var recipes = CreateRecipes(numberOfContainers, recipeFactory, startupConfig);
|
|
|
|
|
var runningPod = controller.BringOnline(recipes, location);
|
2023-09-11 16:57:57 +02:00
|
|
|
|
var containers = CreateContainers(runningPod, recipes, startupConfig);
|
2023-04-12 13:53:55 +02:00
|
|
|
|
|
2023-09-13 15:10:19 +02:00
|
|
|
|
var rc = new RunningContainers(startupConfig, runningPod, containers);
|
|
|
|
|
cluster.Configuration.Hooks.OnContainersStarted(rc);
|
|
|
|
|
return rc;
|
2023-09-11 16:57:57 +02:00
|
|
|
|
});
|
2023-08-14 15:10:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-20 12:55:09 +02:00
|
|
|
|
public CrashWatcher CreateCrashWatcher(RunningContainer container)
|
|
|
|
|
{
|
|
|
|
|
return K8s(c => c.CreateCrashWatcher(container));
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-13 11:07:36 +02:00
|
|
|
|
public void Stop(RunningContainers runningContainers)
|
|
|
|
|
{
|
|
|
|
|
K8s(controller =>
|
|
|
|
|
{
|
|
|
|
|
controller.Stop(runningContainers.RunningPod);
|
2023-09-13 15:10:19 +02:00
|
|
|
|
cluster.Configuration.Hooks.OnContainersStopped(runningContainers);
|
2023-04-13 11:07:36 +02:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-12 13:32:06 +02:00
|
|
|
|
public void DownloadContainerLog(RunningContainer container, ILogHandler logHandler, int? tailLines = null)
|
2023-04-13 11:30:19 +02:00
|
|
|
|
{
|
|
|
|
|
K8s(controller =>
|
|
|
|
|
{
|
2023-08-16 16:13:29 +02:00
|
|
|
|
controller.DownloadPodLog(container.Pod, container.Recipe, logHandler, tailLines);
|
2023-04-13 11:30:19 +02:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-14 09:54:07 +02:00
|
|
|
|
public string ExecuteCommand(RunningContainer container, string command, params string[] args)
|
|
|
|
|
{
|
|
|
|
|
return K8s(controller =>
|
|
|
|
|
{
|
|
|
|
|
return controller.ExecuteCommand(container.Pod, container.Recipe.Name, command, args);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-12 10:31:55 +02:00
|
|
|
|
public void DeleteNamespace()
|
2023-04-12 15:11:36 +02:00
|
|
|
|
{
|
|
|
|
|
K8s(controller =>
|
|
|
|
|
{
|
2023-09-12 10:31:55 +02:00
|
|
|
|
controller.DeleteNamespace();
|
2023-04-12 15:11:36 +02:00
|
|
|
|
});
|
2023-04-12 13:53:55 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-12 10:31:55 +02:00
|
|
|
|
public void DeleteNamespacesStartingWith(string namespacePrefix)
|
2023-05-03 14:18:37 +02:00
|
|
|
|
{
|
|
|
|
|
K8s(controller =>
|
|
|
|
|
{
|
2023-09-12 10:31:55 +02:00
|
|
|
|
controller.DeleteAllNamespacesStartingWith(namespacePrefix);
|
2023-05-03 14:18:37 +02:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-30 10:56:19 +02:00
|
|
|
|
private RunningContainer[] CreateContainers(RunningPod runningPod, ContainerRecipe[] recipes, StartupConfig startupConfig)
|
2023-04-12 13:53:55 +02:00
|
|
|
|
{
|
2023-04-25 11:31:15 +02:00
|
|
|
|
log.Debug();
|
2023-05-04 11:34:43 +02:00
|
|
|
|
return recipes.Select(r =>
|
|
|
|
|
{
|
|
|
|
|
var servicePorts = runningPod.GetServicePortsForContainerRecipe(r);
|
|
|
|
|
log.Debug($"{r} -> service ports: {string.Join(",", servicePorts.Select(p => p.Number))}");
|
|
|
|
|
|
2023-06-23 10:35:23 +02:00
|
|
|
|
var name = GetContainerName(r, startupConfig);
|
|
|
|
|
|
|
|
|
|
return new RunningContainer(runningPod, r, servicePorts, name,
|
2023-10-19 11:08:30 +02:00
|
|
|
|
CreateContainerPorts(runningPod, r, servicePorts));
|
2023-06-01 09:35:18 +02:00
|
|
|
|
|
2023-05-04 11:34:43 +02:00
|
|
|
|
}).ToArray();
|
2023-04-12 13:53:55 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-06-23 10:35:23 +02:00
|
|
|
|
private string GetContainerName(ContainerRecipe recipe, StartupConfig startupConfig)
|
|
|
|
|
{
|
|
|
|
|
if (startupConfig == null) return "";
|
|
|
|
|
if (!string.IsNullOrEmpty(startupConfig.NameOverride))
|
|
|
|
|
{
|
|
|
|
|
return $"<{startupConfig.NameOverride}{recipe.Number}>";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return $"<{recipe.Name}>";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-19 11:08:30 +02:00
|
|
|
|
private ContainerPort[] CreateContainerPorts(RunningPod pod, ContainerRecipe recipe, Port[] servicePorts)
|
2023-06-01 09:35:18 +02:00
|
|
|
|
{
|
2023-10-19 11:08:30 +02:00
|
|
|
|
var result = new List<ContainerPort>();
|
|
|
|
|
foreach (var exposedPort in recipe.ExposedPorts)
|
|
|
|
|
{
|
|
|
|
|
result.Add(new ContainerPort(
|
|
|
|
|
exposedPort,
|
|
|
|
|
GetContainerExternalAddress(pod, servicePorts, exposedPort),
|
2023-10-23 16:12:45 +02:00
|
|
|
|
GetContainerInternalAddress(pod, exposedPort)));
|
2023-10-19 11:08:30 +02:00
|
|
|
|
}
|
2023-10-23 14:05:08 +02:00
|
|
|
|
foreach (var internalPort in recipe.InternalPorts)
|
|
|
|
|
{
|
|
|
|
|
if (!string.IsNullOrEmpty(internalPort.Tag))
|
|
|
|
|
{
|
|
|
|
|
result.Add(new ContainerPort(
|
|
|
|
|
internalPort,
|
2023-10-23 15:28:20 +02:00
|
|
|
|
new Address(string.Empty, 0),
|
2023-10-23 16:12:45 +02:00
|
|
|
|
GetContainerInternalAddress(pod, internalPort)));
|
2023-10-23 14:05:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-10-19 11:08:30 +02:00
|
|
|
|
|
|
|
|
|
return result.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Address GetContainerExternalAddress(RunningPod pod, Port[] servicePorts, Port exposedPort)
|
|
|
|
|
{
|
|
|
|
|
var servicePort = servicePorts.Single(p => p.Tag == exposedPort.Tag);
|
|
|
|
|
|
2023-06-21 08:28:40 +02:00
|
|
|
|
return new Address(
|
2023-06-01 09:35:18 +02:00
|
|
|
|
pod.Cluster.HostAddress,
|
2023-10-19 11:08:30 +02:00
|
|
|
|
servicePort.Number);
|
2023-06-01 09:35:18 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-23 16:12:45 +02:00
|
|
|
|
private Address GetContainerInternalAddress(RunningPod pod, Port port)
|
2023-06-01 09:35:18 +02:00
|
|
|
|
{
|
2023-06-21 08:28:40 +02:00
|
|
|
|
return new Address(
|
2023-10-23 16:12:45 +02:00
|
|
|
|
$"http://{pod.PodInfo.Ip}",
|
2023-10-23 14:05:08 +02:00
|
|
|
|
port.Number);
|
2023-06-01 09:35:18 +02:00
|
|
|
|
}
|
2023-10-19 11:08:30 +02:00
|
|
|
|
|
2023-04-12 13:53:55 +02:00
|
|
|
|
private ContainerRecipe[] CreateRecipes(int numberOfContainers, ContainerRecipeFactory recipeFactory, StartupConfig startupConfig)
|
|
|
|
|
{
|
2023-04-25 11:31:15 +02:00
|
|
|
|
log.Debug();
|
2023-04-12 13:53:55 +02:00
|
|
|
|
var result = new List<ContainerRecipe>();
|
|
|
|
|
for (var i = 0; i < numberOfContainers; i++)
|
|
|
|
|
{
|
2023-09-13 16:06:05 +02:00
|
|
|
|
var recipe = recipeFactory.CreateRecipe(i, numberSource.GetContainerNumber(), componentFactory, startupConfig);
|
|
|
|
|
if (cluster.Configuration.AddAppPodLabel) recipe.PodLabels.Add("app", recipeFactory.AppName);
|
|
|
|
|
cluster.Configuration.Hooks.OnContainerRecipeCreated(recipe);
|
|
|
|
|
result.Add(recipe);
|
2023-04-12 13:53:55 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result.ToArray();
|
|
|
|
|
}
|
2023-04-12 15:11:36 +02:00
|
|
|
|
|
|
|
|
|
private void K8s(Action<K8sController> action)
|
|
|
|
|
{
|
2023-10-23 15:49:14 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var controller = new K8sController(log, cluster, knownK8SPods, numberSource, k8sNamespace);
|
|
|
|
|
action(controller);
|
|
|
|
|
controller.Dispose();
|
|
|
|
|
}
|
|
|
|
|
catch (k8s.Autorest.HttpOperationException ex)
|
|
|
|
|
{
|
|
|
|
|
log.Error(JsonConvert.SerializeObject(ex));
|
|
|
|
|
throw;
|
|
|
|
|
}
|
2023-04-12 15:11:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private T K8s<T>(Func<K8sController, T> action)
|
2023-08-07 15:51:44 +02:00
|
|
|
|
{
|
2023-10-23 15:49:14 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var controller = new K8sController(log, cluster, knownK8SPods, numberSource, k8sNamespace);
|
|
|
|
|
var result = action(controller);
|
|
|
|
|
controller.Dispose();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
catch (k8s.Autorest.HttpOperationException ex)
|
|
|
|
|
{
|
|
|
|
|
log.Error(JsonConvert.SerializeObject(ex));
|
|
|
|
|
throw;
|
|
|
|
|
}
|
2023-08-10 11:47:34 +02:00
|
|
|
|
}
|
2023-04-13 11:30:19 +02:00
|
|
|
|
}
|
2023-04-12 15:11:36 +02:00
|
|
|
|
|
2023-04-13 11:30:19 +02:00
|
|
|
|
public interface ILogHandler
|
|
|
|
|
{
|
|
|
|
|
void Log(Stream log);
|
2023-04-12 13:53:55 +02:00
|
|
|
|
}
|
2023-04-17 10:31:14 +02:00
|
|
|
|
|
|
|
|
|
public abstract class LogHandler : ILogHandler
|
|
|
|
|
{
|
|
|
|
|
public void Log(Stream log)
|
|
|
|
|
{
|
|
|
|
|
using var reader = new StreamReader(log);
|
|
|
|
|
var line = reader.ReadLine();
|
|
|
|
|
while (line != null)
|
|
|
|
|
{
|
|
|
|
|
ProcessLine(line);
|
|
|
|
|
line = reader.ReadLine();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected abstract void ProcessLine(string line);
|
|
|
|
|
}
|
2023-04-12 13:53:55 +02:00
|
|
|
|
}
|