// CreateBuild creates a new Build. func (r *Etcd) CreateBuild(ctx kapi.Context, build *api.Build) error { key, err := makeBuildKey(ctx, build.Name) if err != nil { return err } err = r.CreateObj(key, build, nil, 0) return etcderr.InterpretCreateError(err, "build", build.Name) }
// CreateBuildConfig creates a new BuildConfig. func (r *Etcd) CreateBuildConfig(ctx kapi.Context, config *api.BuildConfig) error { key, err := makeBuildConfigKey(ctx, config.Name) if err != nil { return err } err = r.CreateObj(key, config, nil, 0) return etcderr.InterpretCreateError(err, "buildConfig", config.Name) }
// CreateController creates a new ReplicationController. func (r *Registry) CreateController(ctx api.Context, controller *api.ReplicationController) error { key, err := makeControllerKey(ctx, controller.Name) if err != nil { return err } err = r.CreateObj(key, controller, 0) return etcderr.InterpretCreateError(err, "replicationController", controller.Name) }
// CreateService creates a new Service. func (r *Registry) CreateService(ctx api.Context, svc *api.Service) error { key, err := makeServiceKey(ctx, svc.Name) if err != nil { return err } err = r.CreateObj(key, svc, 0) return etcderr.InterpretCreateError(err, "service", svc.Name) }
// Create stores the object with a ttl, so that events don't stay in the system forever. func (r registry) Create(ctx api.Context, id string, obj runtime.Object) error { key, err := r.Etcd.KeyFunc(ctx, id) if err != nil { return err } err = r.Etcd.Helper.CreateObj(key, obj, r.ttl) return etcderr.InterpretCreateError(err, r.Etcd.EndpointName, id) }
// Create inserts a new item. func (e *Etcd) Create(ctx api.Context, id string, obj runtime.Object) error { key, err := e.KeyFunc(ctx, id) if err != nil { return err } err = e.Helper.CreateObj(key, obj, 0) return etcderr.InterpretCreateError(err, e.EndpointName, id) }
// CreateRoute creates a new Route. func (registry *Etcd) CreateRoute(ctx kapi.Context, route *api.Route) error { key, err := makeRouteKey(ctx, route.Name) if err != nil { return err } err = registry.Create(key, route, nil, 0) return etcderr.InterpretCreateError(err, "route", route.Name) }
// CreateController creates a new ReplicationController. func (r *Registry) CreateController(ctx api.Context, controller *api.ReplicationController) (*api.ReplicationController, error) { key, err := makeControllerKey(ctx, controller.Name) if err != nil { return nil, err } out := &api.ReplicationController{} err = r.CreateObj(key, controller, out, 0) return out, etcderr.InterpretCreateError(err, "replicationController", controller.Name) }
// CreateService creates a new Service. func (r *Registry) CreateService(ctx api.Context, svc *api.Service) (*api.Service, error) { key, err := makeServiceKey(ctx, svc.Name) if err != nil { return nil, err } out := &api.Service{} err = r.CreateObj(key, svc, out, 0) return out, etcderr.InterpretCreateError(err, "service", svc.Name) }
// CreatePod creates a pod based on a specification. func (r *Registry) CreatePod(pod *api.Pod) error { // Set current status to "Waiting". pod.CurrentState.Status = api.PodWaiting pod.CurrentState.Host = "" // DesiredState.Host == "" is a signal to the scheduler that this pod needs scheduling. pod.DesiredState.Status = api.PodRunning pod.DesiredState.Host = "" err := r.CreateObj(makePodKey(pod.ID), pod) return etcderr.InterpretCreateError(err, "pod", pod.ID) }
// CreatePod creates a pod based on a specification. func (r *Registry) CreatePod(ctx api.Context, pod *api.Pod) error { // Set current status to "Waiting". pod.Status.Phase = api.PodPending pod.Status.Host = "" key, err := makePodKey(ctx, pod.Name) if err != nil { return err } err = r.CreateObj(key, pod, 0) return etcderr.InterpretCreateError(err, "pod", pod.Name) }
// CreatePod creates a pod based on a specification. func (r *Registry) CreatePod(ctx api.Context, pod *api.Pod) error { // Set current status to "Waiting". pod.CurrentState.Status = api.PodPending pod.CurrentState.Host = "" // DesiredState.Host == "" is a signal to the scheduler that this pod needs scheduling. pod.DesiredState.Status = api.PodRunning pod.DesiredState.Host = "" key, err := makePodKey(ctx, pod.Name) if err != nil { return err } err = r.CreateObj(key, pod, 0) return etcderr.InterpretCreateError(err, "pod", pod.Name) }
// Create inserts a new item according to the unique key from the object. func (e *Etcd) Create(ctx api.Context, obj runtime.Object) (runtime.Object, error) { trace := util.NewTrace("Create") defer trace.LogIfLong(time.Second) if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil { return nil, err } name, err := e.ObjectNameFunc(obj) if err != nil { return nil, err } key, err := e.KeyFunc(ctx, name) if err != nil { return nil, err } ttl := uint64(0) if e.TTLFunc != nil { ttl, err = e.TTLFunc(obj, false) if err != nil { return nil, err } } trace.Step("About to create object") out := e.NewFunc() if err := e.Helper.CreateObj(key, obj, out, ttl); err != nil { err = etcderr.InterpretCreateError(err, e.EndpointName, name) err = rest.CheckGeneratedNameError(e.CreateStrategy, err, obj) return nil, err } trace.Step("Object created") if e.AfterCreate != nil { if err := e.AfterCreate(out); err != nil { return nil, err } } if e.Decorator != nil { if err := e.Decorator(obj); err != nil { return nil, err } } return out, nil }
// CreateWithName inserts a new item with the provided name // DEPRECATED: use Create instead func (e *Etcd) CreateWithName(ctx api.Context, name string, obj runtime.Object) error { key, err := e.KeyFunc(ctx, name) if err != nil { return err } if e.CreateStrategy != nil { if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil { return err } } ttl, err := e.calculateTTL(obj, 0, false) if err != nil { return err } err = e.Helper.CreateObj(key, obj, nil, ttl) err = etcderr.InterpretCreateError(err, e.EndpointName, name) if err == nil && e.Decorator != nil { err = e.Decorator(obj) } return err }
// Create inserts a new item according to the unique key from the object. func (e *Etcd) Create(ctx api.Context, obj runtime.Object) (runtime.Object, error) { if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil { return nil, err } name, err := e.ObjectNameFunc(obj) if err != nil { return nil, err } key, err := e.KeyFunc(ctx, name) if err != nil { return nil, err } ttl := uint64(0) if e.TTLFunc != nil { ttl, err = e.TTLFunc(obj, false) if err != nil { return nil, err } } out := e.NewFunc() if err := e.Helper.CreateObj(key, obj, out, ttl); err != nil { err = etcderr.InterpretCreateError(err, e.EndpointName, name) err = rest.CheckGeneratedNameError(e.CreateStrategy, err, obj) return nil, err } if e.AfterCreate != nil { if err := e.AfterCreate(out); err != nil { return nil, err } } if e.Decorator != nil { if err := e.Decorator(obj); err != nil { return nil, err } } return out, nil }
// Create stores the object with a ttl, so that events don't stay in the system forever. func (r registry) Create(ctx api.Context, id string, obj runtime.Object) error { err := r.Etcd.Helper.CreateObj(r.Etcd.KeyFunc(id), obj, r.ttl) return etcderr.InterpretCreateError(err, r.Etcd.EndpointName, id) }
// Create inserts a new item. func (e *Etcd) Create(ctx api.Context, id string, obj runtime.Object) error { err := e.Helper.CreateObj(e.KeyFunc(id), obj, 0) return etcderr.InterpretCreateError(err, e.EndpointName, id) }
func (r *Registry) CreateMinion(ctx api.Context, minion *api.Node) error { // TODO: Add some validations. err := r.CreateObj(makeNodeKey(minion.Name), minion, nil, 0) return etcderr.InterpretCreateError(err, "minion", minion.Name) }
// 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 *Etcd) Update(ctx api.Context, obj runtime.Object) (runtime.Object, bool, error) { trace := util.NewTrace("Update " + reflect.TypeOf(obj).String()) defer trace.LogIfLong(time.Second) name, err := e.ObjectNameFunc(obj) if err != nil { return nil, false, err } key, err := e.KeyFunc(ctx, name) if err != nil { return nil, false, 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 etcd object. resourceVersion, err := e.Helper.Versioner.ObjectResourceVersion(obj) if err != nil { return nil, false, err } doUnconditionalUpdate := resourceVersion == 0 && e.UpdateStrategy.AllowUnconditionalUpdate() // TODO: expose TTL creating := false out := e.NewFunc() err = e.Helper.GuaranteedUpdate(key, out, true, func(existing runtime.Object, res tools.ResponseMeta) (runtime.Object, *uint64, error) { version, err := e.Helper.Versioner.ObjectResourceVersion(existing) if err != nil { return nil, nil, err } if version == 0 { if !e.UpdateStrategy.AllowCreateOnUpdate() { return nil, nil, kubeerr.NewNotFound(e.EndpointName, name) } creating = true if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil { return nil, nil, err } ttl, err := e.calculateTTL(obj, 0, false) if err != nil { return nil, nil, err } return obj, &ttl, nil } creating = false if doUnconditionalUpdate { // Update the object's resource version to match the latest etcd object's resource version. err = e.Helper.Versioner.UpdateObject(obj, res.Expiration, res.ResourceVersion) if err != nil { return nil, nil, err } } else { // Check if the object's resource version matches the latest resource version. newVersion, err := e.Helper.Versioner.ObjectResourceVersion(obj) if err != nil { return nil, nil, err } if newVersion != version { return nil, nil, kubeerr.NewConflict(e.EndpointName, name, fmt.Errorf("the object has been modified; please apply your changes to the latest version and try again")) } } if err := rest.BeforeUpdate(e.UpdateStrategy, ctx, obj, existing); err != nil { return nil, nil, err } ttl, err := e.calculateTTL(obj, res.TTL, true) if err != nil { return nil, nil, err } if int64(ttl) != res.TTL { return obj, &ttl, nil } return obj, nil, nil }) if err != nil { if creating { err = etcderr.InterpretCreateError(err, e.EndpointName, name) err = rest.CheckGeneratedNameError(e.CreateStrategy, err, obj) } else { err = etcderr.InterpretUpdateError(err, e.EndpointName, name) } return nil, false, err } if creating { if e.AfterCreate != nil { if err := e.AfterCreate(out); err != nil { return nil, false, err } } } else { if e.AfterUpdate != nil { if err := e.AfterUpdate(out); err != nil { return nil, false, err } } } if e.Decorator != nil { if err := e.Decorator(obj); err != nil { return nil, false, err } } return out, creating, nil }
// 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 *Etcd) Update(ctx api.Context, obj runtime.Object) (runtime.Object, bool, error) { name, err := e.ObjectNameFunc(obj) if err != nil { return nil, false, err } key, err := e.KeyFunc(ctx, name) if err != nil { return nil, false, err } // TODO: expose TTL creating := false out := e.NewFunc() err = e.Helper.GuaranteedUpdate(key, out, true, func(existing runtime.Object) (runtime.Object, uint64, error) { version, err := e.Helper.Versioner.ObjectResourceVersion(existing) if err != nil { return nil, 0, err } if version == 0 { if !e.UpdateStrategy.AllowCreateOnUpdate() { return nil, 0, kubeerr.NewNotFound(e.EndpointName, name) } creating = true if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil { return nil, 0, err } ttl := uint64(0) if e.TTLFunc != nil { ttl, err = e.TTLFunc(obj, true) if err != nil { return nil, 0, err } } return obj, ttl, nil } creating = false newVersion, err := e.Helper.Versioner.ObjectResourceVersion(obj) if err != nil { return nil, 0, err } if newVersion != version { // TODO: return the most recent version to a client? return nil, 0, kubeerr.NewConflict(e.EndpointName, name, fmt.Errorf("the resource was updated to %d", version)) } if err := rest.BeforeUpdate(e.UpdateStrategy, ctx, obj, existing); err != nil { return nil, 0, err } ttl := uint64(0) if e.TTLFunc != nil { ttl, err = e.TTLFunc(obj, false) if err != nil { return nil, 0, err } } return obj, ttl, nil }) if err != nil { if creating { err = etcderr.InterpretCreateError(err, e.EndpointName, name) err = rest.CheckGeneratedNameError(e.CreateStrategy, err, obj) } else { err = etcderr.InterpretUpdateError(err, e.EndpointName, name) } return nil, false, err } if creating { if e.AfterCreate != nil { if err := e.AfterCreate(out); err != nil { return nil, false, err } } } else { if e.AfterUpdate != nil { if err := e.AfterUpdate(out); err != nil { return nil, false, err } } } if e.Decorator != nil { if err := e.Decorator(obj); err != nil { return nil, false, err } } return out, creating, nil }
// CreateService creates a new Service. func (r *Registry) CreateService(svc *api.Service) error { err := r.CreateObj(makeServiceKey(svc.ID), svc) return etcderr.InterpretCreateError(err, "service", svc.ID) }
// CreateController creates a new ReplicationController. func (r *Registry) CreateController(controller *api.ReplicationController) error { err := r.CreateObj(makeControllerKey(controller.ID), controller) return etcderr.InterpretCreateError(err, "replicationController", controller.ID) }
// ApplyBinding implements binding's registry func (r *Registry) ApplyBinding(binding *api.Binding) error { return etcderr.InterpretCreateError(r.assignPod(binding.PodID, binding.Host), "binding", "") }
// 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 *Etcd) Update(ctx api.Context, obj runtime.Object) (runtime.Object, bool, error) { trace := util.NewTrace("Update " + reflect.TypeOf(obj).String()) defer trace.LogIfLong(time.Second) name, err := e.ObjectNameFunc(obj) if err != nil { return nil, false, err } key, err := e.KeyFunc(ctx, name) if err != nil { return nil, false, err } // TODO: expose TTL creating := false out := e.NewFunc() err = e.Helper.GuaranteedUpdate(key, out, true, func(existing runtime.Object, res tools.ResponseMeta) (runtime.Object, *uint64, error) { version, err := e.Helper.Versioner.ObjectResourceVersion(existing) if err != nil { return nil, nil, err } if version == 0 { if !e.UpdateStrategy.AllowCreateOnUpdate() { return nil, nil, kubeerr.NewNotFound(e.EndpointName, name) } creating = true if err := rest.BeforeCreate(e.CreateStrategy, ctx, obj); err != nil { return nil, nil, err } ttl, err := e.calculateTTL(obj, 0, false) if err != nil { return nil, nil, err } return obj, &ttl, nil } creating = false newVersion, err := e.Helper.Versioner.ObjectResourceVersion(obj) if err != nil { return nil, nil, err } if newVersion != version { // TODO: return the most recent version to a client? return nil, nil, kubeerr.NewConflict(e.EndpointName, name, fmt.Errorf("the object has been modified; please apply your changes to the latest version and try again")) } if err := rest.BeforeUpdate(e.UpdateStrategy, ctx, obj, existing); err != nil { return nil, nil, err } ttl, err := e.calculateTTL(obj, res.TTL, true) if err != nil { return nil, nil, err } if int64(ttl) != res.TTL { return obj, &ttl, nil } return obj, nil, nil }) if err != nil { if creating { err = etcderr.InterpretCreateError(err, e.EndpointName, name) err = rest.CheckGeneratedNameError(e.CreateStrategy, err, obj) } else { err = etcderr.InterpretUpdateError(err, e.EndpointName, name) } return nil, false, err } if creating { if e.AfterCreate != nil { if err := e.AfterCreate(out); err != nil { return nil, false, err } } } else { if e.AfterUpdate != nil { if err := e.AfterUpdate(out); err != nil { return nil, false, err } } } if e.Decorator != nil { if err := e.Decorator(obj); err != nil { return nil, false, err } } return out, creating, nil }
// ApplyBinding implements binding's registry func (r *Registry) ApplyBinding(ctx api.Context, binding *api.Binding) error { return etcderr.InterpretCreateError(r.assignPod(ctx, binding), "binding", "") }