C# - PowerShell과 연동하는 방법
지난 글에서 "Private Working Set" 크기를 구하는 Get-Counter 명령을 알아봤는데요,
(Get-Counter "\Process(*)\Working Set - Private").CounterSamples
그러고 보니 한 번도 C#에서 PowerShell 명령어를 수행하는 방법을 소개한 적이 없어 ^^ 이참에 글을 써봅니다.
이를 위해 우선해야 할 것은 "System.Management.Automation" 패키지를 참조 추가하면 됩니다.
System.Management.Automation
; https://www.nuget.org/packages/System.Management.Automation/
만약 .NET Framework 프로젝트라면 다음의 경로에 있는 것을 직접 참조 추가해야 합니다.
// 버전 4.0.3.0: C:\Windows\Microsoft.NET\assembly\GAC_MSIL\System.Management.Automation\
// 버전 1.0.0.0: C:\Windows\assembly\GAC_MSIL\System.Management.Automation
이후, 명령어 실행은 이렇게 간단하게 끝낼 수 있습니다.
using System;
using System.Collections;
using System.Management.Automation;
namespace ConsoleApp1
{
// System.Management.Automation
// .NET Framework
// 버전 4.0.3.0: C:\Windows\Microsoft.NET\assembly\GAC_MSIL\System.Management.Automation\
// 버전 1.0.0.0: C:\Windows\assembly\GAC_MSIL\System.Management.Automation
// .NET Core/5+
// Install-Package System.Management.Automation
internal class Program
{
static void Main(string[] args)
{
PowerShell _ps = PowerShell.Create();
_ps.AddCommand("Get-Counter", false);
_ps.AddArgument(@"\Process(*)\Working Set - Private");
System.Collections.ObjectModel.Collection<PSObject> output = _ps.Invoke();
foreach (var item in output)
{
object objValue = item.Properties["CounterSamples"].Value;
// objValue 타입: Microsoft.PowerShell.Commands.GetCounter.PerformanceCounterSample
}
}
}
}
PowerShell.Invoke 명령을 수행하면, 그 순간 해당 명령어(위의 경우 Get-Counter)와 연관된 모듈이 자동으로 로드됩니다. 하지만, 코드상으로는 object로 반환받은 objValue를 형변환해서 사용해야 할 텐데요, 따라서 이제는 저 형식을 담은 어셈블리를 참조 추가해야 합니다.
Microsoft.PowerShell.Commands.GetCounter.PerformanceCounterSample
; https://learn.microsoft.com/en-us/dotnet/api/microsoft.powershell.commands.getcounter.performancecountersample
문서에 따라 "Microsoft.PowerShell.Commands.Diagnostics.dll" 파일인데요,
[닷넷 프레임워크]
버전 1.0: C:\Windows\assembly\GAC_MSIL\Microsoft.PowerShell.Commands.Diagnostics
버전 4.0: C:\Windows\Microsoft.NET\assembly\GAC_MSIL\Microsoft.PowerShell.Commands.Diagnostics
[닷넷 코어/5+]
Install-Package Microsoft.PowerShell.SDK
간단하게 사용하는 경우에는 이것도 귀찮으니 dynamic을 이용해 처리하는 것도 좋은 선택입니다.
foreach (var item in output)
{
object objValue = item.Properties["CounterSamples"].Value;
var e = (objValue as IEnumerable).GetEnumerator();
while (e.MoveNext())
{
dynamic pcs = e.Current;
Console.WriteLine($"{pcs.Path}\t{pcs.InstanceName}\t{pcs.RawValue}");
}
}
/* 출력 결과:
\\testpc\process(idle)\working set - private idle 8192
\\testpc\process(system)\working set - private system 20480
\\testpc\process(secure system)\working set - private secure system 152301568
\\testpc\process(registry)\working set - private registry 21250048
\\testpc\process(smss)\working set - private smss 315392
\\testpc\process(csrss#1)\working set - private csrss 1794048
\\testpc\process(wininit)\working set - private wininit 1310720
\\testpc\process(csrss)\working set - private csrss 1527808
\\testpc\process(services)\working set - private services 7786496
\\testpc\process(lsaiso)\working set - private lsaiso 950272
\\testpc\process(lsass)\working set - private lsass 18694144
...[생략]...
*/
어렵지 않죠? ^^
(
첨부 파일은 이 글의 예제 코드를 포함합니다.)
참고로, .NET Core/5+의 경우 nuget 패키지를 참조했다면 ps.Invoke(); 수행 시 이런 오류가 발생할 것입니다.
PowerShell ps = PowerShell.Create();
ps.AddCommand("Get-Counter", false);
var output = ps.Invoke();
/*
Unhandled exception. System.Management.Automation.Runspaces.PSSnapInException: Cannot load PowerShell snap-in Microsoft.PowerShell.Diagnostics because of the following error: Could not find file 'C:\testapp\ConsoleApp1\ConsoleApp2\bin\Debug\net9.0\runtimes\win\lib\net9.0\Microsoft.PowerShell.Commands.Diagnostics.dll'.
at System.Management.Automation.Runspaces.PSSnapInHelpers.LoadPSSnapInAssembly(PSSnapInInfo psSnapInInfo)
at System.Management.Automation.Runspaces.InitialSessionState.ImportPSSnapIn(PSSnapInInfo psSnapInInfo, PSSnapInException& warning)
at System.Management.Automation.Runspaces.InitialSessionState.CreateDefault()
at System.Management.Automation.Runspaces.RunspaceFactory.CreateRunspace(PSHost host)
at System.Management.Automation.Runspaces.RunspaceFactory.CreateRunspace()
at System.Management.Automation.PowerShell.Worker.CreateRunspaceIfNeededAndDoWork(Runspace rsToUse, Boolean isSync)
at System.Management.Automation.PowerShell.CoreInvokeHelper[TInput,TOutput](PSDataCollection`1 input, PSDataCollection`1 output, PSInvocationSettings settings)
at System.Management.Automation.PowerShell.CoreInvoke[TInput,TOutput](PSDataCollection`1 input, PSDataCollection`1 output, PSInvocationSettings settings)
at System.Management.Automation.PowerShell.CoreInvoke[TOutput](IEnumerable input, PSDataCollection`1 output, PSInvocationSettings settings)
at System.Management.Automation.PowerShell.Invoke(IEnumerable input, PSInvocationSettings settings)
at System.Management.Automation.PowerShell.Invoke()
at ConsoleApp2.Program.Main(String[] args)
*/
오류 메시지에도 나오지만, Get-Counter를 구현한 Microsoft.PowerShell.Commands.Diagnostics.dll 파일을 현재 디렉터리에서 찾다가 없어서 오류가 발생하고 있습니다.
그러니까, .NET Framework의 경우에는 형식 안정성을 위해 대상 어셈블리(Get-Counter의 경우 Microsoft.PowerShell.Commands.Diagnostics.dll)를 참조하는 것이 선택 사항에 불과하지만, .NET Core/5+의 경우에는 반드시 nuget을 통해 참조 추가를 해야만 합니다. 하지만 개별 어셈블리를 참조 추가하는 경우,
Install-Package Microsoft.PowerShell.Commands.Diagnostics
이후 계속해서 의존성에 따라 System.Diagnostics.PerformanceCounter, Microsoft.PowerShell.ConsoleHost, Microsoft.PowerShell.Commands.Utility, Microsoft.PowerShell.Commands.Management, Microsoft.WSMan.Management 등의 어셈블리를 추가해야 합니다.
이런 불편함을 Microsoft.PowerShell.SDK 단 하나의 패키지 참조로 해결할 수 있습니다. 따라서, 대개의 경우에는 아래와 같은 2개의 패키지 참조를 해야 합니다.
Install-Package System.Management.Automation
Install-Package Microsoft.PowerShell.SDK
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]