2014-06-20 01:03:48 +00:00
|
|
|
/*
|
|
|
|
Copyright 2014 Google Inc. All rights reserved.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2014-08-11 07:34:59 +00:00
|
|
|
package minion
|
2014-06-20 01:03:48 +00:00
|
|
|
|
|
|
|
import (
|
2014-10-17 21:21:53 +00:00
|
|
|
"errors"
|
2014-06-20 01:03:48 +00:00
|
|
|
"fmt"
|
2014-10-17 21:21:53 +00:00
|
|
|
"net"
|
|
|
|
"strconv"
|
2014-06-20 01:03:48 +00:00
|
|
|
|
2014-06-20 21:01:16 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
2014-11-12 17:38:15 +00:00
|
|
|
kerrors "github.com/GoogleCloudPlatform/kubernetes/pkg/api/errors"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/validation"
|
2014-06-20 01:03:48 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/apiserver"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
|
2014-10-17 21:21:53 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/master/ports"
|
2014-09-06 02:22:03 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/runtime"
|
2014-11-26 00:25:23 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/watch"
|
2014-06-20 01:03:48 +00:00
|
|
|
)
|
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// REST adapts minion into apiserver's RESTStorage model.
|
2014-09-08 21:40:56 +00:00
|
|
|
type REST struct {
|
2014-08-11 07:34:59 +00:00
|
|
|
registry Registry
|
2014-06-20 01:03:48 +00:00
|
|
|
}
|
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// NewREST returns a new apiserver.RESTStorage implementation for minion.
|
2014-09-08 21:40:56 +00:00
|
|
|
func NewREST(m Registry) *REST {
|
|
|
|
return &REST{
|
2014-06-20 01:03:48 +00:00
|
|
|
registry: m,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-17 21:21:53 +00:00
|
|
|
var ErrDoesNotExist = errors.New("The requested resource does not exist.")
|
|
|
|
var ErrNotHealty = errors.New("The requested minion is not healthy.")
|
2014-09-09 03:02:21 +00:00
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// Create satisfies the RESTStorage interface.
|
2014-10-24 17:16:02 +00:00
|
|
|
func (rs *REST) Create(ctx api.Context, obj runtime.Object) (<-chan apiserver.RESTResult, error) {
|
2014-12-08 03:44:27 +00:00
|
|
|
minion, ok := obj.(*api.Node)
|
2014-06-25 23:23:15 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("not a minion: %#v", obj)
|
|
|
|
}
|
2014-11-12 17:38:15 +00:00
|
|
|
|
|
|
|
if errs := validation.ValidateMinion(minion); len(errs) > 0 {
|
|
|
|
return nil, kerrors.NewInvalid("minion", minion.Name, errs)
|
2014-06-25 23:23:15 +00:00
|
|
|
}
|
2014-08-12 19:14:00 +00:00
|
|
|
|
2014-11-12 21:27:10 +00:00
|
|
|
api.FillObjectMetaSystemFields(ctx, &minion.ObjectMeta)
|
2014-08-12 19:14:00 +00:00
|
|
|
|
2014-09-06 02:22:03 +00:00
|
|
|
return apiserver.MakeAsync(func() (runtime.Object, error) {
|
2014-10-06 20:37:46 +00:00
|
|
|
err := rs.registry.CreateMinion(ctx, minion)
|
2014-06-25 23:23:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-10-09 03:13:04 +00:00
|
|
|
return minion, nil
|
2014-06-25 23:23:15 +00:00
|
|
|
}), nil
|
2014-06-20 01:03:48 +00:00
|
|
|
}
|
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// Delete satisfies the RESTStorage interface.
|
2014-10-24 17:16:02 +00:00
|
|
|
func (rs *REST) Delete(ctx api.Context, id string) (<-chan apiserver.RESTResult, error) {
|
2014-10-09 03:13:04 +00:00
|
|
|
minion, err := rs.registry.GetMinion(ctx, id)
|
|
|
|
if minion == nil {
|
2014-06-20 01:03:48 +00:00
|
|
|
return nil, ErrDoesNotExist
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-09-06 02:22:03 +00:00
|
|
|
return apiserver.MakeAsync(func() (runtime.Object, error) {
|
2014-09-09 03:02:21 +00:00
|
|
|
return &api.Status{Status: api.StatusSuccess}, rs.registry.DeleteMinion(ctx, id)
|
2014-06-25 23:23:15 +00:00
|
|
|
}), nil
|
2014-06-20 01:03:48 +00:00
|
|
|
}
|
2014-08-11 07:34:59 +00:00
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// Get satisfies the RESTStorage interface.
|
2014-09-26 15:46:04 +00:00
|
|
|
func (rs *REST) Get(ctx api.Context, id string) (runtime.Object, error) {
|
2014-10-09 03:13:04 +00:00
|
|
|
minion, err := rs.registry.GetMinion(ctx, id)
|
|
|
|
if minion == nil {
|
2014-08-11 07:34:59 +00:00
|
|
|
return nil, ErrDoesNotExist
|
|
|
|
}
|
2014-10-09 03:13:04 +00:00
|
|
|
return minion, err
|
2014-08-11 07:34:59 +00:00
|
|
|
}
|
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// List satisfies the RESTStorage interface.
|
2014-09-26 15:46:04 +00:00
|
|
|
func (rs *REST) List(ctx api.Context, label, field labels.Selector) (runtime.Object, error) {
|
2014-09-09 03:02:21 +00:00
|
|
|
return rs.registry.ListMinions(ctx)
|
2014-08-11 07:34:59 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 03:02:21 +00:00
|
|
|
func (rs *REST) New() runtime.Object {
|
2014-12-08 03:44:27 +00:00
|
|
|
return &api.Node{}
|
2014-08-11 07:34:59 +00:00
|
|
|
}
|
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// Update satisfies the RESTStorage interface.
|
2014-11-15 21:32:59 +00:00
|
|
|
func (rs *REST) Update(ctx api.Context, obj runtime.Object) (<-chan apiserver.RESTResult, error) {
|
2014-12-08 03:44:27 +00:00
|
|
|
minion, ok := obj.(*api.Node)
|
2014-11-15 21:32:59 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("not a minion: %#v", obj)
|
|
|
|
}
|
2014-11-17 18:22:27 +00:00
|
|
|
|
|
|
|
// TODO: GetMinion will health check the minion, but we shouldn't require the minion to be
|
|
|
|
// running for updating labels.
|
|
|
|
oldMinion, err := rs.registry.GetMinion(ctx, minion.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if errs := validation.ValidateMinionUpdate(oldMinion, minion); len(errs) > 0 {
|
2014-11-15 21:32:59 +00:00
|
|
|
return nil, kerrors.NewInvalid("minion", minion.Name, errs)
|
|
|
|
}
|
2014-11-17 18:22:27 +00:00
|
|
|
|
2014-11-15 21:32:59 +00:00
|
|
|
return apiserver.MakeAsync(func() (runtime.Object, error) {
|
|
|
|
err := rs.registry.UpdateMinion(ctx, minion)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return rs.registry.GetMinion(ctx, minion.Name)
|
|
|
|
}), nil
|
2014-08-11 07:34:59 +00:00
|
|
|
}
|
|
|
|
|
2014-11-26 00:25:23 +00:00
|
|
|
// Watch returns Minions events via a watch.Interface.
|
|
|
|
// It implements apiserver.ResourceWatcher.
|
|
|
|
func (rs *REST) Watch(ctx api.Context, label, field labels.Selector, resourceVersion string) (watch.Interface, error) {
|
|
|
|
return rs.registry.WatchMinions(ctx, label, field, resourceVersion)
|
2014-08-11 07:34:59 +00:00
|
|
|
}
|
2014-10-17 21:21:53 +00:00
|
|
|
|
|
|
|
// ResourceLocation returns a URL to which one can send traffic for the specified minion.
|
|
|
|
func (rs *REST) ResourceLocation(ctx api.Context, id string) (string, error) {
|
|
|
|
minion, err := rs.registry.GetMinion(ctx, id)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2014-12-09 00:44:48 +00:00
|
|
|
host := minion.Name
|
2014-10-17 21:21:53 +00:00
|
|
|
// TODO: Minion webservers should be secure!
|
|
|
|
return "http://" + net.JoinHostPort(host, strconv.Itoa(ports.KubeletPort)), nil
|
|
|
|
}
|