...
1
16
17 package approval
18
19 import (
20 "context"
21 "fmt"
22 "io"
23
24 "k8s.io/klog/v2"
25
26 "k8s.io/apiserver/pkg/admission"
27 genericadmissioninit "k8s.io/apiserver/pkg/admission/initializer"
28 "k8s.io/apiserver/pkg/authorization/authorizer"
29
30 api "k8s.io/kubernetes/pkg/apis/certificates"
31 "k8s.io/kubernetes/plugin/pkg/admission/certificates"
32 )
33
34
35 const PluginName = "CertificateApproval"
36
37
38 func Register(plugins *admission.Plugins) {
39 plugins.Register(PluginName, func(config io.Reader) (admission.Interface, error) {
40 return NewPlugin(), nil
41 })
42 }
43
44
45 type Plugin struct {
46 *admission.Handler
47 authz authorizer.Authorizer
48 }
49
50
51 func (p *Plugin) SetAuthorizer(authz authorizer.Authorizer) {
52 p.authz = authz
53 }
54
55
56 func (p *Plugin) ValidateInitialization() error {
57 if p.authz == nil {
58 return fmt.Errorf("%s requires an authorizer", PluginName)
59 }
60 return nil
61 }
62
63 var _ admission.ValidationInterface = &Plugin{}
64 var _ genericadmissioninit.WantsAuthorizer = &Plugin{}
65
66
67 func NewPlugin() *Plugin {
68 return &Plugin{
69 Handler: admission.NewHandler(admission.Update),
70 }
71 }
72
73 var csrGroupResource = api.Resource("certificatesigningrequests")
74
75
76
77 func (p *Plugin) Validate(ctx context.Context, a admission.Attributes, _ admission.ObjectInterfaces) error {
78
79
80 if a.GetSubresource() != "approval" ||
81 a.GetResource().GroupResource() != csrGroupResource {
82 return nil
83 }
84
85
86
87
88 csr, ok := a.GetOldObject().(*api.CertificateSigningRequest)
89 if !ok {
90 return admission.NewForbidden(a, fmt.Errorf("expected type CertificateSigningRequest, got: %T", a.GetOldObject()))
91 }
92
93 if !certificates.IsAuthorizedForSignerName(ctx, p.authz, a.GetUserInfo(), "approve", csr.Spec.SignerName) {
94 klog.V(4).Infof("user not permitted to approve CertificateSigningRequest %q with signerName %q", csr.Name, csr.Spec.SignerName)
95 return admission.NewForbidden(a, fmt.Errorf("user not permitted to approve requests with signerName %q", csr.Spec.SignerName))
96 }
97
98 return nil
99 }
100
View as plain text