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

비밀번호

댓글 작성자
 




... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12386정성태10/28/202019239Linux: 34. 사용자 정보를 함께 출력하는 리눅스의 ps 명령어 사용 방법
12385정성태10/28/202016583오류 유형: 673. openssl - req: No value provided for Subject Attribute CN, skipped
12384정성태10/27/202018906오류 유형: 672. AllowPartiallyTrustedCallers 특성이 적용된 어셈블리의 struct 멤버 메서드를 재정의하면 System.Security.VerificationException 예외 발생
12383정성태10/27/202018974.NET Framework: 956. C# 9.0 - (7) 패턴 일치 개선 사항(Pattern matching enhancements) [3]파일 다운로드1
12382정성태10/26/202016711오류 유형: 671. dotnet build - The local source '...' doesn't exist
12381정성태10/26/202018735VC++: 137. C++ stl map의 사용자 정의 타입을 key로 사용하는 방법 [1]파일 다운로드1
12380정성태10/26/202014707오류 유형: 670. Visual Studio - Squash_FailureCommitsReset
12379정성태10/21/202020285.NET Framework: 955. .NET 메서드의 Signature 바이트 코드 분석 [1]파일 다운로드2
12378정성태10/15/202019223.NET Framework: 954. C# - x86/x64 환경에 따라 달라지는 P/Invoke 함수의 export 이름 [1]파일 다운로드1
12377정성태10/15/202019644디버깅 기술: 172. windbg - 파일 열기 시점에 bp를 걸어 파일명 알아내는 방법(Managed/Unmanaged)
12376정성태10/15/202015740오류 유형: 669. windbg - sos의 name2ee 명령어 실행 시 "Failed to request module list." 오류
12375정성태10/15/202017263Windows: 177. 윈도우 탐색기에서 띄우는 cmd.exe 창의 디렉터리 구분 문자가 'Yen(&#0165;)' 기호로 나오는 경우 [1]
12374정성태10/14/202023168.NET Framework: 953. C# 9.0 - (6) 함수 포인터(Function pointers) [1]파일 다운로드2
12373정성태10/14/202016934.NET Framework: 952. OpCodes.Box와 관련해 IL 형식으로 직접 코딩 시 유의할 점
12372정성태10/13/202019336.NET Framework: 951. C# 9.0 - (5) 로컬 함수에 특성 지정 가능(Attributes on local functions)파일 다운로드1
12371정성태10/13/202018247개발 환경 구성: 519. Visual Studio의 Ctrl+Shift+U (Edit.MakeUppercase) 단축키가 동작하지 않는 경우
12370정성태10/13/202018468Linux: 33. Linux - nmcli를 이용한 고정 IP 설정
12369정성태10/12/202022113Windows: 176. Raymond Chen이 한글날에 밝히는 윈도우의 한글 자모 분리 현상 [3]
12368정성태10/12/202018200오류 유형: 668. VSIX 확장 빌드 - The "GetDeploymentPathFromVsixManifest" task failed unexpectedly.
12367정성태10/12/202030698오류 유형: 667. Ubuntu - Temporary failure resolving 'kr.archive.ubuntu.com' [2]
12366정성태10/12/202020038.NET Framework: 950. C# 9.0 - (4) 원시 크기 정수(Native ints) [1]파일 다운로드1
12365정성태10/12/202018384.NET Framework: 949. C# 9.0 - (3) 람다 메서드의 매개 변수 무시(Lambda discard parameters)파일 다운로드1
12364정성태10/11/202019376.NET Framework: 948. C# 9.0 - (2) localsinit 플래그 내보내기 무시(Suppress emitting localsinit flag)파일 다운로드1
12363정성태10/11/202021099.NET Framework: 947. C# 9.0 - (1) 대상으로 형식화된 new 식(Target-typed new expressions) [2]파일 다운로드1
12362정성태10/11/202017640VS.NET IDE: 151. Visual Studio 2019에 .NET 5 rc/preview 적용하는 방법
12361정성태10/11/202019660.NET Framework: 946. C# 9.0을 위한 개발 환경 구성
... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...