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

비밀번호

댓글 작성자
 




... 61  62  [63]  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12364정성태10/11/202019448.NET Framework: 948. C# 9.0 - (2) localsinit 플래그 내보내기 무시(Suppress emitting localsinit flag)파일 다운로드1
12363정성태10/11/202021142.NET Framework: 947. C# 9.0 - (1) 대상으로 형식화된 new 식(Target-typed new expressions) [2]파일 다운로드1
12362정성태10/11/202017703VS.NET IDE: 151. Visual Studio 2019에 .NET 5 rc/preview 적용하는 방법
12361정성태10/11/202019715.NET Framework: 946. C# 9.0을 위한 개발 환경 구성
12360정성태10/8/202014941오류 유형: 666. The type or namespace name '...' does not exist in the namespace 'Microsoft.VisualStudio.TestTools' (are you missing an assembly reference?)
12359정성태10/7/202017171오류 유형: 665. Windows - 재부팅 후 iSCSI 연결이 끊기는 문제
12358정성태10/7/202018046오류 유형: 664. Web Deploy 설치 시 "A newer version of Microsoft Web Deploy 3.6 was found on this machine." 오류 [3]
12357정성태10/7/202015680오류 유형: 663. 이벤트 로그 - The storage optimizer couldn't complete retrim on New Volume
12356정성태10/7/202031303오류 유형: 662. ASP.NET Core와 500.19, 500.21 오류 (0x8007000d)
12355정성태10/3/202014837오류 유형: 661. Hyper-V Linux VM의 Internal 유형의 가상 Switch에 대한 IP 연결이 되지 않는 경우
12354정성태10/2/202028851오류 유형: 660. Web Deploy (msdeploy.axd) 실행 시 오류 기록 [1]
12353정성태10/2/202018308개발 환경 구성: 518. 비주얼 스튜디오에서 IIS 웹 서버로 "Web Deploy"를 이용해 배포하는 방법
12352정성태10/2/202019556개발 환경 구성: 517. Hyper-V Internal 네트워크에 NAT을 이용한 인터넷 연결 제공
12351정성태10/2/202018009오류 유형: 659. Nox 실행이 안 되는 경우 - Unable to bind to the underlying transport for ...
12350정성태9/25/202022429Windows: 175. 윈도우 환경에서 클라이언트 소켓의 최대 접속 수 [2]파일 다운로드1
12349정성태9/25/202016593Linux: 32. Ubuntu 20.04 - docker를 위한 tcp 바인딩 추가
12348정성태9/25/202017611오류 유형: 658. 리눅스 docker - Got permission denied while trying to connect to the Docker daemon socket at unix:///var/run/docker.sock
12347정성태9/25/202033142Windows: 174. WSL 2의 네트워크 통신 방법 [4]
12346정성태9/25/202016623오류 유형: 657. IIS - http://localhost 방문 시 Service Unavailable 503 오류 발생
12345정성태9/25/202016158오류 유형: 656. iisreset 실행 시 "Restart attempt failed." 오류가 발생하지만 웹 서비스는 정상적인 경우파일 다운로드1
12344정성태9/25/202018078Windows: 173. 서비스 관리자에 "IIS Admin Service"가 등록되어 있지 않다면?
12343정성태9/24/202029138.NET Framework: 945. C# - 닷넷 응용 프로그램에서 메모리 누수가 발생할 수 있는 패턴 [5]
12342정성태9/24/202019147디버깅 기술: 171. windbg - 인스턴스가 살아 있어 메모리 누수가 발생하고 있는지 확인하는 방법
12341정성태9/23/202017184.NET Framework: 944. C# - 인스턴스가 살아 있어 메모리 누수가 발생하고 있는지 확인하는 방법파일 다운로드1
12340정성태9/23/202016833.NET Framework: 943. WPF - WindowsFormsHost를 담은 윈도우 생성 시 메모리 누수
12339정성태9/21/202017019오류 유형: 655. 코어 모드의 윈도우는 GUI 모드의 윈도우로 교체가 안 됩니다.
... 61  62  [63]  64  65  66  67  68  69  70  71  72  73  74  75  ...