fleet/orbit/pkg/table/dscl/dscl_darwin.go
Lucas Manuel Rodriguez 2e199dcdab
Fix golangci-lint issue and run Github action on all OSs (#9944)
We have code that builds conditionally depending on the platform (mostly
Orbit code) so we should run `golangci-lint` checks on all OSs.

This adds it to run on macOS, for Windows see:
https://github.com/fleetdm/fleet/issues/9943
2023-02-21 14:30:45 -03:00

113 lines
2.8 KiB
Go

//go:build darwin
// +build darwin
// Package dscl allows querying dscl read commands on the local domain.
package dscl
import (
"context"
"errors"
"fmt"
"os/exec"
"regexp"
"strings"
"github.com/osquery/osquery-go/plugin/table"
)
// Columns is the schema of the table.
func Columns() []table.ColumnDefinition {
return []table.ColumnDefinition{
table.TextColumn("command"), // required (currently only read is supported)
table.TextColumn("path"), // required
table.TextColumn("key"), // required (could be relaxed in the future)
table.TextColumn("value"),
}
}
// Generate is called to return the results for the table at query time.
//
// Constraints for generating can be retrieved from the queryContext.
func Generate(ctx context.Context, queryContext table.QueryContext) ([]map[string]string, error) {
supportedCommands := []string{"read"}
getArgumentOpEqual := func(argName string) string {
argValue := ""
if constraints, ok := queryContext.Constraints[argName]; ok {
for _, constraint := range constraints.Constraints {
if constraint.Operator == table.OperatorEquals {
argValue = constraint.Expression
}
}
}
return argValue
}
command := getArgumentOpEqual("command")
if command == "" {
return nil, fmt.Errorf("missing command argument, supported commands: %+v", supportedCommands)
}
supported := false
for _, supportedCommand := range supportedCommands {
if supportedCommand == command {
supported = true
break
}
}
if !supported {
return nil, fmt.Errorf("unsupported command: %s, supported commands: %+v", command, supportedCommands)
}
path := getArgumentOpEqual("path")
if path == "" {
return nil, errors.New("missing path argument")
}
key := getArgumentOpEqual("key")
if key == "" {
// In the future we can allow this to be empty and return all key/values of a path.
return nil, errors.New("missing key argument")
}
cmd := exec.Command("/usr/bin/dscl", ".", command, path, key)
out, err := cmd.Output()
if err != nil {
return nil, fmt.Errorf("command failed: %w", err)
}
value, err := parseDSCLReadOutput(out)
if err != nil {
return nil, fmt.Errorf("failed to parse dscl value: %w", err)
}
m := []map[string]string{{
"command": command,
"path": path,
"key": key,
"value": "",
}}
if value != nil {
m[0]["value"] = *value
}
return m, nil
}
func parseDSCLReadOutput(out []byte) (*string, error) {
regex := regexp.MustCompile(`(\S):[ \n]([\S\t\f\r\n ]+)`)
outs := string(out)
if strings.TrimSpace(outs) == "" || strings.HasPrefix(outs, "No such key: ") {
return nil, nil
}
matches := regex.FindSubmatch(out)
if matches == nil {
return nil, fmt.Errorf("unexpected entry: %q", outs)
}
value := string(matches[2])
if value[0] == ' ' {
value = value[1:]
}
return &value, nil
}