1
0
Fork 0
mirror of https://github.com/kyverno/kyverno.git synced 2024-12-14 11:57:48 +00:00
kyverno/pkg/webhooks/server.go

399 lines
14 KiB
Go
Raw Normal View History

2019-05-13 18:33:01 +00:00
package webhooks
import (
"context"
"crypto/tls"
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"net/http"
"time"
2020-04-27 13:08:03 +00:00
"github.com/julienschmidt/httprouter"
2020-03-27 13:36:06 +00:00
"github.com/nirmata/kyverno/pkg/openapi"
2020-03-17 18:05:20 +00:00
"github.com/go-logr/logr"
"github.com/nirmata/kyverno/pkg/checker"
2019-08-17 16:58:14 +00:00
kyvernoclient "github.com/nirmata/kyverno/pkg/client/clientset/versioned"
kyvernoinformer "github.com/nirmata/kyverno/pkg/client/informers/externalversions/kyverno/v1"
2019-11-13 21:41:08 +00:00
kyvernolister "github.com/nirmata/kyverno/pkg/client/listers/kyverno/v1"
"github.com/nirmata/kyverno/pkg/config"
client "github.com/nirmata/kyverno/pkg/dclient"
2019-06-27 01:04:50 +00:00
"github.com/nirmata/kyverno/pkg/event"
"github.com/nirmata/kyverno/pkg/policystatus"
2019-11-11 19:10:25 +00:00
"github.com/nirmata/kyverno/pkg/policystore"
2019-11-12 22:41:29 +00:00
"github.com/nirmata/kyverno/pkg/policyviolation"
tlsutils "github.com/nirmata/kyverno/pkg/tls"
2019-11-11 22:52:09 +00:00
userinfo "github.com/nirmata/kyverno/pkg/userinfo"
"github.com/nirmata/kyverno/pkg/webhookconfig"
"github.com/nirmata/kyverno/pkg/webhooks/generate"
v1beta1 "k8s.io/api/admission/v1beta1"
2019-08-24 01:34:23 +00:00
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
rbacinformer "k8s.io/client-go/informers/rbac/v1"
rbaclister "k8s.io/client-go/listers/rbac/v1"
2019-08-12 17:02:07 +00:00
"k8s.io/client-go/tools/cache"
)
// WebhookServer contains configured TLS server with MutationWebhook.
// MutationWebhook gets policies from policyController and takes control of the cluster with kubeclient.
type WebhookServer struct {
2019-11-15 23:59:37 +00:00
server http.Server
client *client.Client
kyvernoClient *kyvernoclient.Clientset
2019-11-15 23:59:37 +00:00
// list/get cluster policy resource
pLister kyvernolister.ClusterPolicyLister
2019-11-15 23:59:37 +00:00
// returns true if the cluster policy store has synced atleast
pSynced cache.InformerSynced
2019-11-15 23:59:37 +00:00
// list/get role binding resource
rbLister rbaclister.RoleBindingLister
2019-11-15 23:59:37 +00:00
// return true if role bining store has synced atleast once
rbSynced cache.InformerSynced
2019-11-15 23:59:37 +00:00
// list/get cluster role binding resource
crbLister rbaclister.ClusterRoleBindingLister
2019-11-15 23:59:37 +00:00
// return true if cluster role binding store has synced atleast once
crbSynced cache.InformerSynced
2019-11-15 23:59:37 +00:00
// generate events
eventGen event.Interface
2019-11-15 23:59:37 +00:00
// webhook registration client
webhookRegistrationClient *webhookconfig.WebhookRegistrationClient
// API to send policy stats for aggregation
statusListener policystatus.Listener
2019-10-19 00:38:46 +00:00
// helpers to validate against current loaded configuration
configHandler config.Interface
2019-10-19 00:38:46 +00:00
// channel for cleanup notification
cleanUp chan<- struct{}
// last request time
lastReqTime *checker.LastReqTime
2019-11-11 19:10:25 +00:00
// store to hold policy meta data for faster lookup
pMetaStore policystore.LookupInterface
2019-11-12 22:41:29 +00:00
// policy violation generator
pvGenerator policyviolation.GeneratorInterface
// generate request generator
grGenerator *generate.Generator
resourceWebhookWatcher *webhookconfig.ResourceWebhookRegister
2020-03-17 18:05:20 +00:00
log logr.Logger
2020-03-27 13:36:06 +00:00
openAPIController *openapi.Controller
}
// NewWebhookServer creates new instance of WebhookServer accordingly to given configuration
// Policy Controller and Kubernetes Client should be initialized in configuration
func NewWebhookServer(
kyvernoClient *kyvernoclient.Clientset,
client *client.Client,
tlsPair *tlsutils.TlsPemPair,
pInformer kyvernoinformer.ClusterPolicyInformer,
rbInformer rbacinformer.RoleBindingInformer,
crbInformer rbacinformer.ClusterRoleBindingInformer,
eventGen event.Interface,
webhookRegistrationClient *webhookconfig.WebhookRegistrationClient,
statusSync policystatus.Listener,
configHandler config.Interface,
pMetaStore policystore.LookupInterface,
pvGenerator policyviolation.GeneratorInterface,
grGenerator *generate.Generator,
resourceWebhookWatcher *webhookconfig.ResourceWebhookRegister,
2020-03-17 18:05:20 +00:00
cleanUp chan<- struct{},
log logr.Logger,
2020-03-27 13:36:06 +00:00
openAPIController *openapi.Controller,
2020-03-17 18:05:20 +00:00
) (*WebhookServer, error) {
if tlsPair == nil {
return nil, errors.New("NewWebhookServer is not initialized properly")
}
var tlsConfig tls.Config
pair, err := tls.X509KeyPair(tlsPair.Certificate, tlsPair.PrivateKey)
if err != nil {
return nil, err
}
tlsConfig.Certificates = []tls.Certificate{pair}
ws := &WebhookServer{
client: client,
kyvernoClient: kyvernoClient,
pLister: pInformer.Lister(),
pSynced: pInformer.Informer().HasSynced,
rbLister: rbInformer.Lister(),
rbSynced: rbInformer.Informer().HasSynced,
crbLister: crbInformer.Lister(),
crbSynced: crbInformer.Informer().HasSynced,
eventGen: eventGen,
webhookRegistrationClient: webhookRegistrationClient,
statusListener: statusSync,
configHandler: configHandler,
cleanUp: cleanUp,
lastReqTime: resourceWebhookWatcher.LastReqTime,
pvGenerator: pvGenerator,
pMetaStore: pMetaStore,
grGenerator: grGenerator,
resourceWebhookWatcher: resourceWebhookWatcher,
2020-03-17 18:05:20 +00:00
log: log,
2020-03-27 13:36:06 +00:00
openAPIController: openAPIController,
}
2020-04-27 13:08:03 +00:00
mux := httprouter.New()
mux.HandlerFunc("POST", config.MutatingWebhookServicePath, ws.handlerFunc(ws.handleMutateAdmissionRequest, true))
mux.HandlerFunc("POST", config.ValidatingWebhookServicePath, ws.handlerFunc(ws.handleValidateAdmissionRequest, true))
mux.HandlerFunc("POST", config.PolicyMutatingWebhookServicePath, ws.handlerFunc(ws.handlePolicyMutation, true))
mux.HandlerFunc("POST", config.PolicyValidatingWebhookServicePath, ws.handlerFunc(ws.handlePolicyValidation, true))
mux.HandlerFunc("POST", config.VerifyMutatingWebhookServicePath, ws.handlerFunc(ws.handleVerifyRequest, false))
ws.server = http.Server{
Addr: ":443", // Listen on port for HTTPS requests
TLSConfig: &tlsConfig,
Handler: mux,
ReadTimeout: 15 * time.Second,
WriteTimeout: 15 * time.Second,
}
return ws, nil
}
2020-03-29 02:06:18 +00:00
func (ws *WebhookServer) handlerFunc(handler func(request *v1beta1.AdmissionRequest) *v1beta1.AdmissionResponse, filter bool) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
startTime := time.Now()
// for every request received on the ep update last request time,
// this is used to verify admission control
ws.lastReqTime.SetTime(time.Now())
admissionReview := ws.bodyToAdmissionReview(r, w)
if admissionReview == nil {
return
2020-03-25 05:23:03 +00:00
}
2020-03-29 02:11:58 +00:00
logger := ws.log.WithValues("kind", admissionReview.Request.Kind, "namespace", admissionReview.Request.Namespace, "name", admissionReview.Request.Name)
2020-03-29 02:06:18 +00:00
defer func() {
2020-03-29 02:11:58 +00:00
logger.V(4).Info("request processed", "processingTime", time.Since(startTime))
2020-03-29 02:06:18 +00:00
}()
admissionReview.Response = &v1beta1.AdmissionResponse{
Allowed: true,
2020-03-25 05:23:03 +00:00
}
2020-03-29 02:06:18 +00:00
// Do not process the admission requests for kinds that are in filterKinds for filtering
request := admissionReview.Request
if filter {
if !ws.configHandler.ToFilter(request.Kind.Kind, request.Namespace, request.Name) {
admissionReview.Response = handler(request)
}
} else {
admissionReview.Response = handler(request)
}
2020-03-29 02:06:18 +00:00
admissionReview.Response.UID = request.UID
2020-03-29 02:06:18 +00:00
responseJSON, err := json.Marshal(admissionReview)
if err != nil {
http.Error(w, fmt.Sprintf("Could not encode response: %v", err), http.StatusInternalServerError)
return
}
2020-03-29 02:06:18 +00:00
w.Header().Set("Content-Type", "application/json; charset=utf-8")
if _, err := w.Write(responseJSON); err != nil {
http.Error(w, fmt.Sprintf("could not write response: %v", err), http.StatusInternalServerError)
}
}
}
2020-01-11 13:03:11 +00:00
func (ws *WebhookServer) handleMutateAdmissionRequest(request *v1beta1.AdmissionRequest) *v1beta1.AdmissionResponse {
2020-03-17 18:05:20 +00:00
logger := ws.log.WithValues("uid", request.UID, "kind", request.Kind.Kind, "namespace", request.Namespace, "name", request.Name, "operation", request.Operation)
policies, err := ws.pMetaStore.ListAll()
if err != nil {
// Unable to connect to policy Lister to access policies
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to list policies. Policies are NOT being applied")
return &v1beta1.AdmissionResponse{Allowed: true}
}
2019-11-11 22:52:09 +00:00
var roles, clusterRoles []string
// getRoleRef only if policy has roles/clusterroles defined
if containRBACinfo(policies) {
roles, clusterRoles, err = userinfo.GetRoleRef(ws.rbLister, ws.crbLister, request)
2019-11-11 22:52:09 +00:00
if err != nil {
// TODO(shuting): continue apply policy if error getting roleRef?
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to get RBAC infromation for request")
2019-11-11 22:52:09 +00:00
}
2019-11-09 02:56:24 +00:00
}
// convert RAW to unstructured
resource, err := convertResource(request.Object.Raw, request.Kind.Group, request.Kind.Version, request.Kind.Kind, request.Namespace)
if err != nil {
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to convert RAW resource to unstructured format")
return &v1beta1.AdmissionResponse{
Allowed: false,
Result: &metav1.Status{
Status: "Failure",
Message: err.Error(),
},
}
}
if checkPodTemplateAnn(resource) {
return &v1beta1.AdmissionResponse{
Allowed: true,
Result: &metav1.Status{
Status: "Success",
},
}
}
2019-08-24 01:34:23 +00:00
// MUTATION
// mutation failure should not block the resource creation
// any mutation failure is reported as the violation
patches := ws.HandleMutation(request, resource, policies, roles, clusterRoles)
2019-08-24 01:34:23 +00:00
// patch the resource with patches before handling validation rules
2020-03-17 18:05:20 +00:00
patchedResource := processResourceWithPatches(patches, request.Object.Raw, logger)
2019-08-24 01:34:23 +00:00
if ws.resourceWebhookWatcher != nil && ws.resourceWebhookWatcher.RunValidationInMutatingWebhook == "true" {
2020-01-11 13:03:11 +00:00
// VALIDATION
2020-01-24 17:54:20 +00:00
ok, msg := ws.HandleValidation(request, policies, patchedResource, roles, clusterRoles)
2020-01-11 13:03:11 +00:00
if !ok {
2020-03-17 18:05:20 +00:00
logger.Info("admission request denied")
2020-01-11 13:03:11 +00:00
return &v1beta1.AdmissionResponse{
Allowed: false,
Result: &metav1.Status{
Status: "Failure",
Message: msg,
},
}
2019-08-24 01:34:23 +00:00
}
}
// GENERATE
// Only applied during resource creation
// Success -> Generate Request CR created successsfully
// Failed -> Failed to create Generate Request CR
if request.Operation == v1beta1.Create {
2020-01-24 17:55:39 +00:00
ok, msg := ws.HandleGenerate(request, policies, patchedResource, roles, clusterRoles)
if !ok {
2020-03-17 18:05:20 +00:00
logger.Info("admission request denied")
return &v1beta1.AdmissionResponse{
Allowed: false,
Result: &metav1.Status{
Status: "Failure",
Message: msg,
},
}
}
}
2019-08-24 01:34:23 +00:00
// Succesfful processing of mutation & validation rules in policy
patchType := v1beta1.PatchTypeJSONPatch
return &v1beta1.AdmissionResponse{
Allowed: true,
Result: &metav1.Status{
Status: "Success",
},
Patch: patches,
PatchType: &patchType,
}
}
2020-01-11 13:03:11 +00:00
func (ws *WebhookServer) handleValidateAdmissionRequest(request *v1beta1.AdmissionRequest) *v1beta1.AdmissionResponse {
2020-03-17 18:05:20 +00:00
logger := ws.log.WithValues("uid", request.UID, "kind", request.Kind.Kind, "namespace", request.Namespace, "name", request.Name, "operation", request.Operation)
policies, err := ws.pMetaStore.ListAll()
2020-01-11 13:03:11 +00:00
if err != nil {
// Unable to connect to policy Lister to access policies
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to list policies. Policies are NOT being applied")
2020-01-11 13:03:11 +00:00
return &v1beta1.AdmissionResponse{Allowed: true}
}
var roles, clusterRoles []string
// getRoleRef only if policy has roles/clusterroles defined
if containRBACinfo(policies) {
roles, clusterRoles, err = userinfo.GetRoleRef(ws.rbLister, ws.crbLister, request)
2020-01-11 13:03:11 +00:00
if err != nil {
// TODO(shuting): continue apply policy if error getting roleRef?
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to get RBAC infromation for request")
2020-01-11 13:03:11 +00:00
}
}
// VALIDATION
ok, msg := ws.HandleValidation(request, policies, nil, roles, clusterRoles)
if !ok {
2020-03-17 18:05:20 +00:00
logger.Info("admission request denied")
2020-01-11 13:03:11 +00:00
return &v1beta1.AdmissionResponse{
Allowed: false,
Result: &metav1.Status{
Status: "Failure",
Message: msg,
},
}
}
return &v1beta1.AdmissionResponse{
Allowed: true,
Result: &metav1.Status{
Status: "Success",
},
}
}
// RunAsync TLS server in separate thread and returns control immediately
2019-10-25 21:55:48 +00:00
func (ws *WebhookServer) RunAsync(stopCh <-chan struct{}) {
2020-03-17 18:05:20 +00:00
logger := ws.log
if !cache.WaitForCacheSync(stopCh, ws.pSynced, ws.rbSynced, ws.crbSynced) {
2020-03-17 18:05:20 +00:00
logger.Info("failed to sync informer cache")
2019-11-15 23:59:37 +00:00
}
go func(ws *WebhookServer) {
2020-03-17 18:05:20 +00:00
logger.V(3).Info("started serving requests", "addr", ws.server.Addr)
2019-08-28 00:00:16 +00:00
if err := ws.server.ListenAndServeTLS("", ""); err != http.ErrServerClosed {
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to listen to requests")
}
}(ws)
2020-03-17 18:05:20 +00:00
logger.Info("starting")
// verifys if the admission control is enabled and active
// resync: 60 seconds
// deadline: 60 seconds (send request)
// max deadline: deadline*3 (set the deployment annotation as false)
go ws.lastReqTime.Run(ws.pLister, ws.eventGen, ws.client, checker.DefaultResync, checker.DefaultDeadline, stopCh)
}
// Stop TLS server and returns control after the server is shut down
func (ws *WebhookServer) Stop(ctx context.Context) {
2020-03-17 18:05:20 +00:00
logger := ws.log
// cleanUp
// remove the static webhookconfigurations
go ws.webhookRegistrationClient.RemoveWebhookConfigurations(ws.cleanUp)
// shutdown http.Server with context timeout
err := ws.server.Shutdown(ctx)
if err != nil {
// Error from closing listeners, or context timeout:
2020-03-17 18:05:20 +00:00
logger.Error(err, "shutting down server")
ws.server.Close()
}
}
// bodyToAdmissionReview creates AdmissionReview object from request body
// Answers to the http.ResponseWriter if request is not valid
func (ws *WebhookServer) bodyToAdmissionReview(request *http.Request, writer http.ResponseWriter) *v1beta1.AdmissionReview {
2020-03-17 18:05:20 +00:00
logger := ws.log
var body []byte
if request.Body != nil {
if data, err := ioutil.ReadAll(request.Body); err == nil {
body = data
}
}
if len(body) == 0 {
2020-03-17 18:05:20 +00:00
logger.Info("empty body")
http.Error(writer, "empty body", http.StatusBadRequest)
return nil
}
contentType := request.Header.Get("Content-Type")
if contentType != "application/json" {
2020-03-17 18:05:20 +00:00
logger.Info("invalid Content-Type", "contextType", contentType)
http.Error(writer, "invalid Content-Type, expect `application/json`", http.StatusUnsupportedMediaType)
return nil
}
admissionReview := &v1beta1.AdmissionReview{}
if err := json.Unmarshal(body, &admissionReview); err != nil {
2020-03-17 18:05:20 +00:00
logger.Error(err, "failed to decode request body to type 'AdmissionReview")
http.Error(writer, "Can't decode body as AdmissionReview", http.StatusExpectationFailed)
return nil
}
2019-06-06 00:43:59 +00:00
return admissionReview
}