golang.org/x/sys@v0.9.0/windows/svc/mgr/service.go (about)

     1  // Copyright 2012 The Go Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  //go:build windows
     6  // +build windows
     7  
     8  package mgr
     9  
    10  import (
    11  	"syscall"
    12  	"unsafe"
    13  
    14  	"golang.org/x/sys/windows"
    15  	"golang.org/x/sys/windows/svc"
    16  )
    17  
    18  // Service is used to access Windows service.
    19  type Service struct {
    20  	Name   string
    21  	Handle windows.Handle
    22  }
    23  
    24  // Delete marks service s for deletion from the service control manager database.
    25  func (s *Service) Delete() error {
    26  	return windows.DeleteService(s.Handle)
    27  }
    28  
    29  // Close relinquish access to the service s.
    30  func (s *Service) Close() error {
    31  	return windows.CloseServiceHandle(s.Handle)
    32  }
    33  
    34  // Start starts service s.
    35  // args will be passed to svc.Handler.Execute.
    36  func (s *Service) Start(args ...string) error {
    37  	var p **uint16
    38  	if len(args) > 0 {
    39  		vs := make([]*uint16, len(args))
    40  		for i := range vs {
    41  			vs[i] = syscall.StringToUTF16Ptr(args[i])
    42  		}
    43  		p = &vs[0]
    44  	}
    45  	return windows.StartService(s.Handle, uint32(len(args)), p)
    46  }
    47  
    48  // Control sends state change request c to the service s. It returns the most
    49  // recent status the service reported to the service control manager, and an
    50  // error if the state change request was not accepted.
    51  // Note that the returned service status is only set if the status change
    52  // request succeeded, or if it failed with error ERROR_INVALID_SERVICE_CONTROL,
    53  // ERROR_SERVICE_CANNOT_ACCEPT_CTRL, or ERROR_SERVICE_NOT_ACTIVE.
    54  func (s *Service) Control(c svc.Cmd) (svc.Status, error) {
    55  	var t windows.SERVICE_STATUS
    56  	err := windows.ControlService(s.Handle, uint32(c), &t)
    57  	if err != nil &&
    58  		err != windows.ERROR_INVALID_SERVICE_CONTROL &&
    59  		err != windows.ERROR_SERVICE_CANNOT_ACCEPT_CTRL &&
    60  		err != windows.ERROR_SERVICE_NOT_ACTIVE {
    61  		return svc.Status{}, err
    62  	}
    63  	return svc.Status{
    64  		State:   svc.State(t.CurrentState),
    65  		Accepts: svc.Accepted(t.ControlsAccepted),
    66  	}, err
    67  }
    68  
    69  // Query returns current status of service s.
    70  func (s *Service) Query() (svc.Status, error) {
    71  	var t windows.SERVICE_STATUS_PROCESS
    72  	var needed uint32
    73  	err := windows.QueryServiceStatusEx(s.Handle, windows.SC_STATUS_PROCESS_INFO, (*byte)(unsafe.Pointer(&t)), uint32(unsafe.Sizeof(t)), &needed)
    74  	if err != nil {
    75  		return svc.Status{}, err
    76  	}
    77  	return svc.Status{
    78  		State:                   svc.State(t.CurrentState),
    79  		Accepts:                 svc.Accepted(t.ControlsAccepted),
    80  		ProcessId:               t.ProcessId,
    81  		Win32ExitCode:           t.Win32ExitCode,
    82  		ServiceSpecificExitCode: t.ServiceSpecificExitCode,
    83  	}, nil
    84  }
    85  
    86  // ListDependentServices returns the names of the services dependent on service s, which match the given status.
    87  func (s *Service) ListDependentServices(status svc.ActivityStatus) ([]string, error) {
    88  	var bytesNeeded, returnedServiceCount uint32
    89  	var services []windows.ENUM_SERVICE_STATUS
    90  	for {
    91  		var servicesPtr *windows.ENUM_SERVICE_STATUS
    92  		if len(services) > 0 {
    93  			servicesPtr = &services[0]
    94  		}
    95  		allocatedBytes := uint32(len(services)) * uint32(unsafe.Sizeof(windows.ENUM_SERVICE_STATUS{}))
    96  		err := windows.EnumDependentServices(s.Handle, uint32(status), servicesPtr, allocatedBytes, &bytesNeeded,
    97  			&returnedServiceCount)
    98  		if err == nil {
    99  			break
   100  		}
   101  		if err != syscall.ERROR_MORE_DATA {
   102  			return nil, err
   103  		}
   104  		if bytesNeeded <= allocatedBytes {
   105  			return nil, err
   106  		}
   107  		// ERROR_MORE_DATA indicates the provided buffer was too small, run the call again after resizing the buffer
   108  		requiredSliceLen := bytesNeeded / uint32(unsafe.Sizeof(windows.ENUM_SERVICE_STATUS{}))
   109  		if bytesNeeded%uint32(unsafe.Sizeof(windows.ENUM_SERVICE_STATUS{})) != 0 {
   110  			requiredSliceLen += 1
   111  		}
   112  		services = make([]windows.ENUM_SERVICE_STATUS, requiredSliceLen)
   113  	}
   114  	if returnedServiceCount == 0 {
   115  		return nil, nil
   116  	}
   117  
   118  	// The slice mutated by EnumDependentServices may have a length greater than returnedServiceCount, any elements
   119  	// past that should be ignored.
   120  	var dependents []string
   121  	for i := 0; i < int(returnedServiceCount); i++ {
   122  		dependents = append(dependents, windows.UTF16PtrToString(services[i].ServiceName))
   123  	}
   124  	return dependents, nil
   125  }