Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add api server extra args map to cluster spec #7755

Merged
merged 9 commits into from
Mar 20, 2024
6 changes: 6 additions & 0 deletions config/crd/bases/anywhere.eks.amazonaws.com_clusters.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,12 @@ spec:
type: object
controlPlaneConfiguration:
properties:
apiServerExtraArgs:
additionalProperties:
type: string
description: APIServerExtraArgs defines the flags to configure
for the API server.
type: object
certSans:
description: CertSANs is a slice of domain names or IPs to be
added as Subject Name Alternatives of the Kube API Servers Certificate.
Expand Down
6 changes: 6 additions & 0 deletions config/manifest/eksa-components.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -3854,6 +3854,12 @@ spec:
type: object
controlPlaneConfiguration:
properties:
apiServerExtraArgs:
additionalProperties:
type: string
description: APIServerExtraArgs defines the flags to configure
for the API server.
type: object
certSans:
description: CertSANs is a slice of domain names or IPs to be
added as Subject Name Alternatives of the Kube API Servers Certificate.
Expand Down
20 changes: 20 additions & 0 deletions internal/pkg/api/cluster.go
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,26 @@ func WithControlPlaneLabel(key string, val string) ClusterFiller {
}
}

// WithControlPlaneAPIServerExtraArgs adds the APIServerExtraArgs to the cluster spec.
func WithControlPlaneAPIServerExtraArgs() ClusterFiller {
sp1999 marked this conversation as resolved.
Show resolved Hide resolved
return func(c *anywherev1.Cluster) {
if c.Spec.ControlPlaneConfiguration.APIServerExtraArgs == nil {
c.Spec.ControlPlaneConfiguration.APIServerExtraArgs = map[string]string{}
}
issuerURL := "https://" + c.Spec.ControlPlaneConfiguration.Endpoint.Host
c.Spec.ControlPlaneConfiguration.APIServerExtraArgs["service-account-jwks-uri"] = issuerURL + "/openid/v1/jwks"
sp1999 marked this conversation as resolved.
Show resolved Hide resolved
}
}

// RemoveAllAPIServerExtraArgs removes all the API server flags from the cluster spec.
func RemoveAllAPIServerExtraArgs() ClusterFiller {
return func(c *anywherev1.Cluster) {
for k := range c.Spec.ControlPlaneConfiguration.APIServerExtraArgs {
delete(c.Spec.ControlPlaneConfiguration.APIServerExtraArgs, k)
}
}
}

// WithPodCidr sets an explicit pod CIDR, overriding the provider's default.
func WithPodCidr(podCidr string) ClusterFiller {
return func(c *anywherev1.Cluster) {
Expand Down
97 changes: 97 additions & 0 deletions internal/pkg/api/cluster_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -231,3 +231,100 @@ func TestWithPodCidr(t *testing.T) {
g.Expect(cluster.Spec.ClusterNetwork.Pods.CidrBlocks).To(Equal([]string{"10.0.0.0/16", "172.16.42.0/20"}))
})
}

func TestWithControlPlaneAPIServerExtraArgs(t *testing.T) {
tests := []struct {
name string
cluster *anywherev1.Cluster
want anywherev1.ControlPlaneConfiguration
}{
{
name: "no control plane api server extra args",
cluster: &anywherev1.Cluster{
Spec: anywherev1.ClusterSpec{
ControlPlaneConfiguration: anywherev1.ControlPlaneConfiguration{
Endpoint: &anywherev1.Endpoint{
Host: "10.20.30.40",
},
},
},
},
want: anywherev1.ControlPlaneConfiguration{
APIServerExtraArgs: map[string]string{
"service-account-jwks-uri": "https://10.20.30.40/openid/v1/jwks",
},
Endpoint: &anywherev1.Endpoint{
Host: "10.20.30.40",
},
},
},
{
name: "with control plane api server extra args",
cluster: &anywherev1.Cluster{
Spec: anywherev1.ClusterSpec{
ControlPlaneConfiguration: anywherev1.ControlPlaneConfiguration{
APIServerExtraArgs: map[string]string{
"service-account-jwks-uri": "https://40.50.60.70/openid/v1/jwks",
},
Endpoint: &anywherev1.Endpoint{
Host: "10.20.30.40",
},
},
},
},
want: anywherev1.ControlPlaneConfiguration{
APIServerExtraArgs: map[string]string{
"service-account-jwks-uri": "https://10.20.30.40/openid/v1/jwks",
},
Endpoint: &anywherev1.Endpoint{
Host: "10.20.30.40",
},
},
},
}
for _, tt := range tests {
t.Run(
tt.name,
func(t *testing.T) {
api.WithControlPlaneAPIServerExtraArgs()(tt.cluster)
g := NewWithT(t)
g.Expect(tt.cluster.Spec.ControlPlaneConfiguration).To(Equal(tt.want))
},
)
}
}

func TestRemoveAllAPIServerExtraArgs(t *testing.T) {
tests := []struct {
name string
cluster *anywherev1.Cluster
want anywherev1.ControlPlaneConfiguration
}{
{
name: "with control plane api server extra args",
cluster: &anywherev1.Cluster{
Spec: anywherev1.ClusterSpec{
ControlPlaneConfiguration: anywherev1.ControlPlaneConfiguration{
APIServerExtraArgs: map[string]string{
"service-account-issuer": "test-service-account-issuer-url",
"service-account-jwks-uri": "test-service-account-jwks-uri",
},
},
},
},
want: anywherev1.ControlPlaneConfiguration{
APIServerExtraArgs: map[string]string{},
},
},
}
for _, tt := range tests {
t.Run(
tt.name,
func(t *testing.T) {
api.RemoveAllAPIServerExtraArgs()(tt.cluster)
g := NewWithT(t)
g.Expect(tt.cluster.Spec.ControlPlaneConfiguration).To(Equal(tt.want))
},
)
}
}
33 changes: 33 additions & 0 deletions pkg/api/v1alpha1/cluster.go
Original file line number Diff line number Diff line change
Expand Up @@ -190,6 +190,8 @@ var clusterConfigValidations = []func(*Cluster) error{
validatePackageControllerConfiguration,
validateEksaVersion,
validateControlPlaneCertSANs,
validateControlPlaneAPIServerExtraArgs,
validateControlPlaneAPIServerOIDCExtraArgs,
}

// GetClusterConfig parses a Cluster object from a multiobject yaml file in disk
Expand Down Expand Up @@ -494,6 +496,37 @@ func validateControlPlaneCertSANs(cfg *Cluster) error {
return nil
}

func validateControlPlaneAPIServerExtraArgs(clusterConfig *Cluster) error {
if clusterConfig.Spec.ControlPlaneConfiguration.APIServerExtraArgs != nil && !features.IsActive(features.APIServerExtraArgsEnabled()) {
return errors.New("please enable feature flag to configure APIServerExtraArgs")
sp1999 marked this conversation as resolved.
Show resolved Hide resolved
}
return nil
}

func validateControlPlaneAPIServerOIDCExtraArgs(clusterConfig *Cluster) error {
oidcFlags := []string{
"oidc-issuer-url",
"oidc-client-id",
"oidc-groups-claim",
"oidc-groups-prefix",
"oidc-required-claim",
"oidc-username-claim",
"oidc-username-prefix",
}
if clusterConfig.Spec.IdentityProviderRefs != nil {
for _, ref := range clusterConfig.Spec.IdentityProviderRefs {
if ref.Kind == OIDCConfigKind {
for _, flag := range oidcFlags {
if _, has := clusterConfig.Spec.ControlPlaneConfiguration.APIServerExtraArgs[flag]; has {
return fmt.Errorf("%s flag is already configured in OIDCConfig. please remove it from apiServerExtraArgs", flag)
sp1999 marked this conversation as resolved.
Show resolved Hide resolved
}
}
}
}
}
return nil
}

func validateWorkerNodeGroups(clusterConfig *Cluster) error {
workerNodeGroupConfigs := clusterConfig.Spec.WorkerNodeGroupConfigurations
if len(workerNodeGroupConfigs) <= 0 {
Expand Down
4 changes: 3 additions & 1 deletion pkg/api/v1alpha1/cluster_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -307,6 +307,8 @@ type ControlPlaneConfiguration struct {
CertSANs []string `json:"certSans,omitempty"`
// MachineHealthCheck is a control-plane level override for the timeouts and maxUnhealthy specified in the top-level MHC configuration. If not configured, the defaults in the top-level MHC configuration are used.
MachineHealthCheck *MachineHealthCheck `json:"machineHealthCheck,omitempty"`
// APIServerExtraArgs defines the flags to configure for the API server.
APIServerExtraArgs map[string]string `json:"apiServerExtraArgs,omitempty"`
sp1999 marked this conversation as resolved.
Show resolved Hide resolved
}

// MachineHealthCheck allows to configure timeouts for machine health checks. Machine Health Checks are responsible for remediating unhealthy Machines.
Expand Down Expand Up @@ -363,7 +365,7 @@ func (n *ControlPlaneConfiguration) Equal(o *ControlPlaneConfiguration) bool {
}
return n.Count == o.Count && n.MachineGroupRef.Equal(o.MachineGroupRef) &&
TaintsSliceEqual(n.Taints, o.Taints) && MapEqual(n.Labels, o.Labels) &&
SliceEqual(n.CertSANs, o.CertSANs)
SliceEqual(n.CertSANs, o.CertSANs) && MapEqual(n.APIServerExtraArgs, o.APIServerExtraArgs)
}

type Endpoint struct {
Expand Down
50 changes: 50 additions & 0 deletions pkg/api/v1alpha1/cluster_types_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1460,6 +1460,10 @@ func TestControlPlaneConfigurationEqual(t *testing.T) {
taints1DiffOrder := []corev1.Taint{taint2, taint1}
taints2 := []corev1.Taint{taint1}

var emptyAPIServerExtraArgs map[string]string
apiServerExtraArgs1 := map[string]string{"key1": "value1"}
apiServerExtraArgs2 := map[string]string{"key2": "value2"}

testCases := []struct {
testName string
cluster1CPConfig, cluster2CPConfig *v1alpha1.ControlPlaneConfiguration
Expand Down Expand Up @@ -1595,6 +1599,52 @@ func TestControlPlaneConfigurationEqual(t *testing.T) {
},
want: true,
},
{
testName: "both api server extra args equal",
cluster1CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: apiServerExtraArgs1,
},
cluster2CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: apiServerExtraArgs1,
},
want: true,
},
{
testName: "different api server extra args",
cluster1CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: apiServerExtraArgs1,
},
cluster2CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: apiServerExtraArgs2,
},
want: false,
},
{
testName: "one api server extra args not present",
cluster1CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: apiServerExtraArgs1,
},
cluster2CPConfig: &v1alpha1.ControlPlaneConfiguration{},
want: false,
},
{
testName: "one api server extra args not present and other empty",
cluster1CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: emptyAPIServerExtraArgs,
},
cluster2CPConfig: &v1alpha1.ControlPlaneConfiguration{},
want: true,
},
{
testName: "both api server extra args empty",
cluster1CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: emptyAPIServerExtraArgs,
},
cluster2CPConfig: &v1alpha1.ControlPlaneConfiguration{
APIServerExtraArgs: emptyAPIServerExtraArgs,
},
want: true,
},
}
for _, tt := range testCases {
t.Run(tt.testName, func(t *testing.T) {
Expand Down
7 changes: 7 additions & 0 deletions pkg/api/v1alpha1/zz_generated.deepcopy.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

20 changes: 20 additions & 0 deletions pkg/clusterapi/extraargs.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,15 @@ func EtcdEncryptionExtraArgs(config *[]v1alpha1.EtcdEncryption) ExtraArgs {
return args
}

// APIServerExtraArgs takes a map of API Server extra args and returns the relevant API server extra args if it's not nil or empty.
func APIServerExtraArgs(apiServerExtraArgs map[string]string) ExtraArgs {
args := ExtraArgs{}
for k, v := range apiServerExtraArgs {
args.AddIfNotEmpty(k, v)
}
return args
}

func PodIAMAuthExtraArgs(podIAMConfig *v1alpha1.PodIAMConfig) ExtraArgs {
if podIAMConfig == nil {
return nil
Expand Down Expand Up @@ -136,6 +145,17 @@ func (e ExtraArgs) Append(args ExtraArgs) ExtraArgs {
return e
}

// SetPodIAMAuthExtraArgs sets the api server extra args for the podIAMConfig.
func SetPodIAMAuthExtraArgs(podIAMConfig *v1alpha1.PodIAMConfig, apiServerExtraArgs map[string]string) {
if podIAMFlags := PodIAMAuthExtraArgs(podIAMConfig); podIAMFlags != nil {
if v, has := apiServerExtraArgs["service-account-issuer"]; has {
apiServerExtraArgs["service-account-issuer"] = strings.Join([]string{v, podIAMFlags["service-account-issuer"]}, ",")
} else {
apiServerExtraArgs["service-account-issuer"] = podIAMFlags["service-account-issuer"]
}
}
}

func (e ExtraArgs) ToPartialYaml() templater.PartialYaml {
p := templater.PartialYaml{}
for k, v := range e {
Expand Down
Loading