mirror of
https://github.com/argoproj/argo-cd.git
synced 2026-02-20 01:28:45 +01:00
165 lines
4.5 KiB
Go
165 lines
4.5 KiB
Go
package sync
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
|
|
"github.com/argoproj/argo-cd/gitops-engine/pkg/sync/common"
|
|
"github.com/argoproj/argo-cd/gitops-engine/pkg/sync/hook"
|
|
"github.com/argoproj/argo-cd/gitops-engine/pkg/sync/syncwaves"
|
|
"github.com/argoproj/argo-cd/gitops-engine/pkg/utils/kube"
|
|
)
|
|
|
|
// syncTask holds the live and target object. At least one should be non-nil. A targetObj of nil
|
|
// indicates the live object needs to be pruned. A liveObj of nil indicates the object has yet to
|
|
// be deployed
|
|
type syncTask struct {
|
|
phase common.SyncPhase
|
|
liveObj *unstructured.Unstructured
|
|
targetObj *unstructured.Unstructured
|
|
skipDryRun bool
|
|
syncStatus common.ResultCode
|
|
operationState common.OperationPhase
|
|
message string
|
|
waveOverride *int
|
|
}
|
|
|
|
func ternary(val bool, a, b string) string {
|
|
if val {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
func (t *syncTask) String() string {
|
|
return fmt.Sprintf("%s/%d %s %s/%s:%s/%s %s->%s (%s,%s,%s)",
|
|
t.phase, t.wave(),
|
|
ternary(t.isHook(), "hook", "resource"), t.group(), t.kind(), t.namespace(), t.name(),
|
|
ternary(t.liveObj != nil, "obj", "nil"), ternary(t.targetObj != nil, "obj", "nil"),
|
|
t.syncStatus, t.operationState, t.message,
|
|
)
|
|
}
|
|
|
|
func (t *syncTask) isPrune() bool {
|
|
return t.targetObj == nil
|
|
}
|
|
|
|
func (t *syncTask) resultKey() string {
|
|
return resourceResultKey(kube.GetResourceKey(t.obj()), t.phase)
|
|
}
|
|
|
|
// return the target object (if this exists) otherwise the live object
|
|
// some caution - often you explicitly want the live object not the target object
|
|
func (t *syncTask) obj() *unstructured.Unstructured {
|
|
return obj(t.targetObj, t.liveObj)
|
|
}
|
|
|
|
func (t *syncTask) wave() int {
|
|
if t.waveOverride != nil {
|
|
return *t.waveOverride
|
|
}
|
|
return syncwaves.Wave(t.obj())
|
|
}
|
|
|
|
func (t *syncTask) isHook() bool {
|
|
return hook.IsHook(t.obj())
|
|
}
|
|
|
|
func (t *syncTask) group() string {
|
|
return t.groupVersionKind().Group
|
|
}
|
|
|
|
func (t *syncTask) kind() string {
|
|
return t.groupVersionKind().Kind
|
|
}
|
|
|
|
func (t *syncTask) version() string {
|
|
return t.groupVersionKind().Version
|
|
}
|
|
|
|
func (t *syncTask) groupVersionKind() schema.GroupVersionKind {
|
|
return t.obj().GroupVersionKind()
|
|
}
|
|
|
|
func (t *syncTask) name() string {
|
|
return t.obj().GetName()
|
|
}
|
|
|
|
func (t *syncTask) namespace() string {
|
|
return t.obj().GetNamespace()
|
|
}
|
|
|
|
func (t *syncTask) pending() bool {
|
|
return t.operationState == ""
|
|
}
|
|
|
|
func (t *syncTask) running() bool {
|
|
return t.operationState.Running()
|
|
}
|
|
|
|
func (t *syncTask) completed() bool {
|
|
return t.operationState.Completed()
|
|
}
|
|
|
|
func (t *syncTask) successful() bool {
|
|
return t.operationState.Successful()
|
|
}
|
|
|
|
func (t *syncTask) pruned() bool {
|
|
return t.syncStatus == common.ResultCodePruned
|
|
}
|
|
|
|
func (t *syncTask) hookType() common.HookType {
|
|
if t.isHook() {
|
|
return common.HookType(t.phase)
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (t *syncTask) hasHookDeletePolicy(policy common.HookDeletePolicy) bool {
|
|
// cannot have a policy if it is not a hook, it is meaningless
|
|
if !t.isHook() {
|
|
return false
|
|
}
|
|
for _, p := range hook.DeletePolicies(t.obj()) {
|
|
if p == policy {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (t *syncTask) deleteBeforeCreation() bool {
|
|
return t.liveObj != nil && t.pending() && t.hasHookDeletePolicy(common.HookDeletePolicyBeforeHookCreation)
|
|
}
|
|
|
|
func (t *syncTask) deleteOnPhaseCompletion() bool {
|
|
return t.deleteOnPhaseFailed() || t.deleteOnPhaseSuccessful()
|
|
}
|
|
|
|
func (t *syncTask) deleteOnPhaseSuccessful() bool {
|
|
return t.liveObj != nil && t.hasHookDeletePolicy(common.HookDeletePolicyHookSucceeded)
|
|
}
|
|
|
|
func (t *syncTask) deleteOnPhaseFailed() bool {
|
|
return t.liveObj != nil && t.hasHookDeletePolicy(common.HookDeletePolicyHookFailed)
|
|
}
|
|
|
|
func (t *syncTask) resourceKey() kube.ResourceKey {
|
|
resourceKey := kube.GetResourceKey(t.obj())
|
|
if t.liveObj != nil {
|
|
// t.targetObj has a namespace set for cluster-scoped resources, which causes kube.GetResourceKey()
|
|
// to produce an incorrect key with the namespace included. For example:
|
|
// rbac.authorization.k8s.io/ClusterRole/my-namespace/my-cluster-role
|
|
// instead of the correct rbac.authorization.k8s.io/ClusterRole//my-cluster-role.
|
|
// To prevent resource lookup issues, we always rely on the namespace of the live object if it is available.
|
|
// This logic will work for both cluster scoped and namespace scoped resources.
|
|
//
|
|
// Refer to https://github.com/argoproj/argo-cd/gitops-engine/blob/8007df5f6c5dd78a1a8cef73569468ce4d83682c/pkg/sync/sync_context.go#L827-L833
|
|
resourceKey.Namespace = t.liveObj.GetNamespace()
|
|
}
|
|
return resourceKey
|
|
}
|