-
Notifications
You must be signed in to change notification settings - Fork 65
/
exec.go
172 lines (144 loc) · 5 KB
/
exec.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
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
/*
Copyright The Ratify Authors.
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.
*/
package plugin
import (
"bufio"
"bytes"
"context"
"encoding/json"
"fmt"
"io"
"os"
"os/exec"
"strings"
"time"
"github.com/ratify-project/ratify/internal/logger"
"github.com/sirupsen/logrus"
)
const (
maxRetryCount = 5
waitDuration = time.Second
)
var logOpt = logger.Option{
ComponentType: logger.Plugin,
}
// Executor is an interface that defines methods to lookup a plugin and execute it.
type Executor interface {
// ExecutePlugin executes the plugin with the given parameters
ExecutePlugin(ctx context.Context, pluginPath string, cmdArgs []string, stdinData []byte, environ []string) ([]byte, error)
// FindInPaths finds the plugin in the given paths
FindInPaths(plugin string, paths []string) (string, error)
}
// DefaultExecutor finds the plugin executable and invokes it as a os command
type DefaultExecutor struct {
Stderr io.Writer
}
// return the command output and the error
func (e *DefaultExecutor) ExecutePlugin(ctx context.Context, pluginPath string, cmdArgs []string, stdinData []byte, environ []string) ([]byte, error) {
stdout := &bytes.Buffer{}
stderr := &bytes.Buffer{}
c := exec.CommandContext(ctx, pluginPath, cmdArgs...)
c.Env = environ
c.Stdin = bytes.NewBuffer(stdinData)
c.Stdout = stdout
c.Stderr = stderr
// DEBUG: log the process details used to launch the binary plugin
if logrus.IsLevelEnabled(logrus.DebugLevel) {
logrus.Debugf("launching plugin %s", pluginPath)
pluginEnv := make([]string, 3)
for _, env := range environ {
// plugins inherit all env vars, but we're interested in the RATIFY_* ones. This also helps to keep secret values out of the logs.
if strings.HasPrefix(env, "RATIFY_") {
pluginEnv = append(pluginEnv, env)
}
}
logrus.Debugf("env vars: %v", pluginEnv)
logrus.Debugf("args: %v", cmdArgs)
logrus.Debugf("stdin: %s", stdinData)
}
// Retry the command on "text file busy" errors
for i := 0; i <= maxRetryCount; i++ {
err := c.Run()
// Command succeeded
if err == nil {
break
}
// If the plugin is about to be completed, then we wait a
// second and try it again
if strings.Contains(err.Error(), "text file busy") {
logrus.Debugf("command returned text file busy, retrying after %v", waitDuration)
time.Sleep(waitDuration)
continue
}
// For all other errors return failed.
return nil, e.pluginErr(err, stdout.Bytes(), stderr.Bytes())
}
pluginOutputJSON, pluginOutputMsgs := parsePluginOutput(stdout, stderr)
// Disregards plugin source stream and logs the plugin messages to stderr
for _, msg := range pluginOutputMsgs {
msg := strings.ToLower(msg)
switch {
case strings.HasPrefix(msg, "info"):
msg = strings.Replace(msg, "info: ", "", -1)
logger.GetLogger(ctx, logOpt).Infof("[Plugin] %s", msg)
case strings.HasPrefix(msg, "warn"):
msg = strings.Replace(msg, "warn: ", "", -1)
logger.GetLogger(ctx, logOpt).Warnf("[Plugin] %s", msg)
case strings.HasPrefix(msg, "debug"):
msg = strings.Replace(msg, "debug: ", "", -1)
logger.GetLogger(ctx, logOpt).Debugf("[Plugin] %s", msg)
default:
fmt.Fprintf(os.Stderr, "[Plugin] %s,", msg)
}
}
return pluginOutputJSON, nil
}
func (e *DefaultExecutor) pluginErr(err error, stdout, stderr []byte) error {
var stdOutBuffer bytes.Buffer
var stdErrBuffer bytes.Buffer
// writing them to buffer to avoid lint error
stdOutBuffer.Write(stdout)
stdErrBuffer.Write(stderr)
errCombined := Error{}
errCombined.Msg = fmt.Sprintf("plugin failed with error: '%v', msg from stError '%v', msg from stdOut '%v'", err.Error(), stdErrBuffer.String(), stdOutBuffer.String())
return &errCombined
}
func (e *DefaultExecutor) FindInPaths(plugin string, paths []string) (string, error) {
return FindInPaths(plugin, paths)
}
func parsePluginOutput(stdout *bytes.Buffer, stderr *bytes.Buffer) ([]byte, []string) {
var obj interface{}
var outputBuffer bytes.Buffer
var jsonBuffer bytes.Buffer
var messages []string
// Combine stderr and stdout
outputBuffer.Write(stderr.Bytes())
outputBuffer.Write(stdout.Bytes())
scanner := bufio.NewScanner(&outputBuffer)
for scanner.Scan() {
line := scanner.Text()
// Assumes a single JSON object is returned somewhere in the output
// does not handle multiple JSON objects
if strings.Contains(line, "{") {
err := json.NewDecoder(strings.NewReader(line)).Decode(&obj)
if err != nil {
continue
}
jsonString, _ := json.Marshal(obj)
jsonBuffer.WriteString(string(jsonString))
} else {
messages = append(messages, line)
}
}
return jsonBuffer.Bytes(), messages
}