package dm import ( "encoding/json" "fmt" "io" "io/ioutil" "net/http" "net/url" "os" fancypath "path" "path/filepath" "strings" "time" "github.com/kubernetes/deployment-manager/common" ) // The default HTTP timeout var DefaultHTTPTimeout = time.Second * 10 // The default HTTP Protocol var DefaultHTTPProtocol = "http" // Client is a DM client. type Client struct { // Timeout on HTTP connections. HTTPTimeout time.Duration // The remote host Host string // The protocol. Currently only http and https are supported. Protocol string // Transport Transport http.RoundTripper // Debug enables http logging Debug bool // Base URL for remote service baseURL *url.URL } // NewClient creates a new DM client. Host name is required. func NewClient(host string) *Client { url, _ := DefaultServerURL(host) return &Client{ HTTPTimeout: DefaultHTTPTimeout, baseURL: url, Transport: http.DefaultTransport, } } // SetDebug enables debug mode which logs http func (c *Client) SetDebug(enable bool) *Client { c.Debug = enable return c } // transport wraps client transport if debug is enabled func (c *Client) transport() http.RoundTripper { if c.Debug { return NewDebugTransport(c.Transport) } return c.Transport } // SetTransport sets a custom Transport. Defaults to http.DefaultTransport func (c *Client) SetTransport(tr http.RoundTripper) *Client { c.Transport = tr return c } // SetTimeout sets a timeout for http connections func (c *Client) SetTimeout(seconds int) *Client { c.HTTPTimeout = time.Duration(time.Duration(seconds) * time.Second) return c } // url constructs the URL. func (c *Client) url(rawurl string) (string, error) { u, err := url.Parse(rawurl) if err != nil { return "", err } return c.baseURL.ResolveReference(u).String(), nil } func (c *Client) agent() string { return fmt.Sprintf("helm/%s", "0.0.1") } // CallService is a low-level function for making an API call. // // This calls the service and then unmarshals the returned data into dest. func (c *Client) CallService(path, method, action string, dest interface{}, reader io.ReadCloser) error { u, err := c.url(path) if err != nil { return err } resp, err := c.callHTTP(u, method, action, reader) if err != nil { return err } if err := json.Unmarshal([]byte(resp), dest); err != nil { return fmt.Errorf("Failed to parse JSON response from service: %s", resp) } return nil } // callHTTP is a low-level primitive for executing HTTP operations. func (c *Client) callHTTP(path, method, action string, reader io.ReadCloser) (string, error) { request, err := http.NewRequest(method, path, reader) // TODO: dynamically set version request.Header.Set("User-Agent", c.agent()) request.Header.Add("Content-Type", "application/json") client := &http.Client{ Timeout: c.HTTPTimeout, Transport: c.transport(), } response, err := client.Do(request) if err != nil { return "", err } defer response.Body.Close() body, err := ioutil.ReadAll(response.Body) if err != nil { return "", err } s := response.StatusCode if s < http.StatusOK || s >= http.StatusMultipleChoices { return "", &HTTPError{StatusCode: s, Message: string(body), URL: request.URL} } return string(body), nil } // DefaultServerURL converts a host, host:port, or URL string to the default base server API path // to use with a Client func DefaultServerURL(host string) (*url.URL, error) { if host == "" { return nil, fmt.Errorf("host must be a URL or a host:port pair") } base := host hostURL, err := url.Parse(base) if err != nil { return nil, err } if hostURL.Scheme == "" { hostURL, err = url.Parse(DefaultHTTPProtocol + "://" + base) if err != nil { return nil, err } } if len(hostURL.Path) > 0 && !strings.HasSuffix(hostURL.Path, "/") { hostURL.Path = hostURL.Path + "/" } return hostURL, nil } // ListDeployments lists the deployments in DM. func (c *Client) ListDeployments() ([]string, error) { var l []string if err := c.CallService("deployments", "GET", "list deployments", &l, nil); err != nil { return nil, err } return l, nil } // PostChart sends a chart to DM for deploying. // // This returns the location for the new chart, typically of the form // `helm:repo/bucket/name-version.tgz`. func (c *Client) PostChart(filename, deployname string) (string, error) { f, err := os.Open(filename) if err != nil { return "", err } u, err := c.url("/v2/charts") request, err := http.NewRequest("POST", u, f) if err != nil { f.Close() return "", err } // There is an argument to be made for using the legacy x-octet-stream for // this. But since we control both sides, we should use the standard one. // Also, gzip (x-compress) is usually treated as a content encoding. In this // case it probably is not, but it makes more sense to follow the standard, // even though we don't assume the remote server will strip it off. request.Header.Add("Content-Type", "application/x-tar") request.Header.Add("Content-Encoding", "gzip") request.Header.Add("X-Deployment-Name", deployname) request.Header.Add("X-Chart-Name", filepath.Base(filename)) request.Header.Set("User-Agent", c.agent()) client := &http.Client{ Timeout: c.HTTPTimeout, Transport: c.transport(), } response, err := client.Do(request) if err != nil { return "", err } // We only want 201 CREATED. Admittedly, we could accept 200 and 202. if response.StatusCode != http.StatusCreated { body, err := ioutil.ReadAll(response.Body) response.Body.Close() if err != nil { return "", err } return "", &HTTPError{StatusCode: response.StatusCode, Message: string(body), URL: request.URL} } loc := response.Header.Get("Location") return loc, nil } // HTTPError is an error caused by an unexpected HTTP status code. // // The StatusCode will not necessarily be a 4xx or 5xx. Any unexpected code // may be returned. type HTTPError struct { StatusCode int Message string URL *url.URL } // Error implements the error interface. func (e *HTTPError) Error() string { return e.Message } // String implmenets the io.Stringer interface. func (e *HTTPError) String() string { return e.Error() } // GetDeployment retrieves the supplied deployment func (c *Client) GetDeployment(name string) (*common.Deployment, error) { var deployment *common.Deployment if err := c.CallService(fancypath.Join("deployments", name), "GET", "get deployment", &deployment, nil); err != nil { return nil, err } return deployment, nil } // DeleteDeployment deletes the supplied deployment func (c *Client) DeleteDeployment(name string) (*common.Deployment, error) { var deployment *common.Deployment if err := c.CallService(filepath.Join("deployments", name), "DELETE", "delete deployment", &deployment, nil); err != nil { return nil, err } return deployment, nil } // PostDeployment posts a deployment objec to the manager service. func (c *Client) PostDeployment(cfg *common.Configuration) error { return c.CallService("/deployments", "POST", "post deployment", cfg, nil) }