ResourceVersion和Generation

kubernetes有两个资源对象相关的属性。

  • ResourceVersion
  • Generation

kubernetes所有资源对象都有此属性,那这两个属性有和差异? 简单来说

  • ResourceVersion: Kubernetes 资源版本控制采用乐观锁的id,apiserver在写入etcd时作冲突检测。 基于底层etcd的revision机制,资源对象每次update时都会改变,且集群范围内唯一。
  • Generation; 初始值为1,随Spec内容的改变而自增.

当 patch 一个k8s object 的时候, 能不能写入etcd 要根据ResourceVersion做并发检查, 写入ectd 之后,generation +1. 从这个例子可以看出, generation 是标识这个k'8s object 本身修改的版本次数。 而ResourceVersion 则是在写入etcd之前做并发校验的。generation 随spec 的变化而自增, 可以这么理解,任何对object 修改都是spec 的变化。因此通常用generation 来判断有没有另外一个actor 来改过object。先来看下ResourceVersion。

ResourceVersion

apiVersion: apps/v1
kind: Deployment
metadata:
  annotations:
    deployment.kubernetes.io/revision: "6"
  creationTimestamp: "2020-08-14T10:23:27Z"
  generation: 7
  labels:
    app: nginx
  name: nginx
  namespace: default
  resourceVersion: "3313340114"
  uid: 215f58cf-4be8-472a-8ea5-be6d33b01256

resourceVersion的维护其实是利用了底层存储etcd的Revision机制。

apiserver存储相关的接口定义都在interfaces.go。

Versioner

定义了与resourceVersion操作相关的接口

// Versioner abstracts setting and retrieving metadata fields from database response
// onto the object ot list. It is required to maintain storage invariants - updating an
// object twice with the same data except for the ResourceVersion and SelfLink must be
// a no-op. A resourceVersion of type uint64 is a 'raw' resourceVersion,
// intended to be sent directly to or from the backend. A resourceVersion of
// type string is a 'safe' resourceVersion, intended for consumption by users.
type Versioner interface {
    // UpdateObject sets storage metadata into an API object. Returns an error if the object
    // cannot be updated correctly. May return nil if the requested object does not need metadata
    // from database.
    UpdateObject(obj runtime.Object, resourceVersion uint64) error
    // UpdateList sets the resource version into an API list object. Returns an error if the object
    // cannot be updated correctly. May return nil if the requested object does not need metadata from
    // database. continueValue is optional and indicates that more results are available if the client
    // passes that value to the server in a subsequent call. remainingItemCount indicates the number
    // of remaining objects if the list is partial. The remainingItemCount field is omitted during
    // serialization if it is set to nil.
    UpdateList(obj runtime.Object, resourceVersion uint64, continueValue string, remainingItemCount *int64) error
    // PrepareObjectForStorage should set SelfLink and ResourceVersion to the empty value. Should
    // return an error if the specified object cannot be updated.
    PrepareObjectForStorage(obj runtime.Object) error
    // ObjectResourceVersion returns the resource version (for persistence) of the specified object.
    // Should return an error if the specified object does not have a persistable version.
    ObjectResourceVersion(obj runtime.Object) (uint64, error)

    // ParseResourceVersion takes a resource version argument and
    // converts it to the storage backend. For watch we should pass to helper.Watch().
    // Because resourceVersion is an opaque value, the default watch
    // behavior for non-zero watch is to watch the next value (if you pass
    // "1", you will see updates from "2" onwards).
    ParseResourceVersion(resourceVersion string) (uint64, error)
}

APIObjectVersioner

Versioner的实现api_object_versioner.go

// APIObjectVersioner implements versioning and extracting etcd node information
// for objects that have an embedded ObjectMeta or ListMeta field.
type APIObjectVersioner struct{}

// UpdateObject implements Versioner
func (a APIObjectVersioner) UpdateObject(obj runtime.Object, resourceVersion uint64) error {
    accessor, err := meta.Accessor(obj)
    if err != nil {
        return err
    }
    versionString := ""
    if resourceVersion != 0 {
        versionString = strconv.FormatUint(resourceVersion, 10)
    }
    accessor.SetResourceVersion(versionString)
    return nil
}
//...省略

根据传入的resourceVersion更新kubernetes 资源中resourceVersion,那么具体resourceVersion是怎么来的呢?回到interfaces.go文件看下storage操作相关的接口定义:

Interface

// Interface offers a common interface for object marshaling/unmarshaling operations and
// hides all the storage-related operations behind it.
type Interface interface {
    // Returns Versioner associated with this interface.
    Versioner() Versioner

    // Create adds a new object at a key unless it already exists. 'ttl' is time-to-live
    // in seconds (0 means forever). If no error is returned and out is not nil, out will be
    // set to the read value from database.
    Create(ctx context.Context, key string, obj, out runtime.Object, ttl uint64) error

    // Delete removes the specified key and returns the value that existed at that spot.
    // If key didn't exist, it will return NotFound storage error.
    // If 'cachedExistingObject' is non-nil, it can be used as a suggestion about the
    // current version of the object to avoid read operation from storage to get it.
    // However, the implementations have to retry in case suggestion is stale.
    Delete(
        ctx context.Context, key string, out runtime.Object, preconditions *Preconditions,
        validateDeletion ValidateObjectFunc, cachedExistingObject runtime.Object) error

    // Watch begins watching the specified key. Events are decoded into API objects,
    // and any items selected by 'p' are sent down to returned watch.Interface.
    // resourceVersion may be used to specify what version to begin watching,
    // which should be the current resourceVersion, and no longer rv+1
    // (e.g. reconnecting without missing any updates).
    // If resource version is "0", this interface will get current object at given key
    // and send it in an "ADDED" event, before watch starts.
    Watch(ctx context.Context, key string, opts ListOptions) (watch.Interface, error)

    // WatchList begins watching the specified key's items. Items are decoded into API
    // objects and any item selected by 'p' are sent down to returned watch.Interface.
    // resourceVersion may be used to specify what version to begin watching,
    // which should be the current resourceVersion, and no longer rv+1
    // (e.g. reconnecting without missing any updates).
    // If resource version is "0", this interface will list current objects directory defined by key
    // and send them in "ADDED" events, before watch starts.
    WatchList(ctx context.Context, key string, opts ListOptions) (watch.Interface, error)

    // Get unmarshals json found at key into objPtr. On a not found error, will either
    // return a zero object of the requested type, or an error, depending on 'opts.ignoreNotFound'.
    // Treats empty responses and nil response nodes exactly like a not found error.
    // The returned contents may be delayed, but it is guaranteed that they will
    // match 'opts.ResourceVersion' according 'opts.ResourceVersionMatch'.
    Get(ctx context.Context, key string, opts GetOptions, objPtr runtime.Object) error

    // GetToList unmarshals json found at key and opaque it into *List api object
    // (an object that satisfies the runtime.IsList definition).
    // The returned contents may be delayed, but it is guaranteed that they will
    // match 'opts.ResourceVersion' according 'opts.ResourceVersionMatch'.
    GetToList(ctx context.Context, key string, opts ListOptions, listObj runtime.Object) error

    // List unmarshalls jsons found at directory defined by key and opaque them
    // into *List api object (an object that satisfies runtime.IsList definition).
    // The returned contents may be delayed, but it is guaranteed that they will
    // match 'opts.ResourceVersion' according 'opts.ResourceVersionMatch'.
    List(ctx context.Context, key string, opts ListOptions, listObj runtime.Object) error

    // GuaranteedUpdate keeps calling 'tryUpdate()' to update key 'key' (of type 'ptrToType')
    // retrying the update until success if there is index conflict.
    // Note that object passed to tryUpdate may change across invocations of tryUpdate() if
    // other writers are simultaneously updating it, so tryUpdate() needs to take into account
    // the current contents of the object when deciding how the update object should look.
    // If the key doesn't exist, it will return NotFound storage error if ignoreNotFound=false
    // or zero value in 'ptrToType' parameter otherwise.
    // If the object to update has the same value as previous, it won't do any update
    // but will return the object in 'ptrToType' parameter.
    // If 'cachedExistingObject' is non-nil, it can be used as a suggestion about the
    // current version of the object to avoid read operation from storage to get it.
    // However, the implementations have to retry in case suggestion is stale.
    //
    // Example:
    //
    // s := /* implementation of Interface */
    // err := s.GuaranteedUpdate(
    //     "myKey", &MyType{}, true,
    //     func(input runtime.Object, res ResponseMeta) (runtime.Object, *uint64, error) {
    //       // Before each invocation of the user defined function, "input" is reset to
    //       // current contents for "myKey" in database.
    //       curr := input.(*MyType)  // Guaranteed to succeed.
    //
    //       // Make the modification
    //       curr.Counter++
    //
    //       // Return the modified object - return an error to stop iterating. Return
    //       // a uint64 to alter the TTL on the object, or nil to keep it the same value.
    //       return cur, nil, nil
    //    },
    // )
    GuaranteedUpdate(
        ctx context.Context, key string, ptrToType runtime.Object, ignoreNotFound bool,
        preconditions *Preconditions, tryUpdate UpdateFunc, cachedExistingObject runtime.Object) error

    // Count returns number of different entries under the key (generally being path prefix).
    Count(key string) (int64, error)
}

数据的CURD等其它,这里我们只选Get来分析下,看下它的实现:

Get

// Get implements storage.Interface.Get.
func (s *store) Get(ctx context.Context, key string, opts storage.GetOptions, out runtime.Object) error {
    key = path.Join(s.pathPrefix, key)
    startTime := time.Now()
    getResp, err := s.client.KV.Get(ctx, key)
    metrics.RecordEtcdRequestLatency("get", getTypeName(out), startTime)
    if err != nil {
        return err
    }
    if err = s.validateMinimumResourceVersion(opts.ResourceVersion, uint64(getResp.Header.Revision)); err != nil {
        return err
    }

    if len(getResp.Kvs) == 0 {
        if opts.IgnoreNotFound {
            return runtime.SetZeroValue(out)
        }
        return storage.NewKeyNotFoundError(key, 0)
    }
    kv := getResp.Kvs[0]

    data, _, err := s.transformer.TransformFromStorage(kv.Value, authenticatedDataString(key))
    if err != nil {
        return storage.NewInternalError(err.Error())
    }

    return decode(s.codec, s.versioner, data, out, kv.ModRevision)
}

kv.ModRevision看到从ETCD读取了key的ModRevision,继续看下decode函数

decode

// decode decodes value of bytes into object. It will also set the object resource version to rev.
// On success, objPtr would be set to the object.
func decode(codec runtime.Codec, versioner storage.Versioner, value []byte, objPtr runtime.Object, rev int64) error {
    if _, err := conversion.EnforcePtr(objPtr); err != nil {
        return fmt.Errorf("unable to convert output object to pointer: %v", err)
    }
    _, _, err := codec.Decode(value, nil, objPtr)
    if err != nil {
        return err
    }
    // being unable to set the version does not prevent the object from being extracted
    if err := versioner.UpdateObject(objPtr, uint64(rev)); err != nil {
        klog.Errorf("failed to update object version: %v", err)
    }
    return nil
}

调用Versioner接口更新资源对象的resourceVersion。

到这里已经可出Kubernetes的resourceVersion是利用了底层ETCD kv版本机制。

Etcd Version

ETCD共四种version

  • Revision
  • ModRevision
  • Version
  • CreateRevision
    关于他们的区别可以看下这个issue:what is different about Revision, ModRevision and Version?

the Revision is the current revision of etcd. It is incremented every time the v3 backed is modified (e.g., Put, Delete, Txn). ModRevision is the etcd revision of the last update to a key. Version is the number of times the key has been modified since it was created. Get(..., WithRev(rev)) will perform a Get as if the etcd store is still at revision rev.

字段 作用范围 说明
Version Key 单个Key的修改次数,单调递增
Revision 全局 Key 在集群中的全局版本号,全局唯一
ModRevison Key Key 最后一次修改时的 Revision
CreateRevision 全局 Key 创建时的 Revision

我们使用docker快速启动一个测试etcd来验证下

rm -rf /tmp/etcd-data.tmp && mkdir -p /tmp/etcd-data.tmp && \
  docker rmi quay.io/coreos/etcd:v3.5.1 || true && \
  docker run \
  -d \
  -p 2379:2379 \
  -p 2380:2380 \
  --mount type=bind,source=/tmp/etcd-data.tmp,destination=/etcd-data \
  --name etcd-gcr-v3.5.1 \
  quay.io/coreos/etcd:v3.5.1 \
  /usr/local/bin/etcd \
  --name s1 \
  --data-dir /etcd-data \
  --listen-client-urls http://0.0.0.0:2379 \
  --advertise-client-urls http://0.0.0.0:2379 \
  --listen-peer-urls http://0.0.0.0:2380 \
  --initial-advertise-peer-urls http://0.0.0.0:2380 \
  --initial-cluster s1=http://0.0.0.0:2380 \
  --initial-cluster-token tkn \
  --initial-cluster-state new \
  --log-level info \
  --logger zap \
  --log-outputs stderr 

插入、更新数据,查看相关version的变化

root@641fe4972263:/# etcdctl put k1 v1
OK
root@641fe4972263:/# etcdctl get k1 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 6,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azE=",
      "create_revision": 6,
      "mod_revision": 6,
      "version": 1,
      "value": "djE="
    }
  ],
  "count": 1
}
root@641fe4972263:/# etcdctl put k2 v2
OK
root@641fe4972263:/# etcdctl get k2 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 7,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azI=",
      "create_revision": 7,
      "mod_revision": 7,
      "version": 1,
      "value": "djI="
    }
  ],
  "count": 1
}
root@641fe4972263:/# etcdctl get k1 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 7,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azE=",
      "create_revision": 6,
      "mod_revision": 6,
      "version": 1,
      "value": "djE="
    }
  ],
  "count": 1
}
root@641fe4972263:/# etcdctl put k1 nv1
OK
root@641fe4972263:/# etcdctl get k1 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 8,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azE=",
      "create_revision": 6,
      "mod_revision": 8,
      "version": 2,
      "value": "bnYx"
    }
  ],
  "count": 1
}
root@641fe4972263:/# etcdctl get k2 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 8,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azI=",
      "create_revision": 7,
      "mod_revision": 7,
      "version": 1,
      "value": "djI="
    }
  ],
  "count": 1
}

删除key并查看相关version的变化

root@641fe4972263:/# etcdctl del k1
1
root@641fe4972263:/# etcdctl get k1
root@641fe4972263:/# etcdctl get k1 -w json
{"header":{"cluster_id":18011104697467366872,"member_id":6460912315094810421,"revision":9,"raft_term":3}}

root@641fe4972263:/# etcdctl get k1 --rev=6 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 9,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azE=",
      "create_revision": 6,
      "mod_revision": 6,
      "version": 1,
      "value": "djE="
    }
  ],
  "count": 1
}
root@641fe4972263:/# etcdctl put k1 dnv1
OK
root@641fe4972263:/# etcdctl get k1 -w json|jq
{
  "header": {
    "cluster_id": 18011104697467367000,
    "member_id": 6460912315094811000,
    "revision": 10,
    "raft_term": 3
  },
  "kvs": [
    {
      "key": "azE=",
      "create_revision": 10,
      "mod_revision": 10,
      "version": 1,
      "value": "ZG52MQ=="
    }
  ],
  "count": 1
}

k8s 资源对象版本冲突处理

我们日常使用kubectl apply执行更新可能遇到过下边的错误信息:

[root@xnile]$ kubectl apply -f nginx-test.yaml
Error from server (Conflict): error when applying patch:
{"metadata":{"generation":7,"resourceVersion":"3313340114"},"status":{"observedGeneration":7}}
to:
Resource: "apps/v1, Resource=deployments", GroupVersionKind: "apps/v1, Kind=Deployment"
Name: "nginx-test", Namespace: "default"
for: "nginx-test.yaml": Operation cannot be fulfilled on deployments.apps "nginx-test": the object has been modified; please apply your changes to the latest version and try again

这种情况通常会出现在我们先用kubectl get deploy xxx -o > xxx.yml导出当前配置,修改后再apply应该更改的时候。

对象版本冲突错误提示信息定义在这里store.go#L230

//https://github.com/kubernetes/kubernetes/blob/master/staging/src/k8s.io/apiserver/pkg/registry/generic/registry/store.go#L230
OptimisticLockErrorMsg        = "the object has been modified; please apply your changes to the latest version and try again"

更新冲突错误提示信息定义我们来到store.go#L496

// Update performs an atomic update and set of the object. Returns the result of the update
// or an error. If the registry allows create-on-update, the create flow will be executed.
// A bool is returned along with the object and any errors, to indicate object creation.
func (e *Store) Update(ctx context.Context, name string, objInfo rest.UpdatedObjectInfo, createValidation rest.ValidateObjectFunc, updateValidation rest.ValidateObjectUpdateFunc, forceAllowCreate bool, options *metav1.UpdateOptions) (runtime.Object, bool, error) {
    key, err := e.KeyFunc(ctx, name)
    if err != nil {
        return nil, false, err
    }

    var (
        creatingObj runtime.Object
        creating    = false
    )

    qualifiedResource := e.qualifiedResourceFromContext(ctx)
    storagePreconditions := &storage.Preconditions{}
    if preconditions := objInfo.Preconditions(); preconditions != nil {
        storagePreconditions.UID = preconditions.UID
        storagePreconditions.ResourceVersion = preconditions.ResourceVersion
    }

    out := e.NewFunc()
    // deleteObj is only used in case a deletion is carried out
    var deleteObj runtime.Object
    err = e.Storage.GuaranteedUpdate(ctx, key, out, true, storagePreconditions, func(existing runtime.Object, res storage.ResponseMeta) (runtime.Object, *uint64, error) {
        // 获取etcd中当前资源的最新版本的数据
        existingResourceVersion, err := e.Storage.Versioner().ObjectResourceVersion(existing)
        if err != nil {
            return nil, nil, err
        }
        if existingResourceVersion == 0 {
            if !e.UpdateStrategy.AllowCreateOnUpdate() && !forceAllowCreate {
                return nil, nil, apierrors.NewNotFound(qualifiedResource, name)
            }
        }

        // Given the existing object, get the new object
        obj, err := objInfo.UpdatedObject(ctx, existing)
        if err != nil {
            return nil, nil, err
        }

        // If AllowUnconditionalUpdate() is true and the object specified by
        // the user does not have a resource version, then we populate it with
        // the latest version. Else, we check that the version specified by
        // the user matches the version of latest storage object.
        newResourceVersion, err := e.Storage.Versioner().ObjectResourceVersion(obj)
        if err != nil {
            return nil, nil, err
        }
        // 无条件更新,通常是yaml文件中不带resourceVersion,然后就在etcd最新版本数据的基础上进行更新
        doUnconditionalUpdate := newResourceVersion == 0 && e.UpdateStrategy.AllowUnconditionalUpdate()

        if existingResourceVersion == 0 {
            var finishCreate FinishFunc = finishNothing

            if e.BeginCreate != nil {
                fn, err := e.BeginCreate(ctx, obj, newCreateOptionsFromUpdateOptions(options))
                if err != nil {
                    return nil, nil, err
                }
                finishCreate = fn
                defer func() {
                    finishCreate(ctx, false)
                }()
            }

            creating = true
            creatingObj = obj
            if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil {
                return nil, nil, err
            }
            // at this point we have a fully formed object.  It is time to call the validators that the apiserver
            // handling chain wants to enforce.
            if createValidation != nil {
                if err := createValidation(ctx, obj.DeepCopyObject()); err != nil {
                    return nil, nil, err
                }
            }
            ttl, err := e.calculateTTL(obj, 0, false)
            if err != nil {
                return nil, nil, err
            }

            // The operation has succeeded.  Call the finish function if there is one,
            // and then make sure the defer doesn't call it again.
            fn := finishCreate
            finishCreate = finishNothing
            fn(ctx, true)

            return obj, &ttl, nil
        }

        creating = false
        creatingObj = nil
        if doUnconditionalUpdate {
            // Update the object's resource version to match the latest
            // storage object's resource version.
            err = e.Storage.Versioner().UpdateObject(obj, res.ResourceVersion)
            if err != nil {
                return nil, nil, err
            }
        } else {
            // Check if the object's resource version matches the latest
            // resource version.
            if newResourceVersion == 0 {
                // TODO: The Invalid error should have a field for Resource.
                // After that field is added, we should fill the Resource and
                // leave the Kind field empty. See the discussion in #18526.
                qualifiedKind := schema.GroupKind{Group: qualifiedResource.Group, Kind: qualifiedResource.Resource}
                fieldErrList := field.ErrorList{field.Invalid(field.NewPath("metadata").Child("resourceVersion"), newResourceVersion, "must be specified for an update")}
                return nil, nil, apierrors.NewInvalid(qualifiedKind, name, fieldErrList)
            }
            // 数据已经发生变化,提示版本冲突
            if newResourceVersion != existingResourceVersion {
                return nil, nil, apierrors.NewConflict(qualifiedResource, name, fmt.Errorf(OptimisticLockErrorMsg))
            }
        }

        var finishUpdate FinishFunc = finishNothing

        if e.BeginUpdate != nil {
            fn, err := e.BeginUpdate(ctx, obj, existing, options)
            if err != nil {
                return nil, nil, err
            }
            finishUpdate = fn
            defer func() {
                finishUpdate(ctx, false)
            }()
        }

        if err := rest.BeforeUpdate(e.UpdateStrategy, ctx, obj, existing); err != nil {
            return nil, nil, err
        }
        // at this point we have a fully formed object.  It is time to call the validators that the apiserver
        // handling chain wants to enforce.
        if updateValidation != nil {
            if err := updateValidation(ctx, obj.DeepCopyObject(), existing.DeepCopyObject()); err != nil {
                return nil, nil, err
            }
        }
        // Check the default delete-during-update conditions, and store-specific conditions if provided
        if ShouldDeleteDuringUpdate(ctx, key, obj, existing) &&
            (e.ShouldDeleteDuringUpdate == nil || e.ShouldDeleteDuringUpdate(ctx, key, obj, existing)) {
            deleteObj = obj
            return nil, nil, errEmptiedFinalizers
        }
        ttl, err := e.calculateTTL(obj, res.TTL, true)
        if err != nil {
            return nil, nil, err
        }

        // The operation has succeeded.  Call the finish function if there is one,
        // and then make sure the defer doesn't call it again.
        fn := finishUpdate
        finishUpdate = finishNothing
        fn(ctx, true)

        if int64(ttl) != res.TTL {
            return obj, &ttl, nil
        }
        return obj, nil, nil
    }, dryrun.IsDryRun(options.DryRun), nil)

    if err != nil {
        // delete the object
        if err == errEmptiedFinalizers {
            return e.deleteWithoutFinalizers(ctx, name, key, deleteObj, storagePreconditions, newDeleteOptionsFromUpdateOptions(options))
        }
        if creating {
            err = storeerr.InterpretCreateError(err, qualifiedResource, name)
            err = rest.CheckGeneratedNameError(ctx, e.CreateStrategy, err, creatingObj)
        } else {
            err = storeerr.InterpretUpdateError(err, qualifiedResource, name)
        }
        return nil, false, err
    }

    if creating {
        if e.AfterCreate != nil {
            e.AfterCreate(out, newCreateOptionsFromUpdateOptions(options))
        }
    } else {
        if e.AfterUpdate != nil {
            e.AfterUpdate(out, options)
        }
    }
    if e.Decorator != nil {
        e.Decorator(out)
    }
    return out, creating, nil
}

根据更新资源时是否带有resourceVersion分两种情况:

未带resourceVersion:无条件更新,获得etcd中最新的数据然后再此基础上更新
带有resourceVersion:和etcd中modRevision对比,不一样就提示版本冲突,说明数据已发生修改,当前要修改的版本已不是最新数据。

Generation

func (deploymentStrategy) PrepareForCreate(ctx context.Context, obj runtime.Object) {
    deployment := obj.(*apps.Deployment)
    deployment.Status = apps.DeploymentStatus{}
    deployment.Generation = 1

    pod.DropDisabledTemplateFields(&deployment.Spec.Template, nil)
}

func (deploymentStrategy) PrepareForUpdate(ctx context.Context, obj, old runtime.Object) {
    newDeployment := obj.(*apps.Deployment)
    oldDeployment := old.(*apps.Deployment)
    newDeployment.Status = oldDeployment.Status

    pod.DropDisabledTemplateFields(&newDeployment.Spec.Template, &oldDeployment.Spec.Template)

    // Spec updates bump the generation so that we can distinguish between
    // scaling events and template changes, annotation updates bump the generation
    // because annotations are copied from deployments to their replica sets.
    if !apiequality.Semantic.DeepEqual(newDeployment.Spec, oldDeployment.Spec) ||
        !apiequality.Semantic.DeepEqual(newDeployment.Annotations, oldDeployment.Annotations) {
        newDeployment.Generation = oldDeployment.Generation + 1
    }
}

Generation初始值为1,随Spec内容的改变而自增。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容