pulumi/pkg/backend/cloud/client/api.go

256 lines
8.0 KiB
Go

// Copyright 2016-2018, Pulumi Corporation.
//
// 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"
"context"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
"runtime"
"strings"
"github.com/google/go-querystring/query"
"github.com/opentracing/opentracing-go"
"github.com/pkg/errors"
"github.com/pulumi/pulumi/pkg/apitype"
"github.com/pulumi/pulumi/pkg/backend"
"github.com/pulumi/pulumi/pkg/util/contract"
"github.com/pulumi/pulumi/pkg/util/httputil"
"github.com/pulumi/pulumi/pkg/util/logging"
"github.com/pulumi/pulumi/pkg/version"
)
// UpdateKind is an enum for describing the kinds of updates we support.
type UpdateKind string
const (
UpdateKindUpdate UpdateKind = "update"
UpdateKindPreview UpdateKind = "preview"
UpdateKindRefresh UpdateKind = "refresh"
UpdateKindDestroy UpdateKind = "destroy"
UpdateKindImport UpdateKind = "import"
)
// StackIdentifier is the set of data needed to identify a Pulumi Cloud stack.
type StackIdentifier struct {
Owner string
Stack string
}
// UpdateIdentifier is the set of data needed to identify an update to a Pulumi Cloud stack.
type UpdateIdentifier struct {
StackIdentifier
UpdateKind UpdateKind
UpdateID string
}
// accessTokenKind is enumerates the various types of access token used with the Pulumi API. These kinds correspond
// directly to the "method" piece of an HTTP `Authorization` header.
type accessTokenKind string
const (
// accessTokenKindAPIToken denotes a standard Pulumi API token.
accessTokenKindAPIToken accessTokenKind = "token"
// accessTokenKindUpdateToken denotes an update lease token.
accessTokenKindUpdateToken accessTokenKind = "update-token"
)
// accessToken is an abstraction over the two different kinds of access tokens used by the Pulumi API.
type accessToken interface {
Kind() accessTokenKind
String() string
}
type httpCallOptions struct {
// RetryAllMethods allows non-GET calls to be retried if the server fails to return a response.
RetryAllMethods bool
}
// apiAccessToken is an implementation of accessToken for Pulumi API tokens (i.e. tokens of kind
// accessTokenKindAPIToken)
type apiAccessToken string
func (apiAccessToken) Kind() accessTokenKind {
return accessTokenKindAPIToken
}
func (t apiAccessToken) String() string {
return string(t)
}
// updateAccessToken is an implementation of accessToken for update lease tokens (i.e. tokens of kind
// accessTokenKindUpdateToken)
type updateAccessToken string
func (updateAccessToken) Kind() accessTokenKind {
return accessTokenKindUpdateToken
}
func (t updateAccessToken) String() string {
return string(t)
}
// pulumiAPICall makes an HTTP request to the Pulumi API.
func pulumiAPICall(ctx context.Context, cloudAPI, method, path string, body []byte, tok accessToken,
opts httpCallOptions) (string, *http.Response, error) {
// Normalize URL components
cloudAPI = strings.TrimSuffix(cloudAPI, "/")
path = cleanPath(path)
url := fmt.Sprintf("%s%s", cloudAPI, path)
req, err := http.NewRequest(method, url, bytes.NewReader(body))
if err != nil {
return "", nil, errors.Wrapf(err, "creating new HTTP request")
}
requestSpan, requestContext := opentracing.StartSpanFromContext(ctx, getEndpointName(method, path),
opentracing.Tag{Key: "method", Value: method},
opentracing.Tag{Key: "path", Value: path},
opentracing.Tag{Key: "api", Value: cloudAPI},
opentracing.Tag{Key: "retry", Value: opts.RetryAllMethods})
defer requestSpan.Finish()
req = req.WithContext(requestContext)
req.Header.Set("Content-Type", "application/json")
// Add a User-Agent header to allow for the backend to make breaking API changes while preserving
// backwards compatibility.
userAgent := fmt.Sprintf("pulumi-cli/1 (%s; %s)", version.Version, runtime.GOOS)
req.Header.Set("User-Agent", userAgent)
// Apply credentials if provided.
if tok.String() != "" {
req.Header.Set("Authorization", fmt.Sprintf("%s %s", tok.Kind(), tok.String()))
}
tracingOptions := backend.TracingOptionsFromContext(requestContext)
if tracingOptions.PropagateSpans {
carrier := opentracing.HTTPHeadersCarrier(req.Header)
if err = requestSpan.Tracer().Inject(requestSpan.Context(), opentracing.HTTPHeaders, carrier); err != nil {
logging.Errorf("injecting tracing headers: %v", err)
}
}
if tracingOptions.TracingHeader != "" {
req.Header.Set("X-Pulumi-Tracing", tracingOptions.TracingHeader)
}
logging.V(7).Infof("Making Pulumi API call: %s", url)
if logging.V(9) {
logging.V(9).Infof("Pulumi API call details (%s): headers=%v; body=%v", url, req.Header, string(body))
}
var resp *http.Response
if req.Method == "GET" || opts.RetryAllMethods {
resp, err = httputil.DoWithRetry(req, http.DefaultClient)
} else {
resp, err = http.DefaultClient.Do(req)
}
if err != nil {
return "", nil, errors.Wrapf(err, "performing HTTP request")
}
logging.V(7).Infof("Pulumi API call response code (%s): %v", url, resp.Status)
requestSpan.SetTag("responseCode", resp.Status)
// For 4xx and 5xx failures, attempt to provide better diagnostics about what may have gone wrong.
if resp.StatusCode >= 400 && resp.StatusCode <= 599 {
defer contract.IgnoreClose(resp.Body)
// Provide a better error if using an authenticated call without having logged in first.
if resp.StatusCode == 401 && tok.Kind() == accessTokenKindAPIToken && tok.String() == "" {
return "", nil, errors.New("this command requires logging in; try running 'pulumi login' first")
}
// 4xx and 5xx responses should be of type ErrorResponse. See if we can unmarshal as that
// type, and if not just return the raw response text.
respBody, err := ioutil.ReadAll(resp.Body)
if err != nil {
return "", nil, errors.Wrapf(
err, "API call failed (%s), could not read response", resp.Status)
}
var errResp apitype.ErrorResponse
if err = json.Unmarshal(respBody, &errResp); err != nil {
errResp.Code = resp.StatusCode
errResp.Message = strings.TrimSpace(string(respBody))
}
return "", nil, &errResp
}
return url, resp, nil
}
// pulumiRESTCall calls the Pulumi REST API marshalling reqObj to JSON and using that as
// the request body (use nil for GETs), and if successful, marshalling the responseObj
// as JSON and storing it in respObj (use nil for NoContent). The error return type might
// be an instance of apitype.ErrorResponse, in which case will have the response code.
func pulumiRESTCall(ctx context.Context, cloudAPI, method, path string, queryObj, reqObj, respObj interface{},
tok accessToken, opts httpCallOptions) error {
// Compute query string from query object
querystring := ""
if queryObj != nil {
queryValues, err := query.Values(queryObj)
if err != nil {
return errors.Wrapf(err, "marshalling query object as JSON")
}
query := queryValues.Encode()
if len(query) > 0 {
querystring = "?" + query
}
}
// Compute request body from request object
var reqBody []byte
var err error
if reqObj != nil {
reqBody, err = json.Marshal(reqObj)
if err != nil {
return errors.Wrapf(err, "marshalling request object as JSON")
}
}
// Make API call
url, resp, err := pulumiAPICall(ctx, cloudAPI, method, path+querystring, reqBody, tok, opts)
if err != nil {
return err
}
defer contract.IgnoreClose(resp.Body)
// Read API response
respBody, err := ioutil.ReadAll(resp.Body)
if err != nil {
return errors.Wrapf(err, "reading response from API")
}
if logging.V(9) {
logging.V(7).Infof("Pulumi API call response body (%s): %v", url, string(respBody))
}
if respObj != nil {
if err = json.Unmarshal(respBody, respObj); err != nil {
return errors.Wrapf(err, "unmarshalling response object")
}
}
return nil
}