mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-05 23:07:44 +00:00
1db46919ab
Due to a bug in Go (golang/go#51778), cmd/go doesn't warn about your Go version being older than the go.mod's declared Go version in that case that package loading fails before the build starts, such as when you use packages that are only in the current version of Go, like our use of net/netip. This change works around that Go bug by adding build tags and a pre-Go1.18-only file that will cause Go 1.17 and earlier to fail like: $ ~/sdk/go1.17/bin/go install ./cmd/tailscaled # tailscale.com/cmd/tailscaled ./required_version.go:11:2: undefined: you_need_Go_1_18_to_compile_Tailscale note: module requires Go 1.18 Change-Id: I39f5820de646703e19dde448dd86a7022252f75c Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
127 lines
3.2 KiB
Go
127 lines
3.2 KiB
Go
// Copyright (c) 2021 Tailscale Inc & AUTHORS All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
//go:build go1.18
|
|
// +build go1.18
|
|
|
|
package main
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"time"
|
|
|
|
"golang.org/x/sys/windows"
|
|
"golang.org/x/sys/windows/svc"
|
|
"golang.org/x/sys/windows/svc/mgr"
|
|
"tailscale.com/logtail/backoff"
|
|
"tailscale.com/types/logger"
|
|
"tailscale.com/util/osshare"
|
|
)
|
|
|
|
func init() {
|
|
installSystemDaemon = installSystemDaemonWindows
|
|
uninstallSystemDaemon = uninstallSystemDaemonWindows
|
|
}
|
|
|
|
func installSystemDaemonWindows(args []string) (err error) {
|
|
m, err := mgr.Connect()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to connect to Windows service manager: %v", err)
|
|
}
|
|
|
|
service, err := m.OpenService(serviceName)
|
|
if err == nil {
|
|
service.Close()
|
|
return fmt.Errorf("service %q is already installed", serviceName)
|
|
}
|
|
|
|
// no such service; proceed to install the service.
|
|
|
|
exe, err := os.Executable()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c := mgr.Config{
|
|
ServiceType: windows.SERVICE_WIN32_OWN_PROCESS,
|
|
StartType: mgr.StartAutomatic,
|
|
ErrorControl: mgr.ErrorNormal,
|
|
DisplayName: serviceName,
|
|
Description: "Connects this computer to others on the Tailscale network.",
|
|
}
|
|
|
|
service, err = m.CreateService(serviceName, exe, c)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create %q service: %v", serviceName, err)
|
|
}
|
|
defer service.Close()
|
|
|
|
// Exponential backoff is often too aggressive, so use (mostly)
|
|
// squares instead.
|
|
ra := []mgr.RecoveryAction{
|
|
{mgr.ServiceRestart, 1 * time.Second},
|
|
{mgr.ServiceRestart, 2 * time.Second},
|
|
{mgr.ServiceRestart, 4 * time.Second},
|
|
{mgr.ServiceRestart, 9 * time.Second},
|
|
{mgr.ServiceRestart, 16 * time.Second},
|
|
{mgr.ServiceRestart, 25 * time.Second},
|
|
{mgr.ServiceRestart, 36 * time.Second},
|
|
{mgr.ServiceRestart, 49 * time.Second},
|
|
{mgr.ServiceRestart, 64 * time.Second},
|
|
}
|
|
const resetPeriodSecs = 60
|
|
err = service.SetRecoveryActions(ra, resetPeriodSecs)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to set service recovery actions: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func uninstallSystemDaemonWindows(args []string) (ret error) {
|
|
// Remove file sharing from Windows shell (noop in non-windows)
|
|
osshare.SetFileSharingEnabled(false, logger.Discard)
|
|
|
|
m, err := mgr.Connect()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to connect to Windows service manager: %v", err)
|
|
}
|
|
defer m.Disconnect()
|
|
|
|
service, err := m.OpenService(serviceName)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open %q service: %v", serviceName, err)
|
|
}
|
|
|
|
st, err := service.Query()
|
|
if err != nil {
|
|
service.Close()
|
|
return fmt.Errorf("failed to query service state: %v", err)
|
|
}
|
|
if st.State != svc.Stopped {
|
|
service.Control(svc.Stop)
|
|
}
|
|
err = service.Delete()
|
|
service.Close()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to delete service: %v", err)
|
|
}
|
|
|
|
bo := backoff.NewBackoff("uninstall", logger.Discard, 30*time.Second)
|
|
end := time.Now().Add(15 * time.Second)
|
|
for time.Until(end) > 0 {
|
|
service, err = m.OpenService(serviceName)
|
|
if err != nil {
|
|
// service is no longer openable; success!
|
|
break
|
|
}
|
|
service.Close()
|
|
bo.BackOff(context.Background(), errors.New("service not deleted"))
|
|
}
|
|
return nil
|
|
}
|