-
Notifications
You must be signed in to change notification settings - Fork 74
/
Copy pathmain_windows.go
123 lines (106 loc) · 3.39 KB
/
main_windows.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
// Copyright 2022 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//go:build windows
package main
import (
"context"
"flag"
"fmt"
"os"
"github.com/GoogleCloudPlatform/ops-agent/cmd/google_cloud_ops_agent_diagnostics/utils"
"go.opentelemetry.io/otel"
"golang.org/x/sys/windows/svc"
"golang.org/x/sys/windows/svc/debug"
"golang.org/x/sys/windows/svc/eventlog"
)
const (
DiagnosticsEventID uint32 = 2
ERROR_SUCCESS uint32 = 0
ERROR_FILE_NOT_FOUND uint32 = 2
ERROR_INVALID_DATA uint32 = 13
ERROR_INVALID_PARAMETER uint32 = 87
)
type service struct {
ctx context.Context
log debug.Log
userConf string
}
func (s *service) Handle(err error) {
s.log.Error(DiagnosticsEventID, fmt.Sprintf("error collecting metrics: %v", err))
}
func run(ctx context.Context) error {
name := "google-cloud-ops-agent-diagnostics"
elog, err := eventlog.Open(name)
if err != nil {
// probably futile
return err
}
defer elog.Close()
elog.Info(DiagnosticsEventID, fmt.Sprintf("starting %s service", name))
err = svc.Run(name, &service{ctx: ctx, log: elog})
if err != nil {
elog.Error(DiagnosticsEventID, fmt.Sprintf("%s service failed: %v", name, err))
return err
}
elog.Info(DiagnosticsEventID, fmt.Sprintf("%s service stopped", name))
return nil
}
func (s *service) Execute(args []string, r <-chan svc.ChangeRequest, changes chan<- svc.Status) (ssec bool, errno uint32) {
ctx, cancel := context.WithCancel(s.ctx)
defer cancel()
const cmdsAccepted = svc.AcceptStop | svc.AcceptShutdown
changes <- svc.Status{State: svc.StartPending}
allArgs := append([]string{}, os.Args[1:]...)
allArgs = append(allArgs, args[1:]...)
if err := s.parseFlags(allArgs); err != nil {
s.log.Error(DiagnosticsEventID, fmt.Sprintf("failed to parse arguments: %v", err))
return false, ERROR_INVALID_PARAMETER
}
_, _, err := utils.GetUserAndMergedConfigs(ctx, s.userConf)
if err != nil {
s.log.Error(DiagnosticsEventID, fmt.Sprintf("failed to obtain unified configuration: %v", err))
return false, ERROR_FILE_NOT_FOUND
}
s.log.Info(DiagnosticsEventID, "obtained unified configuration")
changes <- svc.Status{State: svc.Running, Accepts: cmdsAccepted}
go func() {
// Manage windows service signals
defer func() {
changes <- svc.Status{State: svc.StopPending}
}()
for {
select {
case c := <-r:
switch c.Cmd {
case svc.Interrogate:
changes <- c.CurrentStatus
case svc.Stop, svc.Shutdown:
cancel()
return
default:
s.log.Error(DiagnosticsEventID, fmt.Sprintf("unexpected control request #%d", c))
}
}
}
}()
// Set otel error handler
otel.SetErrorHandler(s)
return false, ERROR_SUCCESS
}
func (s *service) parseFlags(args []string) error {
s.log.Info(DiagnosticsEventID, fmt.Sprintf("args: %#v", args))
var fs flag.FlagSet
fs.StringVar(&s.userConf, "config", "", "path to the user specified agent config")
return fs.Parse(args)
}