Skip to content

Commit

Permalink
generate: translate genpolicy logs, show warnings
Browse files Browse the repository at this point in the history
Signed-off-by: Paul Meyer <[email protected]>
  • Loading branch information
katexochen committed Jun 26, 2024
1 parent 2b7c9e4 commit 3964914
Show file tree
Hide file tree
Showing 2 changed files with 140 additions and 4 deletions.
70 changes: 66 additions & 4 deletions cli/cmd/generate.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
package cmd

import (
"bufio"
"bytes"
"context"
"crypto/ecdsa"
Expand All @@ -16,10 +17,12 @@ import (
"encoding/pem"
"errors"
"fmt"
"io"
"log/slog"
"os"
"os/exec"
"path/filepath"
"regexp"
"slices"
"strings"

Expand Down Expand Up @@ -388,6 +391,60 @@ func addWorkloadOwnerKeyToManifest(manifst *manifest.Manifest, keyPath string) e
return nil
}

type logTranslator struct {
r *io.PipeReader
w *io.PipeWriter
logger *slog.Logger
stopDoneC chan struct{}
}

func newLogTranslator(logger *slog.Logger) logTranslator {
r, w := io.Pipe()
l := logTranslator{
r: r,
w: w,
logger: logger,
stopDoneC: make(chan struct{}),
}
l.startTranslate()
return l
}

func (l logTranslator) Write(p []byte) (n int, err error) {
return l.w.Write(p)
}

var genpolicyLogPrefixReg = regexp.MustCompile(`^\[[^\]\s]+\s+(\w+)\s+([^\]\s]+)\] (.*)`)

func (l logTranslator) startTranslate() {
go func() {
defer close(l.stopDoneC)
scanner := bufio.NewScanner(l.r)
for scanner.Scan() {
line := scanner.Text()
match := genpolicyLogPrefixReg.FindStringSubmatch(line)
if len(match) != 4 {
// genpolicy prints some warnings without the logger
l.logger.Warn(line)
} else {
switch match[1] {
case "ERROR":
l.logger.Error(match[3], "position", match[2])
case "WARN":
l.logger.Warn(match[3], "position", match[2])
case "INFO": // prints quite a lot, only show on debug
l.logger.Debug(match[3], "position", match[2])
}
}
}
}()
}

func (l logTranslator) stop() {
l.w.Close()
<-l.stopDoneC
}

func generatePolicyForFile(ctx context.Context, genpolicyPath, regoPath, policyPath, yamlPath string, logger *slog.Logger) ([32]byte, error) {
args := []string{
"--raw-out",
Expand All @@ -399,22 +456,27 @@ func generatePolicyForFile(ctx context.Context, genpolicyPath, regoPath, policyP
}
genpolicy := exec.CommandContext(ctx, genpolicyPath, args...)
genpolicy.Env = append(genpolicy.Env, "RUST_LOG=info")
var stdout, stderr bytes.Buffer

logFilter := newLogTranslator(logger)
defer logFilter.stop()
var stdout bytes.Buffer
genpolicy.Stdout = &stdout
genpolicy.Stderr = &stderr
genpolicy.Stderr = logFilter

if err := genpolicy.Run(); err != nil {
var exitErr *exec.ExitError
if errors.As(err, &exitErr) {
return [32]byte{}, fmt.Errorf("genpolicy failed with exit code %d: %s",
exitErr.ExitCode(), stderr.String())
return [32]byte{}, fmt.Errorf("genpolicy failed with exit code %d", exitErr.ExitCode())
}
return [32]byte{}, fmt.Errorf("genpolicy failed: %w", err)
}

if stdout.Len() == 0 {
logger.Info("Policy output is empty, ignoring the file", "yamlPath", yamlPath)
return [32]byte{}, nil
}
policyHash := sha256.Sum256(stdout.Bytes())

return policyHash, nil
}

Expand Down
74 changes: 74 additions & 0 deletions cli/cmd/generate_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
package cmd

import (
"testing"

"github.com/stretchr/testify/assert"
)

func TestGenpolicyLogPrefixReg(t *testing.T) {
testCases := []struct {
logLine string
wantMatch bool
wantLevel string
wantPosition string
wantMessage string
}{
{
logLine: `[2024-06-26T09:09:40Z INFO genpolicy::registry] ============================================`,
wantMatch: true,
wantLevel: "INFO",
wantPosition: "genpolicy::registry",
wantMessage: "============================================",
},
{
logLine: `[2024-06-26T09:09:40Z INFO genpolicy::registry] Pulling manifest and config for "mcr.microsoft.com/oss/kubernetes/pause:3.6"`,
wantMatch: true,
wantLevel: "INFO",
wantPosition: "genpolicy::registry",
wantMessage: `Pulling manifest and config for "mcr.microsoft.com/oss/kubernetes/pause:3.6"`,
},
{
logLine: `[2024-06-26T09:09:41Z INFO genpolicy::registry] Using cache file`,
wantMatch: true,
wantLevel: "INFO",
wantPosition: "genpolicy::registry",
wantMessage: "Using cache file",
},
{
logLine: `[2024-06-26T09:09:41Z INFO genpolicy::registry] dm-verity root hash: 1e306eb31693964ac837ac74bc57b50c87c549f58b0da2789e3915f923f21b81`,
wantMatch: true,
wantLevel: "INFO",
wantPosition: "genpolicy::registry",
wantMessage: "dm-verity root hash: 1e306eb31693964ac837ac74bc57b50c87c549f58b0da2789e3915f923f21b81",
},
{
logLine: `[2024-06-26T09:09:44Z WARN genpolicy::registry] Using cache file`,
wantMatch: true,
wantLevel: "WARN",
wantPosition: "genpolicy::registry",
wantMessage: "Using cache file",
},
{
logLine: `Success!"`,
wantMatch: false,
},
}

for _, tc := range testCases {
t.Run("", func(t *testing.T) {
assert := assert.New(t)

match := genpolicyLogPrefixReg.FindStringSubmatch(tc.logLine)

if !tc.wantMatch {
assert.Nil(match)
return
}
assert.Len(match, 4)
assert.Equal(tc.wantLevel, match[1])
assert.Equal(tc.wantPosition, match[2])
assert.Equal(tc.wantMessage, match[3])
})
}
}

0 comments on commit 3964914

Please sign in to comment.