mirror of
https://github.com/actions/runner.git
synced 2025-12-11 21:06:55 +00:00
Added ability to run Dockerfile.SUFFIX ContainerAction (#1738)
* Added ability to run Dockerfile.SUFFIX ContainerAction * Extracted IsDockerFile method * reformatted, moved from index to Last() * extracted IsDockerfile to DockerUtil with L0 * added check for IsDockerfile to account for docker:// * updated test to clearly show path/dockerfile:tag * fail if Data.Image is not Dockerfile or docker://[image]
This commit is contained in:
@@ -1004,7 +1004,7 @@ namespace GitHub.Runner.Worker
|
|||||||
if (actionDefinitionData.Execution.ExecutionType == ActionExecutionType.Container)
|
if (actionDefinitionData.Execution.ExecutionType == ActionExecutionType.Container)
|
||||||
{
|
{
|
||||||
var containerAction = actionDefinitionData.Execution as ContainerActionExecutionData;
|
var containerAction = actionDefinitionData.Execution as ContainerActionExecutionData;
|
||||||
if (containerAction.Image.EndsWith("Dockerfile") || containerAction.Image.EndsWith("dockerfile"))
|
if (DockerUtil.IsDockerfile(containerAction.Image))
|
||||||
{
|
{
|
||||||
var dockerFileFullPath = Path.Combine(actionEntryDirectory, containerAction.Image);
|
var dockerFileFullPath = Path.Combine(actionEntryDirectory, containerAction.Image);
|
||||||
executionContext.Debug($"Dockerfile for action: '{dockerFileFullPath}'.");
|
executionContext.Debug($"Dockerfile for action: '{dockerFileFullPath}'.");
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
|
|
||||||
namespace GitHub.Runner.Worker.Container
|
namespace GitHub.Runner.Worker.Container
|
||||||
@@ -17,7 +18,7 @@ namespace GitHub.Runner.Worker.Container
|
|||||||
string pattern = $"^(?<{targetPort}>\\d+)/(?<{proto}>\\w+) -> (?<{host}>.+):(?<{hostPort}>\\d+)$";
|
string pattern = $"^(?<{targetPort}>\\d+)/(?<{proto}>\\w+) -> (?<{host}>.+):(?<{hostPort}>\\d+)$";
|
||||||
|
|
||||||
List<PortMapping> portMappings = new List<PortMapping>();
|
List<PortMapping> portMappings = new List<PortMapping>();
|
||||||
foreach(var line in portMappingLines)
|
foreach (var line in portMappingLines)
|
||||||
{
|
{
|
||||||
Match m = Regex.Match(line, pattern, RegexOptions.None, TimeSpan.FromSeconds(1));
|
Match m = Regex.Match(line, pattern, RegexOptions.None, TimeSpan.FromSeconds(1));
|
||||||
if (m.Success)
|
if (m.Success)
|
||||||
@@ -61,5 +62,15 @@ namespace GitHub.Runner.Worker.Container
|
|||||||
}
|
}
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static bool IsDockerfile(string image)
|
||||||
|
{
|
||||||
|
if (image.StartsWith("docker://", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
var imageWithoutPath = image.Split('/').Last();
|
||||||
|
return imageWithoutPath.StartsWith("Dockerfile.") || imageWithoutPath.StartsWith("dockerfile.") || imageWithoutPath.EndsWith("Dockerfile") || imageWithoutPath.EndsWith("dockerfile");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -44,9 +44,8 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
{
|
{
|
||||||
Data.Image = Data.Image.Substring("docker://".Length);
|
Data.Image = Data.Image.Substring("docker://".Length);
|
||||||
}
|
}
|
||||||
else if (Data.Image.EndsWith("Dockerfile") || Data.Image.EndsWith("dockerfile"))
|
else if (DockerUtil.IsDockerfile(Data.Image))
|
||||||
{
|
{
|
||||||
// ensure docker file exist
|
|
||||||
var dockerFile = Path.Combine(ActionDirectory, Data.Image);
|
var dockerFile = Path.Combine(ActionDirectory, Data.Image);
|
||||||
ArgUtil.File(dockerFile, nameof(Data.Image));
|
ArgUtil.File(dockerFile, nameof(Data.Image));
|
||||||
|
|
||||||
@@ -68,6 +67,10 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
|
|
||||||
Data.Image = imageName;
|
Data.Image = imageName;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException($"'{Data.Image}' should be either '[path]/Dockerfile' or 'docker://image[:tag]'.");
|
||||||
|
}
|
||||||
|
|
||||||
string type = Action.Type == Pipelines.ActionSourceType.Repository ? "Dockerfile" : "DockerHub";
|
string type = Action.Type == Pipelines.ActionSourceType.Repository ? "Dockerfile" : "DockerHub";
|
||||||
// Set extra telemetry base on the current context.
|
// Set extra telemetry base on the current context.
|
||||||
|
|||||||
@@ -144,5 +144,48 @@ namespace GitHub.Runner.Common.Tests.Worker.Container
|
|||||||
var actual = DockerUtil.ParseRegistryHostnameFromImageName(input);
|
var actual = DockerUtil.ParseRegistryHostnameFromImageName(input);
|
||||||
Assert.Equal(expected, actual);
|
Assert.Equal(expected, actual);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[Trait("Level", "L0")]
|
||||||
|
[Trait("Category", "Worker")]
|
||||||
|
[InlineData("dockerhub/repo", false)]
|
||||||
|
[InlineData("debian:latest", false)]
|
||||||
|
[InlineData("something/dockerfileimage", false)]
|
||||||
|
[InlineData("ghcr.io/docker/dockerfile", true)] // should be false but might break the current workflows
|
||||||
|
[InlineData("Dockerfile", true)]
|
||||||
|
[InlineData("Dockerfile.", true)]
|
||||||
|
[InlineData(".Dockerfile", true)]
|
||||||
|
[InlineData(".Dockerfile.", false)]
|
||||||
|
[InlineData("dockerfile", true)]
|
||||||
|
[InlineData("dockerfile.", true)]
|
||||||
|
[InlineData(".dockerfile", true)]
|
||||||
|
[InlineData(".dockerfile.", false)]
|
||||||
|
[InlineData("Dockerfile.test", true)]
|
||||||
|
[InlineData("test.Dockerfile", true)]
|
||||||
|
[InlineData("docker/dockerfile:latest", false)]
|
||||||
|
[InlineData("/some/path/dockerfile:latest", false)]
|
||||||
|
[InlineData("dockerfile:latest", false)]
|
||||||
|
[InlineData("Dockerfile:latest", false)]
|
||||||
|
[InlineData("dockerfile-latest", false)]
|
||||||
|
[InlineData("Dockerfile-latest", false)]
|
||||||
|
[InlineData("dockerfile.latest", true)]
|
||||||
|
[InlineData("Dockerfile.latest", true)]
|
||||||
|
[InlineData("../dockerfile/dockerfileone", false)]
|
||||||
|
[InlineData("../Dockerfile/dockerfileone", false)]
|
||||||
|
[InlineData("../dockerfile.test", true)]
|
||||||
|
[InlineData("../Dockerfile.test", true)]
|
||||||
|
[InlineData("./dockerfile/image", false)]
|
||||||
|
[InlineData("./Dockerfile/image", false)]
|
||||||
|
[InlineData("example/Dockerfile.test", true)]
|
||||||
|
[InlineData("./example/Dockerfile.test", true)]
|
||||||
|
[InlineData("example/test.dockerfile", true)]
|
||||||
|
[InlineData("./example/test.dockerfile", true)]
|
||||||
|
[InlineData("docker://Dockerfile", false)]
|
||||||
|
[InlineData("docker://ubuntu:latest", false)]
|
||||||
|
public void IsDockerfile(string input, bool expected)
|
||||||
|
{
|
||||||
|
var actual = DockerUtil.IsDockerfile(input);
|
||||||
|
Assert.Equal(expected, actual);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user