2014-06-22 19:05:34 +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-06-22 21:18:01 +00:00
|
|
|
package client
|
2014-06-22 19:05:34 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2014-07-17 23:09:29 +00:00
|
|
|
"fmt"
|
2014-06-22 19:05:34 +00:00
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"path"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
|
2014-07-20 20:41:15 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/tools"
|
2014-07-17 23:09:29 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/watch"
|
2014-07-09 05:35:58 +00:00
|
|
|
"github.com/golang/glog"
|
2014-06-22 19:05:34 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Server contains info locating a kubernetes api server.
|
|
|
|
// Example usage:
|
|
|
|
// auth, err := LoadAuth(filename)
|
2014-06-22 21:18:01 +00:00
|
|
|
// c := New(url, auth)
|
|
|
|
// resp, err := c.Verb("GET").
|
2014-06-22 19:05:34 +00:00
|
|
|
// Path("pods").
|
|
|
|
// Selector("area=staging").
|
|
|
|
// Timeout(10*time.Second).
|
|
|
|
// Do()
|
|
|
|
// list, ok := resp.(api.PodList)
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Verb begins a request with a verb (GET, POST, PUT, DELETE)
|
2014-06-22 21:18:01 +00:00
|
|
|
func (c *Client) Verb(verb string) *Request {
|
2014-06-22 19:05:34 +00:00
|
|
|
return &Request{
|
2014-06-26 23:10:38 +00:00
|
|
|
verb: verb,
|
|
|
|
c: c,
|
|
|
|
path: "/api/v1beta1",
|
2014-07-07 16:36:55 +00:00
|
|
|
sync: c.Sync,
|
|
|
|
timeout: c.Timeout,
|
|
|
|
pollPeriod: c.PollPeriod,
|
2014-06-22 19:05:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Post begins a POST request.
|
2014-06-23 00:02:48 +00:00
|
|
|
func (c *Client) Post() *Request {
|
|
|
|
return c.Verb("POST")
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Put begins a PUT request.
|
2014-06-23 00:02:48 +00:00
|
|
|
func (c *Client) Put() *Request {
|
|
|
|
return c.Verb("PUT")
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Get begins a GET request.
|
2014-06-23 00:02:48 +00:00
|
|
|
func (c *Client) Get() *Request {
|
|
|
|
return c.Verb("GET")
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Delete begins a DELETE request.
|
2014-06-23 00:02:48 +00:00
|
|
|
func (c *Client) Delete() *Request {
|
|
|
|
return c.Verb("DELETE")
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// 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)
|
2014-06-26 23:10:38 +00:00
|
|
|
}
|
|
|
|
|
2014-06-22 19:05:34 +00:00
|
|
|
// Request allows for building up a request to a server in a chained fashion.
|
2014-06-23 00:02:48 +00:00
|
|
|
// Any errors are stored until the end of your call, so you only have to
|
|
|
|
// check once.
|
2014-06-22 19:05:34 +00:00
|
|
|
type Request struct {
|
2014-06-26 23:10:38 +00:00
|
|
|
c *Client
|
|
|
|
err error
|
|
|
|
verb string
|
|
|
|
path string
|
|
|
|
body io.Reader
|
|
|
|
selector labels.Selector
|
|
|
|
timeout time.Duration
|
|
|
|
sync bool
|
|
|
|
pollPeriod time.Duration
|
2014-06-22 19:05:34 +00:00
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Path appends an item to the request path. You must call Path at least once.
|
2014-06-22 19:05:34 +00:00
|
|
|
func (r *Request) Path(item string) *Request {
|
|
|
|
if r.err != nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.path = path.Join(r.path, item)
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Sync sets sync/async call status.
|
2014-06-25 23:23:15 +00:00
|
|
|
func (r *Request) Sync(sync bool) *Request {
|
|
|
|
if r.err != nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.sync = sync
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// AbsPath overwrites an existing path with the path parameter.
|
2014-06-24 21:57:09 +00:00
|
|
|
func (r *Request) AbsPath(path string) *Request {
|
|
|
|
if r.err != nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.path = path
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// ParseSelector parses the given string as a resource label selector. Optional.
|
2014-06-23 00:02:48 +00:00
|
|
|
func (r *Request) ParseSelector(item string) *Request {
|
2014-06-22 19:05:34 +00:00
|
|
|
if r.err != nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.selector, r.err = labels.ParseSelector(item)
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Selector makes the request use the given selector.
|
2014-06-23 00:02:48 +00:00
|
|
|
func (r *Request) Selector(s labels.Selector) *Request {
|
|
|
|
if r.err != nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.selector = s
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Timeout makes the request use the given duration as a timeout. Optional.
|
2014-06-22 19:05:34 +00:00
|
|
|
func (r *Request) Timeout(d time.Duration) *Request {
|
|
|
|
if r.err != nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.timeout = d
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Body makes the request use obj as the body. Optional.
|
2014-06-22 19:05:34 +00:00
|
|
|
// 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
|
|
|
|
}
|
2014-06-23 00:02:48 +00:00
|
|
|
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)
|
2014-06-24 21:57:09 +00:00
|
|
|
case io.Reader:
|
|
|
|
r.body = obj.(io.Reader)
|
2014-06-23 00:02:48 +00:00
|
|
|
default:
|
|
|
|
data, err := api.Encode(obj)
|
|
|
|
if err != nil {
|
|
|
|
r.err = err
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
r.body = bytes.NewBuffer(data)
|
|
|
|
}
|
2014-06-22 19:05:34 +00:00
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-06-26 23:10:38 +00:00
|
|
|
// 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 {
|
2014-06-22 19:05:34 +00:00
|
|
|
if r.err != nil {
|
2014-06-26 23:10:38 +00:00
|
|
|
return r
|
2014-06-22 19:05:34 +00:00
|
|
|
}
|
2014-06-26 23:10:38 +00:00
|
|
|
r.pollPeriod = d
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-17 23:09:29 +00:00
|
|
|
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)
|
|
|
|
}
|
2014-07-20 20:41:15 +00:00
|
|
|
return watch.NewStreamWatcher(tools.NewAPIEventDecoder(response.Body)), nil
|
2014-07-17 23:09:29 +00:00
|
|
|
}
|
|
|
|
|
2014-07-08 07:15:41 +00:00
|
|
|
// Do formats and executes the request. Returns the API object received, or an error.
|
2014-06-26 23:10:38 +00:00
|
|
|
func (r *Request) Do() Result {
|
|
|
|
for {
|
|
|
|
if r.err != nil {
|
|
|
|
return Result{err: r.err}
|
2014-06-25 23:23:15 +00:00
|
|
|
}
|
2014-07-17 23:09:29 +00:00
|
|
|
req, err := http.NewRequest(r.verb, r.finalURL(), r.body)
|
2014-06-26 23:10:38 +00:00
|
|
|
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 {
|
2014-07-09 05:35:58 +00:00
|
|
|
glog.Infof("Waiting for completion of /operations/%s", statusErr.Status.Details)
|
2014-06-26 23:10:38 +00:00
|
|
|
time.Sleep(r.pollPeriod)
|
|
|
|
// Make a poll request
|
|
|
|
pollOp := r.c.PollFor(statusErr.Status.Details).PollPeriod(r.pollPeriod)
|
|
|
|
// Could also say "return r.Do()" but this way doesn't grow the callstack.
|
|
|
|
r = pollOp
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2014-06-23 01:14:32 +00:00
|
|
|
}
|
2014-06-26 23:10:38 +00:00
|
|
|
return Result{respBody, err}
|
2014-06-23 01:14:32 +00:00
|
|
|
}
|
2014-06-23 00:02:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
2014-06-22 19:05:34 +00:00
|
|
|
}
|
2014-06-23 00:02:48 +00:00
|
|
|
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
|
2014-06-22 19:05:34 +00:00
|
|
|
}
|