Microsoft MVP성태의 닷넷 이야기
Windows: 276. 명령행에서 원격 서비스를 동기/비동기로 시작/중지 [링크 복사], [링크+제목 복사],
조회: 3920
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13464정성태11/28/202310404닷넷: 2174. C# - .NET 7부터 UnmanagedCallersOnly 함수 export 기능을 AOT 빌드에 통합파일 다운로드1
13463정성태11/27/20239762오류 유형: 881. Visual Studio - NU1605: Warning As Error: Detected package downgrade
13462정성태11/27/202310358오류 유형: 880. Visual Studio - error CS0246: The type or namespace name '...' could not be found
13461정성태11/26/202310159닷넷: 2173. .NET Core 3/5+ 기반의 COM Server를 registry 등록 없이 사용하는 방법파일 다운로드1
13460정성태11/26/202310201닷넷: 2172. .NET 6+ 기반의 COM Server 내에 Type Library를 내장하는 방법파일 다운로드1
13459정성태11/26/202310678닷넷: 2171. .NET Core 3/5+ 기반의 COM Server를 기존의 regasm처럼 등록하는 방법파일 다운로드1
13458정성태11/26/202310745닷넷: 2170. .NET Core/5+ 기반의 COM Server를 tlb 파일을 생성하는 방법(tlbexp)
13457정성태11/25/202310389VS.NET IDE: 187. Visual Studio - 16.9 버전부터 추가된 "Display inline type hints" 옵션
13456정성태11/25/202311115닷넷: 2169. C# - OpenAI를 사용해 PDF 데이터를 대상으로 OpenAI 챗봇 작성 [1]파일 다운로드1
13455정성태11/25/202310768닷넷: 2168. C# - Azure.AI.OpenAI 패키지로 OpenAI 사용파일 다운로드1
13454정성태11/23/202311127닷넷: 2167. C# - Qdrant Vector DB를 이용한 Embedding 벡터 값 보관/조회 (Azure OpenAI) [1]파일 다운로드1
13453정성태11/23/20239677오류 유형: 879. docker desktop 설치 시 "Invalid JSON string. (Exception from HRESULT: 0x83750007)"
13452정성태11/22/202310198닷넷: 2166. C# - Azure OpenAI API를 이용해 사용자가 제공하는 정보를 대상으로 검색하는 방법파일 다운로드1
13451정성태11/21/202310304닷넷: 2165. C# - Azure OpenAI API를 이용해 ChatGPT처럼 동작하는 콘솔 응용 프로그램 제작파일 다운로드1
13450정성태11/21/202310107닷넷: 2164. C# - Octokit을 이용한 GitHub Issue 검색파일 다운로드1
13449정성태11/21/202310568개발 환경 구성: 688. Azure OpenAI 서비스 신청 방법
13448정성태11/20/202310302닷넷: 2163. .NET 8 - Dynamic PGO를 결합한 성능 향상파일 다운로드1
13447정성태11/16/202310673닷넷: 2162. ASP.NET Core 웹 사이트의 SSL 설정을 코드로 하는 방법
13446정성태11/16/202310600닷넷: 2161. .NET Conf 2023 - Day 1 Blazor 개요 정리
13445정성태11/15/202311633Linux: 62. 리눅스/WSL에서 CA 인증서를 저장하는 방법
13444정성태11/15/202310810닷넷: 2160. C# 12 - Experimental 특성 지원
13443정성태11/14/202310509개발 환경 구성: 687. OpenSSL로 생성한 사용자 인증서를 ASP.NET Core 웹 사이트에 적용하는 방법
13442정성태11/13/202310188개발 환경 구성: 686. 비주얼 스튜디오로 실행한 ASP.NET Core 사이트를 WSL 2 인스턴스에서 https로 접속하는 방법
13441정성태11/12/202310811닷넷: 2159. C# - ASP.NET Core 프로젝트에서 서버 Socket을 직접 생성하는 방법파일 다운로드1
13440정성태11/11/20239667Windows: 253. 소켓 Listen 시 방화벽의 Public/Private 제어 기능이 비활성화된 경우
13439정성태11/10/202311563닷넷: 2158. C# - 소켓 포트를 미리 시스템에 등록/예약해 사용하는 방법(Port Exclusion Ranges)파일 다운로드1
... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...