This document answers some of the frequently asked questions w.r.t AKO.
The key deleteConfig in the data section of AKO configmap can be used to clean up the setup. Edit AKO configmap and set deleteConfig: "true" to delete ako created objects in Avi. After the flag is set in configmap, annotation AviObjectDeletionStatus
is added in the AKO statefulset with the value as Started
.
annotations:
AviObjectDeletionStatus: Started
After all relevant objects gets deleted from Avi, the value of the annotation is changed to Done
.
annotations:
AviObjectDeletionStatus: Done
To re-create the objects in Avi, the configmap has to be edited to set deleteConfig: "false".
AKO follows hostname based sharding to sync multiple ingresses with same hostname to a single virtual service. When an ingress object is created with multiple hostnames, AKO generates an md5 hash using the hostname and the Shard VS number. This uniquely maps an FQDN to a given Shared VS and avoids DNS conflicts. During initial clean bootup, if the Shared VS does not exist in Avi - AKO creates the same and then patches the ingress FQDN to it either in the form of a pool (for insecure routes) or in the form of an SNI child virtual service (in case of secure routes).
The Shared VSes aren't deleted if all the FQDNs mapped to it are removed from Kubernetes. However, if the user wants AKO to delete unused shared VSes - a pod restart is required that would evaluate the VS and delete it appropriately.
Even though the unused shared VSes are deleted, the shared VS VIPs are retained to regain the same FQDN to VS VIP mapping. To delete the retained shared VS VIPs, the user has to manually delete them from the controller UI or shell.
If you create ingress with an insecure host/path combination then AKO creates a corresponding Avi Pool object and patches the pool on one of the existing shard virtual services. The shard VS has a datascript associated with it that reads the host/path of the incoming request and appropriately selects a pool by matching it with the priority label specified for each pool member (corresponding to a host/path combination).
For secure ingresses, an SNI virtual service is created which although is a dedicated virtual service, does not have any IP addresses associated with it. The SNI virtual service is a child to a parent virtual service and is created based on the secret object specified in the ingress file against the host/path that is meant to be accessed securely.
In the current AKO model, the Shard VS size is an enum. It allows 3 pre-fixed sets of values viz. LARGE
, MEDIUM
and SMALL
. They
respectively correspond to 8, 4 and 1 virtual service. The decision of selecting one of these sizes for Shard VS is driven by the
size of the Kubernetes cluster's ingress requirements. Typically, it's advised to always go with the highest possible Shard VS number
that is - LARGE
to account for future expansion.
The shard size can be set to DEDICATED
to disable shard mode to create dedicated Virtual Services per hostname.
To Shard to virtual services, AKO uses a sharding mechanism that is driven by the hostname
of each rule within an ingress object. This ensures that a unique hostname is always sharded consistently to the same virtual service.
Since the sharding logic is determined by the number of Shard virtual services, changing the Shard VS number has the potential hazard of messing up an existing cluster's already synced objects. Hence it's recommended that the Shard VS numbers are not changed once fixed.
Altering the shard VS number is considered as disruptive. This is because dynamic re-adjustment of shard numbers may re-balance the ingress to VS mapping. Hence if you want to alter the shard VS number, first delete the older configmap and trigger a complete cleanup of the VSes in the controller. Followed by an edit of the configmap and restart of AKO.
Currently, the number of A records allowed per virtual service is 1000. If a shard size of SMALL
is selected and the number of A records via the Ingress objects exceed 1000, then a greater shardSize
has to be configured via the shardSize
knob. Alternatively one can create a separate IngressClass for a set of Ingress objects and specify a shardSize
in the AviInfraSettings
CRD which would allow AKO to place the A records scoped to the VS that is mapped to the IngressClass.
Static routes are created with cluster name as the label. While deploying AKO the admin or the operator decides a Service Engine Group for a given Kubernetes cluster. The same labels are tagged on the routes of this AKO cluster. These routes are pushed to the Service Engine's created on the Service Engine Group. The static routes map each POD CIDR with the Kubernetes node's IP address. However, for static routes to work, the Service Engines must be L2 adjacent to your Kubernetes nodes.
The ingress API does not prohibit the user from creating the same SNI hostname across multiple namespaces. AKO will create 1 SNI virtual service and gather all paths associated with it across namespaces to create corresponding switching rules. However, the user needs to denote each ingress with the TLS secret for a given hostname to qualify the host for the SNI virtual service.
Consider the below example:
Ingress 1 (default namespace) --> SNI hostname --> foo.com path: /foo, Secret: foo
Ingress 1 (foo namespace) --> SNI hostname --> foo.com path: /bar, Secret: foo
In the above case, only 1 SNI virtual service will be created with a sslkeyandcertificate as foo
.
However if the following happens:
Ingress 1 (default namespace) --> SNI hostname --> foo.com path: /foo, Secret: foo
Ingress 1 (foo namespace) --> SNI hostname --> foo.com path: /bar, Secret: bar
Then the behaviour of the SNI virtual service would be indeterministic since the secrets for the same SNI are different. This is not supported.
AKO runs a refresh cycle that currently just refreshes the cloud object parameters. However, if some out of band operations are performed on objects created by AKO via directly interacting with the Avi APIs, AKO may not always be able to remediate an error caused due to this.
AKO has the best effort, retry layer implementation that would try to detect a problem (For example an SNI VS deleted from the Avi UI), but it is not guaranteed to work for all such manual operations.
Upon reboot of AKO - a full reconciliation loop is run and most of the out-of-band changes are overwritten with AKO's view of the intended model. This does not happen in every full sync cycle.
What is the expected behaviour for the same host/path combination across different secure/insecure ingresses?
The ingress API allows users to add duplicate hostpaths bound to separate backend services. Something like this:
Ingress1 (default namespace) --> foo.com path: /foo, Service: svc1
Ingress2 (default namespace) --> foo.com path: /foo, Service: svc2
Also, ingress allows you to have a mix of secure and insecure hostpath bound to the same backend services like so:
Ingress1 (default namespace) --> SNI hostname --> foo.com path: /foo, Secret: secret1
Ingress2 (default namespace) --> foo.com path: /foo, Service: svc2
AKO does not explicitly handle these conditions and would continue syncing these objects on the Avi controller, but this may lead to traffic issues. AKO does the best effort of detecting some of these conditions by printing them in logs. A sample log statement looks like this:
key: Ingress/default/ingress2, msg: Duplicate entries found for hostpath default/ingress2: foo.com/foo in ingresses: ["default/ingress1"]
AKO programs a static route for every node IP and the POD CIDR associated with it. Even though node state changes to NotReady
in Kubernetes this configuration is stored in the node object and does not change when the node rebooted/shutdown.
Hence AKO will not remove the static routes until the Kubernetes node is completely removed from the cluster.
The short answer is No. The ingress objects should point to the service of type clusterIP. Loadbalancer services either point to an ingress controller POD if one is using an in cluster ingress controller or they can directly point to application PODs that need layer 4 load-balancing.
If you have such a configuration where the ingress objects are pointing to services of the type load balancer, AKO's behaviour would be indeterministic.
AKO would stop processing kubernetes objects and no update would be made to the AVI Controller. After the connection to AVI Controller is restored, AKO pod has to be rebooted. This can be done by deleting the exiting POD and ako deployment would bring up a new POD, which would start processing kubernetes objects after verifying connectivity to AVI Controller.
AKO does not process ingress objects in openshift environment. If any route corresponding to the ingress object is found, AKO would process that route.
A set of shared Virtual Services are created for passthrough hosts to listen on port 443 to handle secure traffic using L4 datascript. These virtual services have names of the format 'cluster-name'-Shared-Passthrough
-'shard-number'. Number of shards can be configured using the flag passthroughShardSize
while installation using helm.
For passthrough routes, the supported values for insecureEdgeTerminationPolicy are None and Redirect. To handle insecure traffic for passthrough routes a set of shared Virtual Services are created with names of the format 'cluster-name'-Shared-Passthrough
-'shard-number'-insecure
. These Virtual Services listen on port 80. If for any passthrough route, the insecureEdgeTerminationPolicy is found to be 'Redirect', then an HTTP Policy is configured in the insecure passthrough shared VS to send appropriate response to an incoming insecure traffic.
AKO goes for a reboot and retries some of the invalid input errors. Below are some of the cases to look out for in the logs.
- If an invalid cloud name is given in
values.yaml
or if ipam_provider_ref is not set in the vCenter and No Access clouds. - If the same Service Engine Group is used for multiple clusters for vCenter and No Access clouds in Cluster IP mode. This happens as AKO expects unique SE group per cluster if routes are configured by AKO for POD reachability. Look for the
Labels does not match with cluster name
message in the logs which points to two clusters using the same Service Engine Group.
The default behaviour for AKO s to populate all the Node IP as pool server. If master node is not schedulable then, it will be marked down. nodePortSelector
can be used to specify the labels
for the node. In that case, all the node with that label will be picked for the pool server. If the master node is not schedulable then, the fix is to remove the nodePortSelector
label for the master node.
For secure routes having termination type edge/reencrypt, key and certificate must be specified in the spec of the route. AKO would not handle routes of these types without key and certificate.
AKO would reject those routes, as each backend should be unique ith it's own weight. Multiple backends having same service would make weight calculation indeterministic.
No. The feature NodePortLocal can be used only with Antrea CNI and the feature must be enabled in Antrea feature gates.
No. Users can only use service of type ClusterIP as backend of Ingresses in this mode.
Can we use serviceType NodePort or ClusterIP in AKO, if the CNI type is Antrea and NodePortLocal feature is enabled in Antrea ?
Yes. AKO would create AVI objects based on the relevant serviceType set in AKO.
The serviceType
in AKO can be changed from ClusterIP
to NodePortLocal
or NodePort
. The serviceType
change is considered disruptive.
Hence before the serviceType
change, all the existing AKO configuration must be deleted. This can be achieved as follows:
- Set the
deleteConfig
flag totrue
. - Wait for AKO to delete all the relevant Avi configuration and update the deletion status in AKO's statefulset status.
- Change the
serviceType
- Set the
deleteConfig
flag tofalse
- Reboot AKO
For example, during the change of serviceType
from ClusterIP
to NodePortLocal
, the deleteConfig
flag will:
- Delete the static routes.
- Delete the SE group labels.
No. After changing the serviceType, AKO has to be rebooted and all objects which are not required, would be deleted as part of the reboot process.
If serviceType is changed from NodePortLocal, would AKO remove NPL annotation from the Services automatically ?
No. After changing the serviceType, the users have to remove NPL annotation from the Services themselves.
Markers, associated with an Avi object, are visible on the Avi Vantage UI and the Avi shell.
- Avi Vantage UI:
Markers are visible on UI for avi controller version >= 20.1.6. In order to view marker labels for an object, do the following:
- Edit an object.
- Navigate to the
Advanced
tab. - The Role-Based Access Control (RBAC) shows the markers associated with the object.
- Avi Shell
Markers can be viewed for an object with the command.
[avi-shell] show <object type> <object name>
Sample command with output:
[avi-shell] show virtualservice kubernetes--Shared-L7-EVH-0
+------------------------------------+----------------------------------------------------------------------------------+
| Field | Value |
+------------------------------------+----------------------------------------------------------------------------------+
| uuid | virtualservice-bc4e964a-af79-4b8f-91b2-de0e7ee9388d |
| name | kubernetes--Shared-L7-EVH-0 |
| enabled | True |
| vsvip_ref | kubernetes--Shared-L7-EVH-0 |
| use_vip_as_snat | False |
| . | . |
| . | . |
| . | . |
| . | . |
| . | . |
| markers[1] | |
| key | clustername |
| values[1] | kubernetes |
| allow_invalid_client_cert | False |
| vh_type | VS_TYPE_VH_ENHANCED |
+------------------------------------+----------------------------------------------------------------------------------+