2017-07-13 00:24:09 +00:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
2017-08-11 16:59:57 +00:00
|
|
|
"regexp"
|
2017-07-13 00:24:09 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/aquasecurity/kube-bench/check"
|
|
|
|
"github.com/fatih/color"
|
2017-07-25 00:34:07 +00:00
|
|
|
"github.com/golang/glog"
|
2017-07-13 00:24:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// Print colors
|
|
|
|
colors = map[check.State]*color.Color{
|
|
|
|
check.PASS: color.New(color.FgGreen),
|
|
|
|
check.FAIL: color.New(color.FgRed),
|
|
|
|
check.WARN: color.New(color.FgYellow),
|
|
|
|
check.INFO: color.New(color.FgBlue),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2017-08-06 16:59:03 +00:00
|
|
|
func printlnWarn(msg string) {
|
2017-07-25 00:34:07 +00:00
|
|
|
fmt.Fprintf(os.Stderr, "[%s] %s\n",
|
|
|
|
colors[check.WARN].Sprintf("%s", check.WARN),
|
|
|
|
msg,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2017-08-06 16:59:03 +00:00
|
|
|
func sprintlnWarn(msg string) string {
|
2017-07-25 00:34:07 +00:00
|
|
|
return fmt.Sprintf("[%s] %s",
|
|
|
|
colors[check.WARN].Sprintf("%s", check.WARN),
|
|
|
|
msg,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func exitWithError(err error) {
|
|
|
|
fmt.Fprintf(os.Stderr, "\n%v\n", err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func continueWithError(err error, msg string) string {
|
2017-07-13 00:24:09 +00:00
|
|
|
if err != nil {
|
2017-07-25 00:34:07 +00:00
|
|
|
glog.V(1).Info(err)
|
2017-07-13 00:24:09 +00:00
|
|
|
}
|
2017-07-25 00:34:07 +00:00
|
|
|
|
|
|
|
if msg != "" {
|
|
|
|
fmt.Fprintf(os.Stderr, "%s\n", msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
2017-07-13 00:24:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func cleanIDs(list string) []string {
|
|
|
|
list = strings.Trim(list, ",")
|
|
|
|
ids := strings.Split(list, ",")
|
|
|
|
|
|
|
|
for _, id := range ids {
|
|
|
|
id = strings.Trim(id, " ")
|
|
|
|
}
|
|
|
|
|
|
|
|
return ids
|
|
|
|
}
|
|
|
|
|
2017-08-15 15:44:40 +00:00
|
|
|
// ps execs out to the ps command; it's separated into a function so we can write tests
|
|
|
|
func ps(proc string) string {
|
|
|
|
cmd := exec.Command("ps", "-C", proc, "-o", "cmd", "--no-headers")
|
2017-07-13 00:24:09 +00:00
|
|
|
out, err := cmd.Output()
|
2017-07-25 00:34:07 +00:00
|
|
|
if err != nil {
|
|
|
|
continueWithError(fmt.Errorf("%s: %s", cmd.Args, err), "")
|
|
|
|
}
|
2017-07-13 00:24:09 +00:00
|
|
|
|
2017-08-15 15:44:40 +00:00
|
|
|
return string(out)
|
|
|
|
}
|
2017-07-13 00:24:09 +00:00
|
|
|
|
2017-08-15 15:44:40 +00:00
|
|
|
// verifyBin checks that the binary specified is running
|
|
|
|
func verifyBin(bin string, psFunc func(string) string) bool {
|
2017-07-13 00:24:09 +00:00
|
|
|
|
2017-08-15 15:44:40 +00:00
|
|
|
// Strip any quotes
|
|
|
|
bin = strings.Trim(bin, "'\"")
|
|
|
|
|
|
|
|
// bin could consist of more than one word
|
|
|
|
// We'll search for running processes with the first word, and then check the whole
|
|
|
|
// proc as supplied is included in the results
|
|
|
|
proc := strings.Fields(bin)[0]
|
|
|
|
out := psFunc(proc)
|
|
|
|
|
2017-08-30 16:48:12 +00:00
|
|
|
if !strings.Contains(out, bin) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure we're not just matching on a partial word (e.g. if we're looking for apiserver, don't match on kube-apiserver)
|
|
|
|
// This will give a false positive for matching "one two" against "zero one two-x" but it will do for now
|
|
|
|
for _, f := range strings.Fields(out) {
|
|
|
|
if f == proc {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
2017-07-13 00:24:09 +00:00
|
|
|
}
|
|
|
|
|
2017-08-30 11:07:46 +00:00
|
|
|
// findExecutable looks through a list of possible executable names and finds the first one that's running
|
|
|
|
func findExecutable(candidates []string, psFunc func(string) string) (string, error) {
|
|
|
|
for _, c := range candidates {
|
|
|
|
if verifyBin(c, psFunc) {
|
|
|
|
return c, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", fmt.Errorf("no candidates running")
|
|
|
|
}
|
|
|
|
|
2017-08-11 16:59:57 +00:00
|
|
|
func verifyKubeVersion(major string, minor string) {
|
2017-07-13 00:24:09 +00:00
|
|
|
// These executables might not be on the user's path.
|
|
|
|
|
2017-08-11 16:59:57 +00:00
|
|
|
_, err := exec.LookPath("kubectl")
|
2017-07-25 00:34:07 +00:00
|
|
|
if err != nil {
|
2017-08-06 16:59:03 +00:00
|
|
|
continueWithError(err, sprintlnWarn("Kubernetes version check skipped"))
|
2017-07-25 00:34:07 +00:00
|
|
|
return
|
|
|
|
}
|
2017-07-13 01:01:18 +00:00
|
|
|
|
2017-08-11 16:59:57 +00:00
|
|
|
cmd := exec.Command("kubectl", "version")
|
2017-07-13 00:24:09 +00:00
|
|
|
out, err := cmd.Output()
|
2017-07-25 00:34:07 +00:00
|
|
|
if err != nil {
|
2017-08-11 16:59:57 +00:00
|
|
|
s := fmt.Sprintf("Kubernetes version check skipped with error %v", err)
|
|
|
|
continueWithError(err, sprintlnWarn(s))
|
2017-08-15 15:44:40 +00:00
|
|
|
if len(out) == 0 {
|
|
|
|
return
|
|
|
|
}
|
2017-07-25 00:34:07 +00:00
|
|
|
}
|
2017-07-13 00:24:09 +00:00
|
|
|
|
2017-08-11 16:59:57 +00:00
|
|
|
msg := checkVersion("Client", string(out), major, minor)
|
|
|
|
if msg != "" {
|
|
|
|
continueWithError(fmt.Errorf(msg), msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
msg = checkVersion("Server", string(out), major, minor)
|
|
|
|
if msg != "" {
|
|
|
|
continueWithError(fmt.Errorf(msg), msg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var regexVersionMajor = regexp.MustCompile("Major:\"([0-9]+)\"")
|
|
|
|
var regexVersionMinor = regexp.MustCompile("Minor:\"([0-9]+)\"")
|
|
|
|
|
|
|
|
func checkVersion(x string, s string, expMajor string, expMinor string) string {
|
|
|
|
regexVersion, err := regexp.Compile(x + " Version: version.Info{(.*)}")
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Sprintf("Error checking Kubernetes version: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ss := regexVersion.FindString(s)
|
|
|
|
major := versionMatch(regexVersionMajor, ss)
|
|
|
|
minor := versionMatch(regexVersionMinor, ss)
|
|
|
|
if major == "" || minor == "" {
|
|
|
|
return fmt.Sprintf("Couldn't find %s version from kubectl output '%s'", x, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
if major != expMajor || minor != expMinor {
|
|
|
|
return fmt.Sprintf("Unexpected %s version %s.%s", x, major, minor)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func versionMatch(r *regexp.Regexp, s string) string {
|
|
|
|
match := r.FindStringSubmatch(s)
|
|
|
|
if len(match) < 2 {
|
|
|
|
return ""
|
2017-07-13 00:24:09 +00:00
|
|
|
}
|
2017-08-11 16:59:57 +00:00
|
|
|
return match[1]
|
2017-07-13 00:24:09 +00:00
|
|
|
}
|
2017-08-15 16:00:35 +00:00
|
|
|
|
|
|
|
func multiWordReplace(s string, subname string, sub string) string {
|
|
|
|
f := strings.Fields(sub)
|
|
|
|
if len(f) > 1 {
|
|
|
|
sub = "'" + sub + "'"
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Replace(s, subname, sub, -1)
|
|
|
|
}
|