mirror of
https://github.com/golang/sys.git
synced 2026-02-09 04:06:04 +03:00
Currently, doing s.UpdateConfig(s.Config()) will destroy the service, because s.Config() fails to populate the SidType member, but UpdateConfig will set the SidType, so that expression effectively zeros out the SidType. Fix this by having Config() fetch the SidType in the same way that it fetches the other additional fields there, such as DelayedStartUp. Change-Id: Idb917ef1e942020499b411b7777b995c29f0e7d2 Reviewed-on: https://go-review.googlesource.com/c/sys/+/270897 Trust: Jason A. Donenfeld <Jason@zx2c4.com> Run-TryBot: Jason A. Donenfeld <Jason@zx2c4.com> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
181 lines
5.3 KiB
Go
181 lines
5.3 KiB
Go
// Copyright 2012 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// +build windows
|
|
|
|
package mgr
|
|
|
|
import (
|
|
"syscall"
|
|
"unsafe"
|
|
|
|
"golang.org/x/sys/windows"
|
|
)
|
|
|
|
const (
|
|
// Service start types.
|
|
StartManual = windows.SERVICE_DEMAND_START // the service must be started manually
|
|
StartAutomatic = windows.SERVICE_AUTO_START // the service will start by itself whenever the computer reboots
|
|
StartDisabled = windows.SERVICE_DISABLED // the service cannot be started
|
|
|
|
// The severity of the error, and action taken,
|
|
// if this service fails to start.
|
|
ErrorCritical = windows.SERVICE_ERROR_CRITICAL
|
|
ErrorIgnore = windows.SERVICE_ERROR_IGNORE
|
|
ErrorNormal = windows.SERVICE_ERROR_NORMAL
|
|
ErrorSevere = windows.SERVICE_ERROR_SEVERE
|
|
)
|
|
|
|
// TODO(brainman): Password is not returned by windows.QueryServiceConfig, not sure how to get it.
|
|
|
|
type Config struct {
|
|
ServiceType uint32
|
|
StartType uint32
|
|
ErrorControl uint32
|
|
BinaryPathName string // fully qualified path to the service binary file, can also include arguments for an auto-start service
|
|
LoadOrderGroup string
|
|
TagId uint32
|
|
Dependencies []string
|
|
ServiceStartName string // name of the account under which the service should run
|
|
DisplayName string
|
|
Password string
|
|
Description string
|
|
SidType uint32 // one of SERVICE_SID_TYPE, the type of sid to use for the service
|
|
DelayedAutoStart bool // the service is started after other auto-start services are started plus a short delay
|
|
}
|
|
|
|
func toStringSlice(ps *uint16) []string {
|
|
r := make([]string, 0)
|
|
p := unsafe.Pointer(ps)
|
|
|
|
for {
|
|
s := windows.UTF16PtrToString((*uint16)(p))
|
|
if len(s) == 0 {
|
|
break
|
|
}
|
|
|
|
r = append(r, s)
|
|
offset := unsafe.Sizeof(uint16(0)) * (uintptr)(len(s)+1)
|
|
p = unsafe.Pointer(uintptr(p) + offset)
|
|
}
|
|
|
|
return r
|
|
}
|
|
|
|
// Config retrieves service s configuration paramteres.
|
|
func (s *Service) Config() (Config, error) {
|
|
var p *windows.QUERY_SERVICE_CONFIG
|
|
n := uint32(1024)
|
|
for {
|
|
b := make([]byte, n)
|
|
p = (*windows.QUERY_SERVICE_CONFIG)(unsafe.Pointer(&b[0]))
|
|
err := windows.QueryServiceConfig(s.Handle, p, n, &n)
|
|
if err == nil {
|
|
break
|
|
}
|
|
if err.(syscall.Errno) != syscall.ERROR_INSUFFICIENT_BUFFER {
|
|
return Config{}, err
|
|
}
|
|
if n <= uint32(len(b)) {
|
|
return Config{}, err
|
|
}
|
|
}
|
|
|
|
b, err := s.queryServiceConfig2(windows.SERVICE_CONFIG_DESCRIPTION)
|
|
if err != nil {
|
|
return Config{}, err
|
|
}
|
|
p2 := (*windows.SERVICE_DESCRIPTION)(unsafe.Pointer(&b[0]))
|
|
|
|
b, err = s.queryServiceConfig2(windows.SERVICE_CONFIG_DELAYED_AUTO_START_INFO)
|
|
if err != nil {
|
|
return Config{}, err
|
|
}
|
|
p3 := (*windows.SERVICE_DELAYED_AUTO_START_INFO)(unsafe.Pointer(&b[0]))
|
|
delayedStart := false
|
|
if p3.IsDelayedAutoStartUp != 0 {
|
|
delayedStart = true
|
|
}
|
|
|
|
b, err = s.queryServiceConfig2(windows.SERVICE_CONFIG_SERVICE_SID_INFO)
|
|
if err != nil {
|
|
return Config{}, err
|
|
}
|
|
sidType := *(*uint32)(unsafe.Pointer(&b[0]))
|
|
|
|
return Config{
|
|
ServiceType: p.ServiceType,
|
|
StartType: p.StartType,
|
|
ErrorControl: p.ErrorControl,
|
|
BinaryPathName: windows.UTF16PtrToString(p.BinaryPathName),
|
|
LoadOrderGroup: windows.UTF16PtrToString(p.LoadOrderGroup),
|
|
TagId: p.TagId,
|
|
Dependencies: toStringSlice(p.Dependencies),
|
|
ServiceStartName: windows.UTF16PtrToString(p.ServiceStartName),
|
|
DisplayName: windows.UTF16PtrToString(p.DisplayName),
|
|
Description: windows.UTF16PtrToString(p2.Description),
|
|
DelayedAutoStart: delayedStart,
|
|
SidType: sidType,
|
|
}, nil
|
|
}
|
|
|
|
func updateDescription(handle windows.Handle, desc string) error {
|
|
d := windows.SERVICE_DESCRIPTION{Description: toPtr(desc)}
|
|
return windows.ChangeServiceConfig2(handle,
|
|
windows.SERVICE_CONFIG_DESCRIPTION, (*byte)(unsafe.Pointer(&d)))
|
|
}
|
|
|
|
func updateSidType(handle windows.Handle, sidType uint32) error {
|
|
return windows.ChangeServiceConfig2(handle, windows.SERVICE_CONFIG_SERVICE_SID_INFO, (*byte)(unsafe.Pointer(&sidType)))
|
|
}
|
|
|
|
func updateStartUp(handle windows.Handle, isDelayed bool) error {
|
|
var d windows.SERVICE_DELAYED_AUTO_START_INFO
|
|
if isDelayed {
|
|
d.IsDelayedAutoStartUp = 1
|
|
}
|
|
return windows.ChangeServiceConfig2(handle,
|
|
windows.SERVICE_CONFIG_DELAYED_AUTO_START_INFO, (*byte)(unsafe.Pointer(&d)))
|
|
}
|
|
|
|
// UpdateConfig updates service s configuration parameters.
|
|
func (s *Service) UpdateConfig(c Config) error {
|
|
err := windows.ChangeServiceConfig(s.Handle, c.ServiceType, c.StartType,
|
|
c.ErrorControl, toPtr(c.BinaryPathName), toPtr(c.LoadOrderGroup),
|
|
nil, toStringBlock(c.Dependencies), toPtr(c.ServiceStartName),
|
|
toPtr(c.Password), toPtr(c.DisplayName))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = updateSidType(s.Handle, c.SidType)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = updateStartUp(s.Handle, c.DelayedAutoStart)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return updateDescription(s.Handle, c.Description)
|
|
}
|
|
|
|
// queryServiceConfig2 calls Windows QueryServiceConfig2 with infoLevel parameter and returns retrieved service configuration information.
|
|
func (s *Service) queryServiceConfig2(infoLevel uint32) ([]byte, error) {
|
|
n := uint32(1024)
|
|
for {
|
|
b := make([]byte, n)
|
|
err := windows.QueryServiceConfig2(s.Handle, infoLevel, &b[0], n, &n)
|
|
if err == nil {
|
|
return b, nil
|
|
}
|
|
if err.(syscall.Errno) != syscall.ERROR_INSUFFICIENT_BUFFER {
|
|
return nil, err
|
|
}
|
|
if n <= uint32(len(b)) {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|