web-dev-qa-db-ja.com

コマンドライン引数を使用してC#からPowerShellスクリプトを実行する

C#内からPowerShellスクリプトを実行する必要があります。スクリプトにはコマンドライン引数が必要です。

これは私がこれまでにやったことです:

RunspaceConfiguration runspaceConfiguration = RunspaceConfiguration.Create();

Runspace runspace = RunspaceFactory.CreateRunspace(runspaceConfiguration);
runspace.Open();

RunspaceInvoke scriptInvoker = new RunspaceInvoke(runspace);

Pipeline pipeline = runspace.CreatePipeline();
pipeline.Commands.Add(scriptFile);

// Execute PowerShell script
results = pipeline.Invoke();

scriptFileには、「C:\ Program Files\MyProgram\Whatever.ps1」などが含まれます。

スクリプトは「-key Value」などのコマンドライン引数を使用しますが、Valueはスペースを含むパスのようなものにすることができます。

これが機能しません。 C#内からPowerShellスクリプトにコマンドライン引数を渡し、スペースが問題にならないようにする方法を知っている人はいますか?

90
Mephisztoe

別のコマンドとしてスクリプトファイルを作成してみてください:

Command myCommand = new Command(scriptfile);

次に、パラメータを追加できます

CommandParameter testParam = new CommandParameter("key","value");
myCommand.Parameters.Add(testParam);

そして最後に

pipeline.Commands.Add(myCommand);

編集済みの完全なコードは次のとおりです:

RunspaceConfiguration runspaceConfiguration = RunspaceConfiguration.Create();

Runspace runspace = RunspaceFactory.CreateRunspace(runspaceConfiguration);
runspace.Open();

RunspaceInvoke scriptInvoker = new RunspaceInvoke(runspace);

Pipeline pipeline = runspace.CreatePipeline();

//Here's how you add a new script with arguments
Command myCommand = new Command(scriptfile);
CommandParameter testParam = new CommandParameter("key","value");
myCommand.Parameters.Add(testParam);

pipeline.Commands.Add(myCommand);

// Execute PowerShell script
results = pipeline.Invoke();
104
Kosi2801

別の解決策があります。おそらく誰かがポリシーを変更する可能性があるため、PowerShellスクリプトの実行が成功するかどうかをテストしたいだけです。引数として、実行するスクリプトのパスを指定するだけです。

ProcessStartInfo startInfo = new ProcessStartInfo();
startInfo.FileName = @"powershell.exe";
startInfo.Arguments = @"& 'c:\Scripts\test.ps1'";
startInfo.RedirectStandardOutput = true;
startInfo.RedirectStandardError = true;
startInfo.UseShellExecute = false;
startInfo.CreateNoWindow = true;
Process process = new Process();
process.StartInfo = startInfo;
process.Start();

string output = process.StandardOutput.ReadToEnd();
Assert.IsTrue(output.Contains("StringToBeVerifiedInAUnitTest"));

string errors = process.StandardError.ReadToEnd();
Assert.IsTrue(string.IsNullOrEmpty(errors));

スクリプトの内容は次のとおりです。

$someVariable = "StringToBeVerifiedInAUnitTest"
$someVariable
29
Jowen

Commands.AddScriptメソッドへのパラメータの受け渡しについて、より明確になる可能性はありますか?

C:\ Foo1.PS1 Hello World Hunger C:\ Foo2.PS1 Hello World

scriptFile = "C:\ Foo1.PS1"

parameters = "parm1 parm2 parm3" ... paramsの可変長

これを解決しました...名前としてnullを渡し、値としてparamをCommandParametersのコレクションに渡します

私の機能は次のとおりです。

private static void RunPowershellScript(string scriptFile, string scriptParameters)
{
    RunspaceConfiguration runspaceConfiguration = RunspaceConfiguration.Create();
    Runspace runspace = RunspaceFactory.CreateRunspace(runspaceConfiguration);
    runspace.Open();
    RunspaceInvoke scriptInvoker = new RunspaceInvoke(runspace);
    Pipeline pipeline = runspace.CreatePipeline();
    Command scriptCommand = new Command(scriptFile);
    Collection<CommandParameter> commandParameters = new Collection<CommandParameter>();
    foreach (string scriptParameter in scriptParameters.Split(' '))
    {
        CommandParameter commandParm = new CommandParameter(null, scriptParameter);
        commandParameters.Add(commandParm);
        scriptCommand.Parameters.Add(commandParm);
    }
    pipeline.Commands.Add(scriptCommand);
    Collection<PSObject> psObjects;
    psObjects = pipeline.Invoke();
}
9
twalker

AddScriptメソッドでパイプラインを使用することもできます。

string cmdArg = ".\script.ps1 -foo bar"            
Collection<PSObject> psresults;
using (Pipeline pipeline = _runspace.CreatePipeline())
            {
                pipeline.Commands.AddScript(cmdArg);
                pipeline.Commands[0].MergeMyResults(PipelineResultTypes.Error, PipelineResultTypes.Output);
                psresults = pipeline.Invoke();
            }
return psresults;

文字列と、渡したパラメータを受け取ります。

5
James Pogran

以下は、使用した場合にスクリプトにパラメータを追加する方法です

pipeline.Commands.AddScript(Script);

これは、HashMapをパラメーターとして使用することで、キーはスクリプト内の変数の名前であり、値は変数の値です。

pipeline.Commands.AddScript(script));
FillVariables(pipeline, scriptParameter);
Collection<PSObject> results = pipeline.Invoke();

また、変数を埋める方法は次のとおりです。

private static void FillVariables(Pipeline pipeline, Hashtable scriptParameters)
{
  // Add additional variables to PowerShell
  if (scriptParameters != null)
  {
    foreach (DictionaryEntry entry in scriptParameters)
    {
      CommandParameter Param = new CommandParameter(entry.Key as String, entry.Value);
      pipeline.Commands[0].Parameters.Add(Param);
    }
  }
}

これにより、複数のパラメーターをスクリプトに簡単に追加できます。また、スクリプト内の変数から値を取得したい場合は、次のように気づきました。

Object resultcollection = runspace.SessionStateProxy.GetVariable("results");

//結果はvの名前

何らかの理由でKosi2801がスクリプト変数リストに独自の変数が入力されないように提案するので、何らかの理由でそれを行う必要があります。

3
Ruud

私にとって、C#からPowerShellスクリプトを実行する最も柔軟な方法は、PowerShell.Create()。AddScript()を使用することでした

コードのスニペットは

string scriptDirectory = Path.GetDirectoryName(
    ConfigurationManager.AppSettings["PathToTechOpsTooling"]);

var script =    
    "Set-Location " + scriptDirectory + Environment.NewLine +
    "Import-Module .\\script.psd1" + Environment.NewLine +
    "$data = Import-Csv -Path " + tempCsvFile + " -Encoding UTF8" + 
        Environment.NewLine +
    "New-Registration -server " + dbServer + " -DBName " + dbName + 
       " -Username \"" + user.Username + "\" + -Users $userData";

_powershell = PowerShell.Create().AddScript(script);
_powershell.Invoke<User>();
foreach (var errorRecord in _powershell.Streams.Error)
    Console.WriteLine(errorRecord);

Streams.Errorをチェックすると、エラーがあるかどうかを確認できます。コレクションを確認するのは本当に便利でした。 Userは、PowerShellスクリプトが返すオブジェクトのタイプです。

2
Andy

私のほうがもう少し小さくシンプルです:

/// <summary>
/// Runs a PowerShell script taking it's path and parameters.
/// </summary>
/// <param name="scriptFullPath">The full file path for the .ps1 file.</param>
/// <param name="parameters">The parameters for the script, can be null.</param>
/// <returns>The output from the PowerShell execution.</returns>
public static ICollection<PSObject> RunScript(string scriptFullPath, ICollection<CommandParameter> parameters = null)
{
    var runspace = RunspaceFactory.CreateRunspace();
    runspace.Open();
    var pipeline = runspace.CreatePipeline();
    var cmd = new Command(scriptFullPath);
    if (parameters != null)
    {
        foreach (var p in parameters)
        {
            cmd.Parameters.Add(p);
        }
    }
    pipeline.Commands.Add(cmd);
    var results = pipeline.Invoke();
    pipeline.Dispose();
    runspace.Dispose();
    return results;
}
2
Pedro Luz