Microsoft MVP성태의 닷넷 이야기
닷넷: 2325. C# - PowerShell과 연동하는 방법 [링크 복사], [링크+제목 복사],
조회: 8344
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... [151]  152  153  154  155  156  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1370정성태10/31/201225953VS.NET IDE: 75. Visual Studio - "Active Solution Platform" 변경을 툴바에서 하는 방법
1369정성태10/31/201239038개발 환경 구성: 174. 윈도우에서 Mono 개발 환경 구성 [4]
1368정성태10/31/201231225개발 환경 구성: 173. Windows Phone SDK 8.0 설치
1367정성태10/30/201238214개발 환경 구성: 172. IIS 7.5부터 지원되는 웹 사이트 자동 시작 모드 [1]
1366정성태10/24/201229847개발 환경 구성: 171. GTK+를 윈도우 환경에 수작업 설치
1365정성태10/24/201228703개발 환경 구성: 170. 우분투 데스크톱 Active Directory 가입하기 [2]
1364정성태10/19/201225579Windows: 66. Hyper-V 2012에서 별도의 네트워크 카드를 이용한 Live Migration
1363정성태10/16/201233145개발 환경 구성: 169. Objective-C의 대안 - Xamarin의 Mono를 이용한 C# iOS 개발 환경 [2]
1362정성태10/16/201233142개발 환경 구성: 168. 우분투 서버 Active Directory 가입하기
1361정성태10/12/201226458.NET Framework: 341. .NET COM+ 를 Managed/Native 클라이언트에서 각각 호출했을 때의 콜 스택 비교 [4]파일 다운로드1
1360정성태10/9/201231252.NET Framework: 340. Windows Server 2012 - .NET Framework 1.x 미지원
1359정성태10/9/201267236Windows: 65. 윈도우 8 - Internet Explorer 10을 32비트 또는 64비트로 통합 [5]
1358정성태9/27/201230524.NET Framework: 339. .NET Profiler 주의 사항 - 하나의 exe 프로세스 내에 다중 .NET 런타임 사용
1357정성태9/27/201227064Windows: 64. Hyper-V - Windows XP의 Live Migration 오류
1356정성태9/26/201233609Windows: 63. 윈도우 서버 2012 - Hyper-V의 새로운 기능 Live Migration [6]
1355정성태9/21/201231710Team Foundation Server: 49. TFS 2012 Express의 필수 보완 작업: 데이터베이스 백업 [1]
1354정성태9/19/201228665.NET Framework: 338. .NET CLR GC 시간 측정하는 방법파일 다운로드1
1353정성태9/17/201229535.NET Framework: 337. Python의 생성기와 코루틴을 C#으로 표현하면. [2]파일 다운로드1
1352정성태9/13/201227477.NET Framework: 336. .NET Profiler가 COM 개체일까?
1351정성태9/13/201232240디버깅 기술: 49. windbg - .NET Framework 스레드 개체의 COM Apartment 유형 확인하는 방법
1350정성태9/12/201232843개발 환경 구성: 167. (실은) 무료가 아니었던 AWS EC2 서비스 [4]
1349정성태9/11/201264719VS.NET IDE: 74. Visual Studio의 '새 파일'을 UTF-8 인코딩으로 지정하는 방법 [4]
1348정성태9/11/201232120오류 유형: 164. Active Directory - Functional Level 승격이 안 되는 문제
1347정성태9/10/201234975Windows: 62. 윈도우 서버 2012 - Hyper-V 서버 마이그레이션 [1]
1346정성태9/10/201235096Windows: 61. 윈도우 서버 2012 - Active Directory 서버 마이그레이션
1345정성태9/10/201239886스크립트: 12. 파이썬 - Win32 DLL 연동 [2]
... [151]  152  153  154  155  156  157  158  159  160  161  162  163  164  165  ...