
Contains breaking API change on api.Status#Details (type change) Turn Details from string -> StatusDetails - a general bucket for keyed error behavior. Define an open enumeration ReasonType exposed as Reason on the status object to provide machine readable subcategorization beyond HTTP Status Code. Define a human readable field Message which is common convention (previously this was joined into Details). Precedence order: HTTP Status Code, Reason, Details. apiserver would impose restraints on the ReasonTypes defined by the main apiobject, and ensure their use is consistent. There are four long term scenarios this change supports: 1. Allow a client access to a machine readable field that can be easily switched on for improving or translating the generic server Message. 2. Return a 404 when a composite operation on multiple resources fails with enough data so that a client can distinguish which item does not exist. E.g. resource Parent and resource Child, POST /parents/1/children to create a new Child, but /parents/1 is deleted. POST returns 404, ReasonTypeNotFound, and Details.ID = "1", Details.Kind = "parent" 3. Allow a client to receive validation data that is keyed by attribute for building user facing UIs around field submission. Validation is usually expressed as map[string][]string, but that type is less appropriate for many other uses. 4. Allow specific API errors to return more granular failure status for specific operations. An example might be a minion proxy, where the operation that failed may be both proxying OR the minion itself. In this case a reason may be defined "proxy_failed" corresponding to 502, where the Details field may be extended to contain a nested error object. At this time only ID and Kind are exposed
299 lines
7.1 KiB
Go
299 lines
7.1 KiB
Go
/*
|
|
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.
|
|
*/
|
|
|
|
package client
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"path"
|
|
"time"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/tools"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/watch"
|
|
"github.com/golang/glog"
|
|
)
|
|
|
|
// Server contains info locating a kubernetes api server.
|
|
// Example usage:
|
|
// auth, err := LoadAuth(filename)
|
|
// c := New(url, auth)
|
|
// resp, err := c.Verb("GET").
|
|
// Path("pods").
|
|
// Selector("area=staging").
|
|
// Timeout(10*time.Second).
|
|
// Do()
|
|
// list, ok := resp.(api.PodList)
|
|
|
|
// Verb begins a request with a verb (GET, POST, PUT, DELETE)
|
|
func (c *Client) Verb(verb string) *Request {
|
|
return &Request{
|
|
verb: verb,
|
|
c: c,
|
|
path: "/api/v1beta1",
|
|
sync: c.Sync,
|
|
timeout: c.Timeout,
|
|
pollPeriod: c.PollPeriod,
|
|
}
|
|
}
|
|
|
|
// Post begins a POST request.
|
|
func (c *Client) Post() *Request {
|
|
return c.Verb("POST")
|
|
}
|
|
|
|
// Put begins a PUT request.
|
|
func (c *Client) Put() *Request {
|
|
return c.Verb("PUT")
|
|
}
|
|
|
|
// Get begins a GET request.
|
|
func (c *Client) Get() *Request {
|
|
return c.Verb("GET")
|
|
}
|
|
|
|
// Delete begins a DELETE request.
|
|
func (c *Client) Delete() *Request {
|
|
return c.Verb("DELETE")
|
|
}
|
|
|
|
// PollFor makes a request to do a single poll of the completion of the given operation.
|
|
func (c *Client) PollFor(operationID string) *Request {
|
|
return c.Get().Path("operations").Path(operationID).Sync(false).PollPeriod(0)
|
|
}
|
|
|
|
// Request allows for building up a request to a server in a chained fashion.
|
|
// Any errors are stored until the end of your call, so you only have to
|
|
// check once.
|
|
type Request struct {
|
|
c *Client
|
|
err error
|
|
verb string
|
|
path string
|
|
body io.Reader
|
|
selector labels.Selector
|
|
timeout time.Duration
|
|
sync bool
|
|
pollPeriod time.Duration
|
|
}
|
|
|
|
// Path appends an item to the request path. You must call Path at least once.
|
|
func (r *Request) Path(item string) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.path = path.Join(r.path, item)
|
|
return r
|
|
}
|
|
|
|
// Sync sets sync/async call status.
|
|
func (r *Request) Sync(sync bool) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.sync = sync
|
|
return r
|
|
}
|
|
|
|
// AbsPath overwrites an existing path with the path parameter.
|
|
func (r *Request) AbsPath(path string) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.path = path
|
|
return r
|
|
}
|
|
|
|
// ParseSelector parses the given string as a resource label selector. Optional.
|
|
func (r *Request) ParseSelector(item string) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.selector, r.err = labels.ParseSelector(item)
|
|
return r
|
|
}
|
|
|
|
// Selector makes the request use the given selector.
|
|
func (r *Request) Selector(s labels.Selector) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.selector = s
|
|
return r
|
|
}
|
|
|
|
// Timeout makes the request use the given duration as a timeout. Optional.
|
|
func (r *Request) Timeout(d time.Duration) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.timeout = d
|
|
return r
|
|
}
|
|
|
|
// Body makes the request use obj as the body. Optional.
|
|
// If obj is a string, try to read a file of that name.
|
|
// If obj is a []byte, send it directly.
|
|
// Otherwise, assume obj is an api type and marshall it correctly.
|
|
func (r *Request) Body(obj interface{}) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
switch t := obj.(type) {
|
|
case string:
|
|
data, err := ioutil.ReadFile(t)
|
|
if err != nil {
|
|
r.err = err
|
|
return r
|
|
}
|
|
r.body = bytes.NewBuffer(data)
|
|
case []byte:
|
|
r.body = bytes.NewBuffer(t)
|
|
case io.Reader:
|
|
r.body = obj.(io.Reader)
|
|
default:
|
|
data, err := api.Encode(obj)
|
|
if err != nil {
|
|
r.err = err
|
|
return r
|
|
}
|
|
r.body = bytes.NewBuffer(data)
|
|
}
|
|
return r
|
|
}
|
|
|
|
// PollPeriod sets the poll period.
|
|
// If the server sends back a "working" status message, then repeatedly poll the server
|
|
// to see if the operation has completed yet, waiting 'd' between each poll.
|
|
// If you want to handle the "working" status yourself (it'll be delivered as StatusErr),
|
|
// set d to 0 to turn off this behavior.
|
|
func (r *Request) PollPeriod(d time.Duration) *Request {
|
|
if r.err != nil {
|
|
return r
|
|
}
|
|
r.pollPeriod = d
|
|
return r
|
|
}
|
|
|
|
func (r *Request) finalURL() string {
|
|
finalURL := r.c.host + r.path
|
|
query := url.Values{}
|
|
if r.selector != nil {
|
|
query.Add("labels", r.selector.String())
|
|
}
|
|
if r.sync {
|
|
query.Add("sync", "true")
|
|
if r.timeout != 0 {
|
|
query.Add("timeout", r.timeout.String())
|
|
}
|
|
}
|
|
finalURL += "?" + query.Encode()
|
|
return finalURL
|
|
}
|
|
|
|
// Attempts to begin watching the requested location. Returns a watch.Interface, or an error.
|
|
func (r *Request) Watch() (watch.Interface, error) {
|
|
if r.err != nil {
|
|
return nil, r.err
|
|
}
|
|
req, err := http.NewRequest(r.verb, r.finalURL(), r.body)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if r.c.auth != nil {
|
|
req.SetBasicAuth(r.c.auth.User, r.c.auth.Password)
|
|
}
|
|
response, err := r.c.httpClient.Do(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if response.StatusCode != http.StatusOK {
|
|
return nil, fmt.Errorf("Got status: %v", response.StatusCode)
|
|
}
|
|
return watch.NewStreamWatcher(tools.NewAPIEventDecoder(response.Body)), nil
|
|
}
|
|
|
|
// Do formats and executes the request. Returns the API object received, or an error.
|
|
func (r *Request) Do() Result {
|
|
for {
|
|
if r.err != nil {
|
|
return Result{err: r.err}
|
|
}
|
|
req, err := http.NewRequest(r.verb, r.finalURL(), r.body)
|
|
if err != nil {
|
|
return Result{err: err}
|
|
}
|
|
respBody, err := r.c.doRequest(req)
|
|
if err != nil {
|
|
if statusErr, ok := err.(*StatusErr); ok {
|
|
if statusErr.Status.Status == api.StatusWorking && r.pollPeriod != 0 {
|
|
if statusErr.Status.Details != nil {
|
|
id := statusErr.Status.Details.ID
|
|
if len(id) > 0 {
|
|
glog.Infof("Waiting for completion of /operations/%s", id)
|
|
time.Sleep(r.pollPeriod)
|
|
// Make a poll request
|
|
pollOp := r.c.PollFor(id).PollPeriod(r.pollPeriod)
|
|
// Could also say "return r.Do()" but this way doesn't grow the callstack.
|
|
r = pollOp
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return Result{respBody, err}
|
|
}
|
|
}
|
|
|
|
// Result contains the result of calling Request.Do().
|
|
type Result struct {
|
|
body []byte
|
|
err error
|
|
}
|
|
|
|
// Raw returns the raw result.
|
|
func (r Result) Raw() ([]byte, error) {
|
|
return r.body, r.err
|
|
}
|
|
|
|
// Get returns the result as an object.
|
|
func (r Result) Get() (interface{}, error) {
|
|
if r.err != nil {
|
|
return nil, r.err
|
|
}
|
|
return api.Decode(r.body)
|
|
}
|
|
|
|
// Into stores the result into obj, if possible..
|
|
func (r Result) Into(obj interface{}) error {
|
|
if r.err != nil {
|
|
return r.err
|
|
}
|
|
return api.DecodeInto(r.body, obj)
|
|
}
|
|
|
|
// Returns the error executing the request, nil if no error occurred.
|
|
func (r Result) Error() error {
|
|
return r.err
|
|
}
|