open-vault/api/client.go

197 lines
5 KiB
Go
Raw Normal View History

2015-03-04 21:10:10 +00:00
package api
import (
2015-04-20 18:30:35 +00:00
"errors"
"fmt"
2015-03-04 21:10:10 +00:00
"net/http"
2015-03-09 18:38:50 +00:00
"net/url"
"os"
"strings"
"sync"
"time"
2015-03-04 21:10:10 +00:00
)
2015-04-20 18:30:35 +00:00
var (
2015-09-03 17:38:24 +00:00
errRedirect = errors.New("redirect")
defaultHTTPClientSetup sync.Once
defaultHTTPClient = &http.Client{
Timeout: time.Second * 5,
}
2015-04-20 18:30:35 +00:00
)
2015-03-04 21:10:10 +00:00
// Config is used to configure the creation of the client.
type Config struct {
// Address is the address of the Vault server. This should be a complete
// URL such as "http://vault.example.com". If you need a custom SSL
// cert or want to enable insecure mode, you need to specify a custom
// HttpClient.
Address string
// HttpClient is the HTTP client to use, which will currently always have the
// same values as http.DefaultClient. This is used to control redirect behavior.
2015-03-04 21:10:10 +00:00
HttpClient *http.Client
}
// DefaultConfig returns a default configuration for the client. It is
// safe to modify the return value of this function.
2015-04-23 15:45:37 +00:00
//
// The default Address is https://127.0.0.1:8200, but this can be overridden by
2015-04-23 15:46:22 +00:00
// setting the `VAULT_ADDR` environment variable.
2015-04-23 15:13:52 +00:00
func DefaultConfig() *Config {
config := &Config{
2015-03-04 21:10:10 +00:00
Address: "https://127.0.0.1:8200",
HttpClient: defaultHTTPClient,
2015-08-22 00:36:19 +00:00
}
2015-04-23 15:46:22 +00:00
if addr := os.Getenv("VAULT_ADDR"); addr != "" {
config.Address = addr
}
2015-03-04 21:10:10 +00:00
return config
}
// Client is the client to the Vault API. Create a client with
// NewClient.
type Client struct {
2015-03-09 18:38:50 +00:00
addr *url.URL
2015-04-23 15:13:52 +00:00
config *Config
2015-08-22 00:36:19 +00:00
token string
2015-03-04 21:10:10 +00:00
}
// NewClient returns a new client for the given configuration.
2015-04-23 15:45:37 +00:00
//
// If the environment variable `VAULT_TOKEN` is present, the token will be
// automatically added to the client. Otherwise, you must manually call
// `SetToken()`.
2015-04-23 15:13:52 +00:00
func NewClient(c *Config) (*Client, error) {
2015-03-09 18:38:50 +00:00
u, err := url.Parse(c.Address)
if err != nil {
return nil, err
}
if c.HttpClient == defaultHTTPClient {
2015-09-03 17:38:24 +00:00
defaultHTTPClientSetup.Do(func() {
// Ensure redirects are not automatically followed
c.HttpClient.CheckRedirect = func(req *http.Request, via []*http.Request) error {
return errRedirect
}
})
2015-04-20 18:30:35 +00:00
}
client := &Client{
2015-03-09 18:38:50 +00:00
addr: u,
2015-03-04 21:10:10 +00:00
config: c,
}
if token := os.Getenv("VAULT_TOKEN"); token != "" {
client.SetToken(token)
}
return client, nil
2015-03-04 21:10:10 +00:00
}
2015-03-09 18:38:50 +00:00
2015-03-11 22:42:08 +00:00
// Token returns the access token being used by this client. It will
// return the empty string if there is no token set.
func (c *Client) Token() string {
2015-08-22 00:36:19 +00:00
return c.token
2015-03-11 22:42:08 +00:00
}
2015-03-31 04:20:23 +00:00
// SetToken sets the token directly. This won't perform any auth
2015-09-03 14:36:59 +00:00
// verification, it simply sets the token properly for future requests.
2015-03-31 04:20:23 +00:00
func (c *Client) SetToken(v string) {
2015-08-22 00:36:19 +00:00
c.token = v
2015-03-31 04:20:23 +00:00
}
2015-09-03 14:36:59 +00:00
// ClearToken deletes the token if it is set or does nothing otherwise.
2015-03-11 22:42:08 +00:00
func (c *Client) ClearToken() {
2015-08-22 00:36:19 +00:00
c.token = ""
2015-03-11 22:42:08 +00:00
}
2015-03-09 18:38:50 +00:00
// NewRequest creates a new raw request object to query the Vault server
// configured for this client. This is an advanced method and generally
// doesn't need to be called externally.
func (c *Client) NewRequest(method, path string) *Request {
2015-08-22 00:36:19 +00:00
req := &Request{
2015-03-09 18:38:50 +00:00
Method: method,
URL: &url.URL{
Scheme: c.addr.Scheme,
Host: c.addr.Host,
Path: path,
},
2015-08-22 00:36:19 +00:00
ClientToken: c.token,
Params: make(map[string][]string),
2015-03-09 18:38:50 +00:00
}
2015-08-22 00:36:19 +00:00
return req
2015-03-09 18:38:50 +00:00
}
// RawRequest performs the raw request given. This request may be against
// a Vault server not configured with this client. This is an advanced operation
// that generally won't need to be called externally.
2015-03-11 18:33:20 +00:00
func (c *Client) RawRequest(r *Request) (*Response, error) {
2015-04-20 18:30:35 +00:00
redirectCount := 0
START:
2015-03-09 18:38:50 +00:00
req, err := r.ToHTTP()
if err != nil {
return nil, err
}
2015-04-07 18:15:20 +00:00
var result *Response
2015-03-09 18:38:50 +00:00
resp, err := c.config.HttpClient.Do(req)
2015-04-07 18:15:20 +00:00
if resp != nil {
result = &Response{Response: resp}
}
2015-04-20 18:30:35 +00:00
if err != nil {
if urlErr, ok := err.(*url.Error); ok && urlErr.Err == errRedirect {
2015-04-20 18:30:35 +00:00
err = nil
} else if strings.Contains(err.Error(), "tls: oversized") {
err = fmt.Errorf(
"%s\n\n"+
"This error usually means that the server is running with TLS disabled\n"+
"but the client is configured to use TLS. Please either enable TLS\n"+
"on the server or run the client with -address set to an address\n"+
"that uses the http protocol:\n\n"+
" vault <command> -address http://<address>\n\n"+
"You can also set the VAULT_ADDR environment variable:\n\n\n"+
" VAULT_ADDR=http://<address> vault <command>\n\n"+
"where <address> is replaced by the actual address to the server.",
err)
2015-04-20 18:30:35 +00:00
}
}
2015-03-09 18:38:50 +00:00
if err != nil {
2015-04-07 18:15:20 +00:00
return result, err
2015-03-09 18:38:50 +00:00
}
2015-04-20 18:30:35 +00:00
// Check for a redirect, only allowing for a single redirect
if (resp.StatusCode == 302 || resp.StatusCode == 307) && redirectCount == 0 {
// Parse the updated location
respLoc, err := resp.Location()
if err != nil {
return result, err
}
// Ensure a protocol downgrade doesn't happen
if req.URL.Scheme == "https" && respLoc.Scheme != "https" {
return result, fmt.Errorf("redirect would cause protocol downgrade")
}
// Update the request
r.URL = respLoc
// Reset the request body if any
if err := r.ResetJSONBody(); err != nil {
return result, err
}
// Retry the request
redirectCount++
goto START
}
if err := result.Error(); err != nil {
2015-04-07 18:15:20 +00:00
return result, err
}
return result, nil
2015-03-09 18:38:50 +00:00
}