Microsoft MVP성태의 닷넷 이야기
Windows: 276. 명령행에서 원격 서비스를 동기/비동기로 시작/중지 [링크 복사], [링크+제목 복사],
조회: 3932
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 12개 있습니다.)
개발 환경 구성: 1. batch 파일에서 실행한 exe에서 batch 실행 문맥의 환경 변수 설정
; https://www.sysnet.pe.kr/2/0/238

개발 환경 구성: 89. 배치(batch) 파일에서 또 다른 배치 파일을 동기 방식으로 실행 및 반환값 얻기
; https://www.sysnet.pe.kr/2/0/958

개발 환경 구성: 103. DOS batch - 동기 방식으로 원격 서비스 제어
; https://www.sysnet.pe.kr/2/0/989

개발 환경 구성: 144. 윈도우에서도 유닉스처럼 명령행으로 원격 접속하는 방법
; https://www.sysnet.pe.kr/2/0/1245

개발 환경 구성: 166. DOS - ping 결과에서 평균 응답 시간값 추출하기
; https://www.sysnet.pe.kr/2/0/1340

개발 환경 구성: 215. DOS batch - 하나의 .bat 파일에서 다중 .bat 파일을 (비동기로) 실행하는 방법
; https://www.sysnet.pe.kr/2/0/1629

개발 환경 구성: 242. 배치 파일에서 Thread.Sleep 효과를 주는 방법
; https://www.sysnet.pe.kr/2/0/1768

개발 환경 구성: 328. Visual Studio(devenv.exe)를 배치 파일(.bat)을 통해 실행하는 방법
; https://www.sysnet.pe.kr/2/0/11293

스크립트: 13. 윈도우 배치(Batch) 스크립트에서 날짜/시간 문자열을 구하는 방법
; https://www.sysnet.pe.kr/2/0/11742

스크립트: 16. cmd.exe의 for 문에서는 ERRORLEVEL이 설정되지 않는 문제
; https://www.sysnet.pe.kr/2/0/12039

스크립트: 66. Windows 디렉터리 경로를 WSL의 /mnt 포맷으로 구하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/13685

Windows: 276. 명령행에서 원격 서비스를 동기/비동기로 시작/중지
; https://www.sysnet.pe.kr/2/0/13861




Windows - 명령행에서 원격 서비스를 동기/비동기로 시작/중지

예전에도 설명했지만,

DOS batch - 동기 방식으로 원격 서비스 제어
; https://www.sysnet.pe.kr/2/0/989

net start/stop 명령어는 서비스를 동기 방식으로 시작/중지합니다. 하지만 net 명령어는 원격 서버의 서비스를 제어할 수는 없는데요, 그래서 sc 명령어를 사용할 수는 있지만 아쉽게도 그것은 비동기 방식으로만 가능합니다.

이에 대해 간단하게 go 언어로 작성한 서비스 예제로 실습해 보겠습니다.

Golang - 콘솔 응용 프로그램을 NT 서비스를 지원하도록 변경
; https://www.sysnet.pe.kr/2/0/13854

동기/비동기 서비스 제어를 차이가 느껴지도록 하기 위해 서비스 시작/중지 시에 시간을 8초 정도 지연하는 코드를 추가합니다.

func (m *myService) Execute(args []string, r <-chan svc.ChangeRequest, status chan<- svc.Status) (bool, uint32) {
    const cmdsAccepted = svc.AcceptStop | svc.AcceptShutdown | svc.AcceptPauseAndContinue

    time.Sleep(8 * time.Second) 

    log.Print("Starting service...!")
    go doMyWork(false)

    status <- svc.Status{State: svc.Running, Accepts: cmdsAccepted}

loop:
    for {
        select {
        case c := <-r:
            switch c.Cmd {
            case svc.Interrogate:
                status <- c.CurrentStatus
            case svc.Stop, svc.Shutdown:
                log.Print("Shutting service...!")
                break loop
            case svc.Pause:
                status <- svc.Status{State: svc.Paused, Accepts: cmdsAccepted}
            case svc.Continue:
                status <- svc.Status{State: svc.Running, Accepts: cmdsAccepted}
            default:
                log.Printf("Unexpected service control request #%d\n", c)
            }
        }
    }

    status <- svc.Status{State: svc.StopPending}

    time.Sleep(8 * time.Second) 

    status <- svc.Status{State: svc.Stopped}
    return false, 1
}

자, 그럼 저걸 실습 PC 중 1대에 서비스로 등록 후,

// A 머신에서 실행

c:\temp> sc create go_sample binpath= "c:\temp\go_sample.exe" start= delayed-auto

또 다른 PC에서 아래와 같이 sc 명령어로 원격 서비스를 시작/중지 테스트를 해보면,

// B 머신에서 실행 (A 머신의 이름이 win2012r2라고 가정)

C:\temp> sc \\win2012r2 start go_sample

SERVICE_NAME: go_sample
        TYPE               : 10  WIN32_OWN_PROCESS
        STATE              : 2  START_PENDING
                                (NOT_STOPPABLE, NOT_PAUSABLE, IGNORES_SHUTDOWN)
        WIN32_EXIT_CODE    : 0  (0x0)
        SERVICE_EXIT_CODE  : 0  (0x0)
        CHECKPOINT         : 0x0
        WAIT_HINT          : 0x7d0
        PID                : 4828
        FLAGS              :

C:\temp> sc \\win2012r2 stop go_sample

SERVICE_NAME: go_sample
        TYPE               : 10  WIN32_OWN_PROCESS
        STATE              : 4  RUNNING
                                (STOPPABLE, PAUSABLE, ACCEPTS_SHUTDOWN)
        WIN32_EXIT_CODE    : 0  (0x0)
        SERVICE_EXIT_CODE  : 0  (0x0)
        CHECKPOINT         : 0x0
        WAIT_HINT          : 0x0

8초의 대기 시간 없이 sc.exe는 제어를 곧바로 명령행 shell에 반환을 합니다. 즉, 비동기적으로 호출만 하고 결과를 대기하지 않는 것입니다. (물론, 억지로 서비스 상태가 "STOPPED"로 바뀔 때까지 대기하는 방법을 사용하는 것도 방법일 수 있습니다.)

만약 동기 방식으로 원격 서비스 제어를 원한다면 PowerShell로 대체할 수 있습니다.

// 서비스 중지
C:\temp> powershell "Get-Service -COMPUTER win2012r2 go_sample | Stop-Service"

// 서비스 시작
C:\temp> powershell "Get-Service -COMPUTER win2012r2 go_sample | Start-Service"

2개 모두 동기적으로 호출을 하기 때문에 8초 정도의 대기 시간이 발생하게 됩니다. 여기까지, 상황에 따라 정리해 보면 아래와 같습니다.

  • net 명령어: 동기적 호출, 로컬 제어 O, 원격 제어 X
  • sc 명령어: 비동기적 호출, 원격 서버 제어 가능
  • PowerShell: 동기적 호출, 원격 서버 제어 가능




참고로, go 언어의 경우 Stop에 반응하는 부분에서 아래의 StopPending을 추가하지 않으면,

// status <- svc.Status{State: svc.StopPending}

time.Sleep(8 * time.Second) 

status <- svc.Status{State: svc.Stopped}

해당 서비스를 중지 시 이런 오류가 발생합니다. (그래도 서비스 중지는 됩니다.)

Windows could not stop the go_sample service on Local Computer.
The service did not return an error.  This could be an internal Windows error or an internal service error.
If the problem persists, contact your system administrator.

같은 8초 대기의 상황에서, 시작 시에는 굳이 StartPending을 추가하지 않아도 문제가 없는데, 중지 시에는 반드시 StopPending 단계를 거쳐야만 SCM과의 깔끔한 서비스 절차를 밟게 되는군요. ^^;




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 1/11/2025]

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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  [37]  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13013정성태3/22/202212822개발 환경 구성: 641. WSL 우분투 인스턴스에 파이썬 2.7 개발 환경 구성하는 방법
13012정성태3/21/202212469오류 유형: 803. C# - Local '...' or its members cannot have their address taken and be used inside an anonymous method or lambda expression
13011정성태3/21/202215957오류 유형: 802. 윈도우 운영체제에서 웹캠 카메라 인식이 안 되는 경우
13010정성태3/21/202212681오류 유형: 801. Oracle.ManagedDataAccess.Core - GetTypes 호출 시 "Could not load file or assembly 'System.DirectoryServices.Protocols...'" 오류
13009정성태3/20/202215286개발 환경 구성: 640. docker - ibmcom/db2 컨테이너 실행
13008정성태3/19/202214867VS.NET IDE: 176. 비주얼 스튜디오 - 솔루션 탐색기에서 프로젝트를 선택할 때 csproj 파일이 열리지 않도록 만드는 방법
13007정성태3/18/202213267.NET Framework: 1181. C# - Oracle.ManagedDataAccess의 Pool 및 그것의 연결 개체 수를 알아내는 방법파일 다운로드1
13006정성태3/17/202215674.NET Framework: 1180. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 remuxing.c 예제 포팅
13005정성태3/17/202213957오류 유형: 800. C# - System.InvalidOperationException: Late bound operations cannot be performed on fields with types for which Type.ContainsGenericParameters is true.
13004정성태3/16/202213462디버깅 기술: 182. windbg - 닷넷 메모리 덤프에서 AppDomain에 걸친 정적(static) 필드 값을 조사하는 방법
13003정성태3/15/202213791.NET Framework: 1179. C# - (.NET Framework를 위한) Oracle.ManagedDataAccess 패키지의 성능 카운터 설정 방법
13002정성태3/14/202215391.NET Framework: 1178. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 http_multiclient.c 예제 포팅
13001정성태3/13/202215992.NET Framework: 1177. C# - 닷넷에서 허용하는 메서드의 매개변수와 호출 인자의 최대 수
13000정성태3/12/202214936.NET Framework: 1176. C# - Oracle.ManagedDataAccess.Core의 성능 카운터 설정 방법
12999정성태3/10/202214783.NET Framework: 1175. Visual Studio - 프로젝트 또는 솔루션의 Clean 작업 시 응용 프로그램에서 생성한 파일을 함께 삭제파일 다운로드1
12998정성태3/10/202213189.NET Framework: 1174. C# - ELEMENT_TYPE_FNPTR 유형의 사용 예
12997정성태3/10/202222298오류 유형: 799. Oracle.ManagedDataAccess - "ORA-01882: timezone region not found" 오류가 발생하는 이유
12996정성태3/9/202223743VS.NET IDE: 175. Visual Studio - 인텔리센스에서 오버로드 메서드를 키보드로 선택하는 방법
12995정성태3/8/202216109.NET Framework: 1173. .NET에서 Producer/Consumer를 구현한 BlockingCollection<T>
12994정성태3/8/202215170오류 유형: 798. WinDbg - Failed to load data access module, 0x80004002
12993정성태3/4/202214878.NET Framework: 1172. .NET에서 Producer/Consumer를 구현하는 기초 인터페이스 - IProducerConsumerCollection<T>
12992정성태3/3/202217845.NET Framework: 1171. C# - BouncyCastle을 사용한 암호화/복호화 예제파일 다운로드1
12991정성태3/2/202215646.NET Framework: 1170. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 transcode_aac.c 예제 포팅
12990정성태3/2/202215772오류 유형: 797. msbuild - The BaseOutputPath/OutputPath property is not set for project '[...].vcxproj'
12989정성태3/2/202213764오류 유형: 796. mstest.exe - System.IO.FileNotFoundException: Could not load file or assembly 'Microsoft.VisualStudio.QualityTools.Tips.WebLoadTest.Tip
12988정성태3/2/202212182오류 유형: 795. CI 환경에서 Docker build 시 csproj의 Link 파일에 대한 빌드 오류
... 31  32  33  34  35  36  [37]  38  39  40  41  42  43  44  45  ...