Microsoft MVP성태의 닷넷 이야기
닷넷: 2325. C# - PowerShell과 연동하는 방법 [링크 복사], [링크+제목 복사],
조회: 3317
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 4개 있습니다.)
Windows: 146. PowerShell로 원격 프로세스(EXE, BAT) 실행하는 방법
; https://www.sysnet.pe.kr/2/0/11450

개발 환경 구성: 738. PowerShell - 원격 호출 시 "powershell.exe"가 아닌 "pwsh.exe" 환경으로 명령어를 실행하는 방법
; https://www.sysnet.pe.kr/2/0/13858

닷넷: 2325. C# - PowerShell과 연동하는 방법
; https://www.sysnet.pe.kr/2/0/13892

닷넷: 2326. C# - PowerShell과 연동하는 방법 (두 번째 이야기)
; https://www.sysnet.pe.kr/2/0/13897




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




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/24/2025]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 181  182  [183]  184  185  186  187  188  189  190  191  192  193  194  195  ...
NoWriterDateCnt.TitleFile(s)
396정성태2/13/200730015오류 유형: 18. "Automatic Updates" 서비스 CPU 100% 점유 현상
393정성태11/8/200619546오류 유형: 17. Unable to start debugging - The binding handle is invalid.
371정성태10/23/200618720오류 유형: 16. STS Communication failed.
370정성태11/12/200622610.NET Framework: 75. Windows CardSpace 이야기 (이 글의 내용은 재작성되어질 예정입니다.)
375정성태10/25/200624614    답변글 .NET Framework: 75.1. 개인 발행 카드에 대한 Microsoft 예제 실습(이 글의 내용은 재작성되어질 예정입니다.)
376정성태10/27/200624203    답변글 .NET Framework: 75.2. "Windows CardSpace"와 "인증서 서비스"의 만남(이 글의 내용은 재작성되어질 예정입니다.)
377정성태10/26/200623857    답변글 .NET Framework: 75.3. Managed Card 발행에 대한 Microsoft 예제 실습 (1) - CardWriter (이 글의 내용은 재작성되어질 예정입니다.)
385정성태11/6/200626482    답변글 .NET Framework: 75.4. Managed Card 발행에 대한 Microsoft 예제 실습 (2) - STS 구현 (이 글의 내용은 재작성되어질 예정입니다.) [7]
387정성태11/2/200627264    답변글 .NET Framework: 75.5. Windows CardSpace와 SYSNET 사이트의 만남 (이 글의 내용은 재작성되어질 예정입니다.) [1]
397정성태11/11/200624773    답변글 .NET Framework: 75.6. CardWriter.csproj와 함께 알아보는 인증서 식별 방법(이 글의 내용은 재작성되어질 예정입니다.)
398정성태11/12/200623295    답변글 .NET Framework: 75.7. 카드에 암호 거는 방법(이 글의 내용은 재작성되어질 예정입니다.)
399정성태11/12/200625551    답변글 .NET Framework: 75.8. 인증서/스마트 카드에 기반한 Managed Card - STS 구현(이 글의 내용은 재작성되어질 예정입니다.) [5]
369정성태10/22/200621047오류 유형: 15. 자동 업데이트 실패
367정성태10/22/200636829Windows: 3. IIS 7.0 다중 바인딩 설정하는 방법 [1]
365정성태10/21/200620526Windows: 2. 서버(build 5600)에 IIS 7.0 서비스와 .NET 3.0 설치 방법
359정성태10/17/200616675오류 유형: 14. VS.NET 빌드 오류 - FxCopCmd.exe returned error code 65.
358정성태10/17/200621876오류 유형: 13. WSE 3.0 서비스 관련 WSE101 오류 / Destination Unreachable
357정성태12/1/200624132.NET Framework: 74. WCF 이야기 [4]
378정성태10/28/200628919    답변글 .NET Framework: 74.1. WCF와 WSE 3.0의 활용 [4]파일 다운로드1
379정성태11/3/200627867    답변글 .NET Framework: 74.2. WCF로 구현하는 .NET Remoting [4]파일 다운로드1
380정성태10/28/200626784    답변글 .NET Framework: 74.3. 웹 서비스와 닷넷 리모팅으로써의 WCF 구현파일 다운로드1
381정성태10/28/200629190    답변글 .NET Framework: 74.4. WCF 서비스 참조 추가 메뉴 [2]
382정성태10/28/200635264    답변글 .NET Framework: 74.5. WCF 서비스를 IIS에서 호스팅하는 방법파일 다운로드1
383정성태10/28/200630037    답변글 .NET Framework: 74.6. IIS 6.0: 다중 Endpoint 제공파일 다운로드1
384정성태10/28/200626929    답변글 .NET Framework: 74.7. IIS 7.0: 다중 Endpoint 제공
389정성태11/11/200629732    답변글 .NET Framework: 74.8. WCF에 SSL 적용 (1) - Httpcfg.exe 도구를 이용한 SSL 설정
... 181  182  [183]  184  185  186  187  188  189  190  191  192  193  194  195  ...