mirror of
https://github.com/actions/runner.git
synced 2025-12-11 12:57:05 +00:00
120 lines
4.4 KiB
C#
120 lines
4.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using GitHub.DistributedTask.ObjectTemplating.Tokens;
|
|
using GitHub.DistributedTask.Pipelines.ContextData;
|
|
using GitHub.DistributedTask.WebApi;
|
|
using GitHub.Runner.Common;
|
|
using GitHub.Runner.Sdk;
|
|
using Pipelines = GitHub.DistributedTask.Pipelines;
|
|
|
|
|
|
namespace GitHub.Runner.Worker.Handlers
|
|
{
|
|
[ServiceLocator(Default = typeof(CompositeActionHandler))]
|
|
public interface ICompositeActionHandler : IHandler
|
|
{
|
|
CompositeActionExecutionData Data { get; set; }
|
|
}
|
|
public sealed class CompositeActionHandler : Handler, ICompositeActionHandler
|
|
{
|
|
public CompositeActionExecutionData Data { get; set; }
|
|
|
|
public Task RunAsync(ActionRunStage stage)
|
|
{
|
|
// Validate args.
|
|
Trace.Entering();
|
|
ArgUtil.NotNull(ExecutionContext, nameof(ExecutionContext));
|
|
ArgUtil.NotNull(Inputs, nameof(Inputs));
|
|
|
|
var githubContext = ExecutionContext.ExpressionValues["github"] as GitHubContext;
|
|
ArgUtil.NotNull(githubContext, nameof(githubContext));
|
|
|
|
var tempDirectory = HostContext.GetDirectory(WellKnownDirectory.Temp);
|
|
|
|
// Resolve action steps
|
|
var actionSteps = Data.Steps;
|
|
|
|
// Create Context Data to reuse for each composite action step
|
|
var inputsData = new DictionaryContextData();
|
|
foreach (var i in Inputs)
|
|
{
|
|
inputsData[i.Key] = new StringContextData(i.Value);
|
|
}
|
|
|
|
|
|
// Get Environment Data for Composite Action
|
|
var extraExpressionValues = new Dictionary<string, PipelineContextData>(StringComparer.OrdinalIgnoreCase);
|
|
extraExpressionValues["inputs"] = inputsData;
|
|
var manifestManager = HostContext.GetService<IActionManifestManager>();
|
|
|
|
// Add the composite action environment variables to each step.
|
|
// If the key already exists, we override it since the composite action env variables will have higher precedence
|
|
// Note that for each composite action step, it's environment variables will be set in the StepRunner automatically
|
|
var compositeEnvData = manifestManager.EvaluateCompositeActionEnvironment(ExecutionContext, Data.Environment, extraExpressionValues);
|
|
var envData = new Dictionary<string, string>();
|
|
|
|
// Copy over parent environment
|
|
foreach (var env in Environment)
|
|
{
|
|
envData[env.Key] = env.Value;
|
|
}
|
|
// Overwrite with current env
|
|
foreach (var env in compositeEnvData)
|
|
{
|
|
envData[env.Key] = env.Value;
|
|
}
|
|
|
|
// Add each composite action step to the front of the queue
|
|
int location = 0;
|
|
foreach (Pipelines.ActionStep aStep in actionSteps)
|
|
{
|
|
// Ex:
|
|
// runs:
|
|
// using: "composite"
|
|
// steps:
|
|
// - uses: example/test-composite@v2 (a)
|
|
// - run echo hello world (b)
|
|
// - run echo hello world 2 (c)
|
|
//
|
|
// ethanchewy/test-composite/action.yaml
|
|
// runs:
|
|
// using: "composite"
|
|
// steps:
|
|
// - run echo hello world 3 (d)
|
|
// - run echo hello world 4 (e)
|
|
//
|
|
// Steps processed as follow:
|
|
// | a |
|
|
// | a | => | d |
|
|
// (Run step d)
|
|
// | a |
|
|
// | a | => | e |
|
|
// (Run step e)
|
|
// | a |
|
|
// (Run step a)
|
|
// | b |
|
|
// (Run step b)
|
|
// | c |
|
|
// (Run step c)
|
|
// Done.
|
|
|
|
var actionRunner = HostContext.CreateService<IActionRunner>();
|
|
actionRunner.Action = aStep;
|
|
actionRunner.Stage = stage;
|
|
actionRunner.Condition = aStep.Condition;
|
|
actionRunner.DisplayName = aStep.DisplayName;
|
|
|
|
ExecutionContext.RegisterNestedStep(actionRunner, inputsData, location, envData);
|
|
location++;
|
|
}
|
|
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
}
|
|
}
|