Add mutation for adding digest to image spec

This commit is contained in:
Jonny Stoten
2024-06-20 12:25:04 +01:00
parent 0e3d5b5911
commit 3378c90b3f
8 changed files with 138 additions and 30 deletions

View File

@@ -1,8 +1,17 @@
apiVersion: externaldata.gatekeeper.sh/v1beta1
kind: Provider
metadata:
name: attest-provider
name: attest-provider-validate
spec:
url: https://attest-provider.{{ .Release.Namespace }}:{{ .Values.port }}
url: https://attest-provider.{{ .Release.Namespace }}:{{ .Values.port }}/validate
timeout: {{ .Values.provider.timeout }}
caBundle: {{ required "You must provide .Values.provider.tls.caBundle" .Values.provider.tls.caBundle }}
---
apiVersion: externaldata.gatekeeper.sh/v1beta1
kind: Provider
metadata:
name: attest-provider-mutate
spec:
url: https://attest-provider.{{ .Release.Namespace }}:{{ .Values.port }}/mutate
timeout: {{ .Values.provider.timeout }}
caBundle: {{ required "You must provide .Values.provider.tls.caBundle" .Values.provider.tls.caBundle }}

4
go.mod
View File

@@ -4,6 +4,8 @@ go 1.22.1
require (
github.com/docker/attest v0.1.5-0.20240618150600-86878482c3d9
github.com/google/go-containerregistry v0.19.2
github.com/in-toto/in-toto-golang v0.9.0
github.com/open-policy-agent/frameworks/constraint v0.0.0-20221214024800-b745745c4118
k8s.io/klog/v2 v2.120.1
)
@@ -64,14 +66,12 @@ require (
github.com/gogo/protobuf v1.3.2 // indirect
github.com/golang/snappy v0.0.4 // indirect
github.com/google/certificate-transparency-go v1.1.8 // indirect
github.com/google/go-containerregistry v0.19.2 // indirect
github.com/google/gofuzz v1.2.0 // indirect
github.com/google/uuid v1.6.0 // indirect
github.com/gorilla/mux v1.8.1 // indirect
github.com/hashicorp/go-cleanhttp v0.5.2 // indirect
github.com/hashicorp/go-retryablehttp v0.7.6 // indirect
github.com/hashicorp/hcl v1.0.1-vault-5 // indirect
github.com/in-toto/in-toto-golang v0.9.0 // indirect
github.com/inconshreveable/mousetrap v1.1.0 // indirect
github.com/jedisct1/go-minisign v0.0.0-20230811132847-661be99b8267 // indirect
github.com/jellydator/ttlcache/v3 v3.2.0 // indirect

View File

@@ -46,7 +46,8 @@ func init() {
func main() {
mux := http.NewServeMux()
mux.Handle("POST /", http.TimeoutHandler(handler.Handler(), handlerTimeout, timeoutError))
mux.Handle("POST /validate", http.TimeoutHandler(handler.Validate(), handlerTimeout, timeoutError))
mux.Handle("POST /mutate", http.TimeoutHandler(handler.Mutate(), handlerTimeout, timeoutError))
server := &http.Server{
Addr: fmt.Sprintf(":%d", port),

View File

@@ -1,20 +1,22 @@
apiVersion: mutations.gatekeeper.sh/v1beta1
kind: Assign
metadata:
name: append-valid-suffix-to-image
name: pin-to-digest
spec:
applyTo:
- groups: [""]
kinds: ["Pod"]
versions: ["v1"]
- groups: [""]
kinds: ["Pod"]
versions: ["v1"]
match:
scope: Namespaced
kinds:
- apiGroups: ["*"]
kinds: ["Pod"]
- apiGroups: ["*"]
kinds: ["Pod"]
namespaces: ["test"]
location: "spec.containers[name: *].image"
parameters:
assign:
externalData:
provider: external-data-provider
provider: attest-provider-mutate
dataSource: ValueAtLocation
failurePolicy: Fail

85
pkg/handler/mutate.go Normal file
View File

@@ -0,0 +1,85 @@
package handler
import (
"encoding/json"
"fmt"
"io"
"net/http"
"runtime/debug"
"github.com/docker/attest/pkg/oci"
"github.com/google/go-containerregistry/pkg/name"
"github.com/google/go-containerregistry/pkg/v1/remote"
"github.com/open-policy-agent/frameworks/constraint/pkg/externaldata"
"github.com/open-policy-agent/gatekeeper-external-data-provider/pkg/utils"
"k8s.io/klog/v2"
)
func Mutate() http.Handler {
return http.HandlerFunc(mutate)
}
func mutate(w http.ResponseWriter, req *http.Request) {
defer func() {
if r := recover(); r != nil {
klog.Error(string(debug.Stack()))
klog.ErrorS(fmt.Errorf("%v", r), "panic occurred")
}
}()
// read request body
requestBody, err := io.ReadAll(req.Body)
if err != nil {
utils.SendResponse(nil, fmt.Sprintf("unable to read request body: %v", err), w)
return
}
klog.InfoS("received request", "body", requestBody)
// parse request body
var providerRequest externaldata.ProviderRequest
err = json.Unmarshal(requestBody, &providerRequest)
if err != nil {
utils.SendResponse(nil, fmt.Sprintf("unable to unmarshal request body: %v", err), w)
return
}
results := make([]externaldata.Item, 0)
ctx := req.Context()
opts := oci.WithOptions(ctx, nil)
for _, key := range providerRequest.Request.Keys {
output, err := getReferenceWithDigest(key, opts)
if err != nil {
utils.SendResponse(nil, err.Error(), w)
return
}
results = append(results, externaldata.Item{
Key: key,
Value: output,
})
}
utils.SendResponse(&results, "", w)
}
func getReferenceWithDigest(imageRef string, opts []remote.Option) (string, error) {
ref, err := name.ParseReference(imageRef)
if err != nil {
return "", fmt.Errorf("unable to parse reference %s: %v", imageRef, err)
}
// if it already contains a digest, just return it as is
if _, ok := ref.(name.Digest); ok {
return ref.String(), nil
}
// we need to make a request to the registry to get the digest
desc, err := remote.Head(ref, opts...)
if err != nil {
return "", fmt.Errorf("unable to get digest for reference %s: %v", imageRef, err)
}
return ref.Name() + "@" + desc.Digest.String(), nil
}

17
pkg/handler/tuf.go Normal file
View File

@@ -0,0 +1,17 @@
package handler
import (
"github.com/docker/attest/pkg/tuf"
"github.com/open-policy-agent/gatekeeper-external-data-provider/internal/embed"
)
func createTufClient(outputPath string) (*tuf.TufClient, error) {
// using oci tuf metadata and targets
// metadataURI := "registry-1.docker.io/docker/tuf-metadata:latest"
// targetsURI := "registry-1.docker.io/docker/tuf-targets"
// example using http tuf metadata and targets
metadataURI := "https://docker.github.io/tuf-staging/metadata"
targetsURI := "https://docker.github.io/tuf-staging/targets"
return tuf.NewTufClient(embed.StagingRoot, outputPath, metadataURI, targetsURI, tuf.NewVersionChecker())
}

View File

@@ -11,18 +11,24 @@ import (
"github.com/docker/attest/pkg/attest"
"github.com/docker/attest/pkg/oci"
"github.com/docker/attest/pkg/policy"
"github.com/docker/attest/pkg/tuf"
intoto "github.com/in-toto/in-toto-golang/in_toto"
"github.com/open-policy-agent/frameworks/constraint/pkg/externaldata"
"github.com/open-policy-agent/gatekeeper-external-data-provider/internal/embed"
"github.com/open-policy-agent/gatekeeper-external-data-provider/pkg/utils"
"k8s.io/klog/v2"
)
func Handler() http.Handler {
return http.HandlerFunc(handler)
type ValidationResult struct {
Outcome attest.Outcome `json:"outcome"`
Input *policy.PolicyInput `json:"input"`
VSA *intoto.Statement `json:"vsa"`
Violations []policy.Violation `json:"violations"`
}
func handler(w http.ResponseWriter, req *http.Request) {
func Validate() http.Handler {
return http.HandlerFunc(validate)
}
func validate(w http.ResponseWriter, req *http.Request) {
defer func() {
if r := recover(); r != nil {
klog.Error(string(debug.Stack()))
@@ -56,7 +62,6 @@ func handler(w http.ResponseWriter, req *http.Request) {
utils.SendResponse(nil, err.Error(), w)
}
// iterate over all keys
for _, key := range providerRequest.Request.Keys {
// create a resolver for remote attestations
platform := "linux/amd64"
@@ -103,14 +108,3 @@ func handler(w http.ResponseWriter, req *http.Request) {
}
utils.SendResponse(&results, "", w)
}
func createTufClient(outputPath string) (*tuf.TufClient, error) {
// using oci tuf metadata and targets
metadataURI := "registry-1.docker.io/docker/tuf-metadata:latest"
targetsURI := "registry-1.docker.io/docker/tuf-targets"
// example using http tuf metadata and targets
// metadataURI := "https://docker.github.io/tuf-staging/metadata"
// targetsURI := "https://docker.github.io/tuf-staging/targets"
return tuf.NewTufClient(embed.StagingRoot, outputPath, metadataURI, targetsURI, tuf.NewVersionChecker())
}

View File

@@ -17,7 +17,7 @@ spec:
images := [img | img = input.review.object.spec.containers[_].image]
# send external data request
response := external_data({"provider": "attest-provider", "keys": images})
response := external_data({"provider": "attest-provider-validate", "keys": images})
response_with_error(response)