Sync Options

Argo CD allows users to customize some aspects of how it syncs the desired state in the target cluster. Some Sync Options can defined as annotations in a specific resource. Most of the Sync Options are configured in the Application resource spec.syncPolicy.syncOptions attribute. Multiple Sync Options which are configured with the argocd.argoproj.io/sync-options annotation can be concatenated with a , in the annotation value; white spaces will be trimmed.

Below you can find details about each available Sync Option:

No Prune Resources

v1.1

You may wish to prevent an object from being pruned:

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: Prune=false

In the UI, the pod will simply appear as out-of-sync:

sync option no prune

The sync-status panel shows that pruning was skipped, and why:

sync option no prune

The app will be out of sync if Argo CD expects a resource to be pruned. You may wish to use this along with compare options.

Disable Kubectl Validation

For a certain class of objects, it is necessary to kubectl apply them using the --validate=false flag. Examples of this are kubernetes types which uses RawExtension, such as ServiceCatalog. You can do using this annotations:

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: Validate=false

If you want to exclude a whole class of objects globally, consider setting resource.customizations in system level configuration.

Skip Dry Run for new custom resources types

When syncing a custom resource which is not yet known to the cluster, there are generally two options:

1) The CRD manifest is part of the same sync. Then Argo CD will automatically skip the dry run, the CRD will be applied and the resource can be created. 2) In some cases the CRD is not part of the sync, but it could be created in another way, e.g. by a controller in the cluster. An example is gatekeeper, which creates CRDs in response to user defined ConstraintTemplates. Argo CD cannot find the CRD in the sync and will fail with the error the server could not find the requested resource.

To skip the dry run for missing resource types, use the following annotation:

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: SkipDryRunOnMissingResource=true

The dry run will still be executed if the CRD is already present in the cluster.

No Resource Deletion

For certain resources you might want to retain them even after your application is deleted, for eg. Persistent Volume Claims. In such situations you can stop those resources from being cleaned up during app deletion by using the following annotation:

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: Delete=false

Selective Sync

Currently when syncing using auto sync Argo CD applies every object in the application. For applications containing thousands of objects this takes quite a long time and puts undue pressure on the api server. Turning on selective sync option which will sync only out-of-sync resources.

You can add this option by following ways

1) Add ApplyOutOfSyncOnly=true in manifest

Example:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - ApplyOutOfSyncOnly=true

2) Set sync option via argocd cli

Example:

  1. $ argocd app set guestbook --sync-option ApplyOutOfSyncOnly=true

Resources Prune Deletion Propagation Policy

By default, extraneous resources get pruned using foreground deletion policy. The propagation policy can be controlled using PrunePropagationPolicy sync option. Supported policies are background, foreground and orphan. More information about those policies could be found here.

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - PrunePropagationPolicy=foreground

Prune Last

This feature is to allow the ability for resource pruning to happen as a final, implicit wave of a sync operation, after the other resources have been deployed and become healthy, and after all other waves completed successfully.

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - PruneLast=true

This can also be configured at individual resource level.

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: PruneLast=true

Replace Resource Instead Of Applying Changes

By default, Argo CD executes kubectl apply operation to apply the configuration stored in Git. In some cases kubectl apply is not suitable. For example, resource spec might be too big and won’t fit into kubectl.kubernetes.io/last-applied-configuration annotation that is added by kubectl apply. In such cases you might use Replace=true sync option:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - Replace=true

If the Replace=true sync option is set the Argo CD will use kubectl replace or kubectl create command to apply changes.

Warning

During the sync process, the resources will be synchronized using the ‘kubectl replace/create’ command. This sync option has the potential to be destructive and might lead to resources having to be recreated, which could cause an outage for your application.

This can also be configured at individual resource level.

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: Replace=true

Server-Side Apply

This option enables Kubernetes Server-Side Apply.

By default, Argo CD executes kubectl apply operation to apply the configuration stored in Git. This is a client side operation that relies on kubectl.kubernetes.io/last-applied-configuration annotation to store the previous resource state.

However, there are some cases where you want to use kubectl apply --server-side over kubectl apply:

  • Resource is too big to fit in 262144 bytes allowed annotation size. In this case server-side apply can be used to avoid this issue as the annotation is not used in this case.
  • Patching of existing resources on the cluster that are not fully managed by Argo CD.
  • Use a more declarative approach, which tracks a user’s field management, rather than a user’s last applied state.

If ServerSideApply=true sync option is set, Argo CD will use kubectl apply --server-side command to apply changes.

It can be enabled at the application level like in the example below:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - ServerSideApply=true

To enable ServerSideApply just for an individual resource, the sync-option annotation can be used:

  1. metadata:
  2. annotations:
  3. argocd.argoproj.io/sync-options: ServerSideApply=true

ServerSideApply can also be used to patch existing resources by providing a partial yaml. For example, if there is a requirement to update just the number of replicas in a given Deployment, the following yaml can be provided to Argo CD:

  1. apiVersion: apps/v1
  2. kind: Deployment
  3. metadata:
  4. name: my-deployment
  5. spec:
  6. replicas: 3

Note that by the Deployment schema specification, this isn’t a valid manifest. In this case an additional sync option must be provided to skip schema validation. The example below shows how to configure the application to enable the two necessary sync options:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - ServerSideApply=true
  7. - Validate=false

In this case, Argo CD will use kubectl apply --server-side --validate=false command to apply changes.

Note: Replace=true takes precedence over ServerSideApply=true.

Fail the sync if a shared resource is found

By default, Argo CD will apply all manifests found in the git path configured in the Application regardless if the resources defined in the yamls are already applied by another Application. If the FailOnSharedResource sync option is set, Argo CD will fail the sync whenever it finds a resource in the current Application that is already applied in the cluster by another Application.

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. syncOptions:
  6. - FailOnSharedResource=true

Respect ignore difference configs

This sync option is used to enable Argo CD to consider the configurations made in the spec.ignoreDifferences attribute also during the sync stage. By default, Argo CD uses the ignoreDifferences config just for computing the diff between the live and desired state which defines if the application is synced or not. However during the sync stage, the desired state is applied as-is. The patch is calculated using a 3-way-merge between the live state the desired state and the last-applied-configuration annotation. This sometimes leads to an undesired results. This behavior can be changed by setting the RespectIgnoreDifferences=true sync option like in the example below:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. ignoreDifferences:
  5. - group: "apps"
  6. kind: "Deployment"
  7. jsonPointers:
  8. - /spec/replicas
  9. syncPolicy:
  10. syncOptions:
  11. - RespectIgnoreDifferences=true

The example above shows how an Argo CD Application can be configured so it will ignore the spec.replicas field from the desired state (git) during the sync stage. This is achieve by calculating and pre-patching the desired state before applying it in the cluster. Note that the RespectIgnoreDifferences sync option is only effective when the resource is already created in the cluster. If the Application is being created and no live state exists, the desired state is applied as-is.

Create Namespace

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. metadata:
  4. namespace: argocd
  5. spec:
  6. destination:
  7. server: https://kubernetes.default.svc
  8. namespace: some-namespace
  9. syncPolicy:
  10. syncOptions:
  11. - CreateNamespace=true

The example above shows how an Argo CD Application can be configured so it will create the namespace specified in spec.destination.namespace if it doesn’t exist already. Without this either declared in the Application manifest or passed in the CLI via --sync-option CreateNamespace=true, the Application will fail to sync if the namespace doesn’t exist.

Note that the namespace to be created must be informed in the spec.destination.namespace field of the Application resource. The metadata.namespace field in the Application’s child manifests must match this value, or can be omitted, so resources are created in the proper destination.

Namespace Metadata

We can also add labels and annotations to the namespace through managedNamespaceMetadata. If we extend the example above we could potentially do something like below:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. metadata:
  4. namespace: test
  5. spec:
  6. syncPolicy:
  7. managedNamespaceMetadata:
  8. labels: # The labels to set on the application namespace
  9. any: label
  10. you: like
  11. annotations: # The annotations to set on the application namespace
  12. the: same
  13. applies: for
  14. annotations: on-the-namespace
  15. syncOptions:
  16. - CreateNamespace=true

In order for Argo CD to manage the labels and annotations on the namespace, CreateNamespace=true needs to be set as a sync option, otherwise nothing will happen. If the namespace doesn’t already exist, or if it already exists and doesn’t already have labels and/or annotations set on it, you’re good to go. Using managedNamespaceMetadata will also set the resource tracking label (or annotation) on the namespace, so you can easily track which namespaces are managed by Argo CD.

In the case you do not have any custom annotations or labels but would nonetheless want to have resource tracking set on your namespace, that can be done by setting managedNamespaceMetadata with an empty labels and/or annotations map, like the example below:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. metadata:
  4. namespace: test
  5. spec:
  6. syncPolicy:
  7. managedNamespaceMetadata:
  8. labels: # The labels to set on the application namespace
  9. annotations: # The annotations to set on the application namespace
  10. syncOptions:
  11. - CreateNamespace=true

In the case where Argo CD is “adopting” an existing namespace which already has metadata set on it, we rely on using Server Side Apply in order not to lose metadata which has already been set. The main implication here is that it takes a few extra steps to get rid of an already preexisting field.

Imagine we have a pre-existing namespace as below:

  1. apiVersion: v1
  2. kind: Namespace
  3. metadata:
  4. name: foobar
  5. annotations:
  6. foo: bar
  7. abc: "123"

If we want to manage the foobar namespace with Argo CD and to then also remove the foo: bar annotation, in managedNamespaceMetadata we’d need to first rename the foo value:

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. managedNamespaceMetadata:
  6. annotations:
  7. abc: 123 # adding this is informational with SSA; this would be sticking around in any case until we set a new value
  8. foo: remove-me
  9. syncOptions:
  10. - CreateNamespace=true

Once that has been synced, we’re ok to remove foo

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. managedNamespaceMetadata:
  6. annotations:
  7. abc: 123 # adding this is informational with SSA; this would be sticking around in any case until we set a new value
  8. syncOptions:
  9. - CreateNamespace=true

Another thing to keep mind of is that if you have a k8s manifest for the same namespace in your Argo CD application, that will take precedence and overwrite whatever values that have been set in managedNamespaceMetadata. In other words, if you have an application that sets managedNamespaceMetadata

  1. apiVersion: argoproj.io/v1alpha1
  2. kind: Application
  3. spec:
  4. syncPolicy:
  5. managedNamespaceMetadata:
  6. annotations:
  7. abc: 123
  8. syncOptions:
  9. - CreateNamespace=true

But you also have a k8s manifest with a matching name

  1. apiVersion: v1
  2. kind: Namespace
  3. metadata:
  4. name: foobar
  5. annotations:
  6. foo: bar
  7. something: completely-different

The resulting namespace will have its annotations set to

  1. annotations:
  2. foo: bar
  3. something: completely-different