Microsoft MVP성태의 닷넷 이야기
Windows: 146. PowerShell로 원격 프로세스(EXE, BAT) 실행하는 방법 [링크 복사], [링크+제목 복사],
조회: 25934
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

PowerShell로 원격 프로세스(EXE, BAT) 실행하는 방법

다음과 같은 질문이 있군요. ^^

PowerShell 원격 실행 관련 질문...
; http://lab.gamecodi.com/board/zboard.php?id=GAMECODILAB_QnA_etc&no=5258&z=

이참에 한번 실습해보는 거죠. ^^

실습을 위해 Server1 컴퓨터에 다음과 같은 배치 파일을 하나 만들어 둡니다.

REM c:\temp\test.bat

c:\temp\ConsoleApp1.exe

그런 다음 아래와 같은 소스 코드로 ConsoleApp1.exe를 빌드해 c:\temp 폴더에 복사해 둡니다.

using System.Threading;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            EventWaitHandle ewh = new EventWaitHandle(false, EventResetMode.ManualReset);
            ewh.WaitOne();
        }
    }
}

준비는 끝났고 동작 확인을 위해 다른 컴퓨터에서 Server1 컴퓨터에 PowerShell로 이렇게 명령을 내릴 수 있습니다.

Invoke-Command -ComputerName Server1 -ScriptBlock {Invoke-Expression -Command:"cmd /C 'c:\temp\test.bat'"}

그럼 Server1 컴퓨터에서 ConsoleApp1.exe가 실행된 체로 종료를 안 하고 있기 때문에 위의 PowerShell 명령어도 실행이 끝나지 않고 기다리게 됩니다. 이 상태에서 Server1 컴퓨터의 작업 관리자에서 ConsoleApp1.exe를 강제 종료하면 PowerShell 명령어도 함께 끝나는 것을 확인할 수 있습니다.




하지만 여기서 원하는 것은 PowerShell 명령어가 대상 PC의 명령어 완료를 기다리지 않고 끝나는 것입니다. 이를 위해 test.bat 파일을 다음과 같이 start 명령어로 실행할 수 있습니다.

Invoke-Command -ComputerName Server1 -ScriptBlock {Invoke-Expression -Command:"start 'c:\temp\test.bat'"}

이렇게 하고 실행하면 Invoke-Command는 대상 명령어가 종료될 때까지 기다리지 않고 곧바로 제어를 반환합니다. 그런데 여기서 문제가 있습니다. PowerShell은 원격 명령어 실행을 위해 Session을 Invoke-Command의 명령어 종료와 함께 닫아버린다는 것입니다. 이로 인해 원격 Session 내에서 실행 중이던 프로그램까지도 강제 종료가 됩니다. 즉, 위와 같이 명령어를 실행하면 test.bat 파일(cmd.exe)이 잠깐 실행되다가 곧바로 종료됩니다. (확률적으로 ConsoleApp1.exe도 실행될 때도 있는데 그것 역시 Invoke-Command가 제어를 반환하면서 곧바로 종료합니다.)

이 문제를 해결하려면 Invoke-Command가 제어를 반환해도 원격 Session을 닫지 않고 유지할 수 있어야 합니다. 그리고 바로 이때 사용할 수 있는 옵션이 "-InDisconnectedSession"입니다.

Invoke-Command -InDisconnectedSession -ComputerName Server1 -ScriptBlock {Invoke-Expression -Command:"start 'c:\temp\test.bat'"}

위와 같이 실행하면 화면에는 다음과 같이 끊긴 원격 세션에 대한 정보를 알려주는 메시지가 출력됩니다.

PS C:\Users\TestUser> Invoke-Command -InDisconnectedSession -ComputerName Server1 -ScriptBlock {Invoke-Expression -Command:"start 'c:\temp\test.bat'"}

 Id Name            ComputerName    ComputerType    State         ConfigurationName     Availability
 -- ----            ------------    ------------    -----         -----------------     ------------
 42 Session41       Server1       RemoteMachine   Disconnected  Microsoft.PowerShell          None

PS C:\Users\TestUser>

이와 함께 Server1 컴퓨터에는 ConsoleApp1.exe 프로세스가 떠 있는 것을 확인할 수 있습니다. 만약 원한다면 해당 세션을 다음과 같은 명령어로 강제 종료하는 것도 가능합니다.

PS C:\Windows\system32> Connect-PSSession -ComputerName Server1 -Name Session41

 Id Name            ComputerName    ComputerType    State         ConfigurationName     Availability
 -- ----            ------------    ------------    -----         -----------------     ------------
 84 Session41       Server1       RemoteMachine   Opened        Microsoft.PowerShell          Busy



PS C:\Windows\system32> Remove-PSSession -ComputerName Server1

PS C:\Windows\system32> 

또는,

PS C:\Users\TestUser> Get-PSSession -ComputerName Server1 | Remove-PSSession

PS C:\Users\TestUser> 

관련해서 명령어들은 다음의 글에서 더 자세하게 볼 수 있습니다.

PowerShell disconnected remote sessions
; https://4sysops.com/archives/powershell-disconnected-remote-sessions/




참고로, 다음과 같은 오류가 발생한다면?

PS C:\Users\TestUser> Invoke-Command -ComputerName Server1 -ErrorAction Stop -ScriptBlock {Invoke-Expression -Command:"start 'c:\temp\test.bat'"}

[Server1] Connecting to remote server Server1 failed with the following error message : The client cannot connect 
to the destination specified in the request. Verify that the service on the destination is running and is accepting re
quests. Consult the logs and documentation for the WS-Management service running on the destination, most commonly IIS
 or WinRM. If the destination is the WinRM service, run the following command on the destination to analyze and config
ure the WinRM service: "winrm quickconfig". For more information, see the about_Remote_Troubleshooting Help topic.
At line:1 char:1
+ Invoke-Command -ComputerName Server1 -ScriptBlock ...
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : OpenError: (Server1:String) [], PSRemotingTransportException
    + FullyQualifiedErrorId : CannotConnect,PSSessionStateBroken

명령에서 알려주는 대로 "winrm quickconfig" 명령어를 대상 PC에서 실행해 주면 됩니다.

C:\Windows\system32> winrm quickconfig
WinRM already is set up to receive requests on this machine.
WinRM is not set up to allow remote access to this machine for management.
The following changes must be made:

Create a WinRM listener on HTTP://* to accept WS-Man requests to any IP on this machine.

Make these changes [y/n]? y

WinRM has been updated for remote management.

Created a WinRM listener on HTTP://* to accept WS-Man requests to any IP on this machine.




다음과 같은 오류가 발생할 수도 있습니다.

PS C:\Users\TestUser> Invoke-Command -InDisconnectedSession -ComputerName Server1 -ScriptBlock {Invoke-Expression -Command:"start 'c:\temp\test.bat'"}

Disconnected sessions are supported only when the remote computer is running Windows PowerShell 3.0 or a later version
 of Windows PowerShell.
    + CategoryInfo          : OperationStopped: (Server1:String) [], PSRemotingDataStructureException
    + FullyQualifiedErrorId : JobFailure
    + PSComputerName        : Server1

오류 메시지 그대로 PowerShell 버전이 낮아서 그런 것입니다. 참고로 대상 컴퓨터에서 다음과 같은 명령어로 PowerShell 버전을 확인할 수 있습니다.

PS C:\Users\TestUser> $PSVersionTable.PSVersion

Major  Minor  Build  Revision
-----  -----  -----  --------
2      0      -1     -1




Connect-PSSession 명령어 실행 시 다음과 같이 오류가 발생하는 경우도 있습니다.

PS C:\Windows\system32> Connect-PSSession -ComputerName Server1 -Name Session45
Connect-PSSession : The connect operation failed for session Session45 with the following error message: Connecting to
 remote server Server1 failed with the following error message : The server that is running Windows PowerShell does 
not support connect operations on the protocolversion 2.3  that is negotiated by the client computer. Make sure the cl
ient computer is compatible with the build 6.3.9600.18773 and the protocol version 2.2 of Windows PowerShell. For more
 information, see the about_Remote_Troubleshooting Help topic.
At line:1 char:1
+ Connect-PSSession -ComputerName Server1 -Name Session45
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : InvalidOperation: (:) [Connect-PSSession], RuntimeException
    + FullyQualifiedErrorId : -2141974621,PSSessionConnectFailed,Microsoft.PowerShell.Commands.ConnectPSSessionCommand


메시지 그대로 protocolversion이 맞지 않아서 그런 것입니다. 제가 해보니까, 적어도 PowerShell 5.1 버전 이상에서는 잘 동작했습니다.





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

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/15/2018]

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

비밀번호

댓글 작성자
 



2020-07-27 11시06분
SpaceRunner
; https://github.com/Mr-B0b/SpaceRunner

This tool enables the compilation of a C# program that will execute arbitrary PowerShell code, without launching PowerShell processes through the use of runspace.
정성태

... 61  62  63  64  65  66  67  68  69  70  71  [72]  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
11847정성태3/17/201915206Linux: 7. 리눅스 C/C++ - 공유 라이브러리 동적 로딩 후 export 함수 사용 방법파일 다운로드1
11846정성태3/15/201913579Linux: 6. getenv, setenv가 언어/운영체제마다 호환이 안 되는 문제
11845정성태3/15/201914296Linux: 5. Linux 응용 프로그램의 (C++) so 의존성 줄이기(ReleaseMinDependency) [3]
11844정성태3/14/201915124개발 환경 구성: 434. Visual Studio 2019 - 리눅스 프로젝트를 이용한 공유/실행(so/out) 프로그램 개발 환경 설정 [1]파일 다운로드1
11843정성태3/14/201910862기타: 75. MSDN 웹 사이트를 기본으로 영문 페이지로 열고 싶다면?
11842정성태3/13/201910269개발 환경 구성: 433. 마이크로소프트의 CoreCLR 프로파일러 예제를 Visual Studio CMake로 빌드하는 방법 [1]파일 다운로드1
11841정성태3/13/201910237VS.NET IDE: 132. Visual Studio 2019 - CMake의 컴파일러를 기본 g++에서 clang++로 변경
11840정성태3/13/201911340오류 유형: 526. 윈도우 10 Ubuntu App 환경에서는 USB 외장 하드 접근 불가
11839정성태3/12/201914109디버깅 기술: 124. .NET Core 웹 앱을 호스팅하는 Azure App Services의 프로세스 메모리 덤프 및 windbg 분석 개요 [3]
11838정성태3/7/201916862.NET Framework: 811. (번역글) .NET Internals Cookbook Part 1 - Exceptions, filters and corrupted processes [1]파일 다운로드1
11837정성태3/6/201926614기타: 74. 도서: 시작하세요! C# 7.3 프로그래밍 [10]
11836정성태3/5/201914422오류 유형: 525. Visual Studio 2019 Preview 4/RC - C# 8.0 Missing compiler required member 'System.Range..ctor' [1]
11835정성태3/5/201914190.NET Framework: 810. C# 8.0의 Index/Range 연산자를 .NET Framework에서 사용하는 방법 및 비동기 스트림의 컴파일 방법 [3]파일 다운로드1
11834정성태3/4/201913075개발 환경 구성: 432. Visual Studio 없이 최신 C# (8.0) 컴파일러를 사용하는 방법
11833정성태3/4/201913838개발 환경 구성: 431. Visual Studio 2019 - CMake를 이용한 공유/실행(so/out) 리눅스 프로젝트 설정파일 다운로드1
11832정성태3/4/201910871오류 유형: 524. Visual Studio CMake - rsync: connection unexpectedly closed
11831정성태3/4/201910458오류 유형: 523. Visual Studio 2019 - 새 창으로 뜬 윈도우를 닫을 때 비정상 종료
11830정성태2/26/201910244오류 유형: 522. 이벤트 로그 - Error opening event log file State. Log will not be processed. Return code from OpenEventLog is 87.
11829정성태2/26/201912155개발 환경 구성: 430. 마이크로소프트의 CoreCLR 프로파일러 예제 빌드 방법 - 리눅스 환경 [1]
11828정성태2/26/201918571개발 환경 구성: 429. Component Services 관리자의 RuntimeBroker 설정이 2개 있는 경우 [8]
11827정성태2/26/201912480오류 유형: 521. Visual Studio - Could not start the 'rsync' command on the remote host, please install it using your system package manager.
11826정성태2/26/201912362오류 유형: 520. 우분투에 .NET Core SDK 설치 시 패키지 의존성 오류
11825정성태2/25/201917210개발 환경 구성: 428. Visual Studio 2019 - CMake를 이용한 리눅스 빌드 환경 설정 [1]
11824정성태2/25/201912018오류 유형: 519. The SNMP Service encountered an error while accessing the registry key SYSTEM\CurrentControlSet\Services\SNMP\Parameters\TrapConfiguration. [1]
11823정성태2/21/201913508오류 유형: 518. IIS 관리 콘솔이 뜨지 않는 문제
11822정성태2/20/201911638오류 유형: 517. docker에 설치한 MongoDB 서버로 연결이 안 되는 경우
... 61  62  63  64  65  66  67  68  69  70  71  [72]  73  74  75  ...