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

비밀번호

댓글 작성자
 




... 46  47  [48]  49  50  51  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12740정성태7/29/202114656오류 유형: 747. SharePoint - InvalidOperationException 0x80131509
12739정성태7/28/202115019오류 유형: 746. Azure Active Directory - IDW10106: The 'ClientId' option must be provided.
12738정성태7/28/202115969오류 유형: 745. Azure Active Directory - Client credential flows must have a scope value with /.default suffixed to the resource identifier (application ID URI).
12737정성태7/28/202115072오류 유형: 744. Azure Active Directory - The resource principal named api://...[client_id]... was not found in the tenant
12736정성태7/28/202115365오류 유형: 743. Active Azure Directory에서 "API permissions"의 권한 설정이 "Not granted for ..."로 나오는 문제
12735정성태7/27/202115203.NET Framework: 1081. C# - Azure AD 인증을 지원하는 데스크톱 애플리케이션 예제(Windows Forms) [2]파일 다운로드1
12734정성태7/26/202132021스크립트: 20. 특정 단어로 시작하거나/끝나는 문자열을 포함/제외하는 정규 표현식 - Look-around
12733정성태7/23/202119940.NET Framework: 1081. Self-Contained/SingleFile 유형의 .NET Core/5+ 실행 파일을 임베딩한다면? [1]파일 다운로드2
12732정성태7/23/202113355오류 유형: 742. SharePoint - The super user account utilized by the cache is not configured.
12731정성태7/23/202115388개발 환경 구성: 584. Add Internal URLs 화면에서 "Save" 버튼이 비활성화 된 경우
12730정성태7/23/202116852개발 환경 구성: 583. Visual Studio Code - Go 코드에서 입력을 받는 경우
12729정성태7/22/202115300.NET Framework: 1080. xUnit 단위 테스트에 메서드/클래스 수준의 문맥 제공 - Fixture
12728정성태7/22/202115333.NET Framework: 1079. MSTestv2 단위 테스트에 메서드/클래스/어셈블리 수준의 문맥 제공
12727정성태7/21/202116511.NET Framework: 1078. C# 단위 테스트 - MSTestv2/NUnit의 Assert.Inconclusive 사용법(?) [1]
12726정성태7/21/202116282VS.NET IDE: 169. 비주얼 스튜디오 - 단위 테스트 선택 시 MSTestv2 외의 xUnit, NUnit 사용법 [1]
12725정성태7/21/202114766오류 유형: 741. Failed to find the "go" binary in either GOROOT() or PATH
12724정성태7/21/202117703개발 환경 구성: 582. 윈도우 환경에서 Visual Studio Code + Go (Zip) 개발 환경 [1]
12723정성태7/21/202114534오류 유형: 740. SharePoint - Alternate access mappings have not been configured 경고
12722정성태7/20/202114308오류 유형: 739. MSVCR110.dll이 없어 exe 실행이 안 되는 경우
12721정성태7/20/202115483오류 유형: 738. The trust relationship between this workstation and the primary domain failed. - 세 번째 이야기
12720정성태7/19/202114760Linux: 43. .NET Core/5+ 응용 프로그램의 Ubuntu (Debian) 패키지 준비
12719정성태7/19/202113930오류 유형: 737. SharePoint 설치 시 "0x800710D8 The object identifier does not represent a valid object." 오류 발생
12718정성태7/19/202113934개발 환경 구성: 581. Windows에서 WSL로 파일 복사 시 root 소유권으로 적용되는 문제파일 다운로드1
12717정성태7/18/202114127Windows: 195. robocopy에서 파일의 ADS(Alternate Data Stream) 정보 복사를 제외하는 방법
12716정성태7/17/202115263개발 환경 구성: 580. msbuild의 Exec Task에 robocopy를 사용하는 방법파일 다운로드1
12715정성태7/17/202122445오류 유형: 736. Windows - MySQL zip 파일 버전의 "mysqld --skip-grant-tables" 실행 시 비정상 종료 [1]
... 46  47  [48]  49  50  51  52  53  54  55  56  57  58  59  60  ...