*: Bump version of vmware/govmomi

Bumping version to include changes that
better handle TLS errors. Bump nescessary
to prepare for when the version of Go is
bumped to 1.20

Signed-off-by: Madhav Jivrajani <madhav.jiv@gmail.com>
This commit is contained in:
Madhav Jivrajani
2023-01-02 20:56:02 +05:30
parent 45df8f0bb3
commit 8b064fa4be
263 changed files with 32336 additions and 4373 deletions

View File

@@ -25,6 +25,9 @@ import (
"io/ioutil"
"net/http"
"net/url"
"strings"
"sync"
"time"
"github.com/vmware/govmomi/vapi/internal"
"github.com/vmware/govmomi/vim25"
@@ -33,14 +36,93 @@ import (
// Client extends soap.Client to support JSON encoding, while inheriting security features, debug tracing and session persistence.
type Client struct {
mu sync.Mutex
*soap.Client
sessionID string
}
// Session information
type Session struct {
User string `json:"user"`
Created time.Time `json:"created_time"`
LastAccessed time.Time `json:"last_accessed_time"`
}
// LocalizableMessage represents a localizable error
type LocalizableMessage struct {
Args []string `json:"args,omitempty"`
DefaultMessage string `json:"default_message,omitempty"`
ID string `json:"id,omitempty"`
}
func (m *LocalizableMessage) Error() string {
return m.DefaultMessage
}
// NewClient creates a new Client instance.
func NewClient(c *vim25.Client) *Client {
sc := c.Client.NewServiceClient(internal.Path, "")
sc := c.Client.NewServiceClient(Path, "")
return &Client{sc}
return &Client{Client: sc}
}
// SessionID is set by calling Login() or optionally with the given id param
func (c *Client) SessionID(id ...string) string {
c.mu.Lock()
defer c.mu.Unlock()
if len(id) != 0 {
c.sessionID = id[0]
}
return c.sessionID
}
type marshaledClient struct {
SoapClient *soap.Client
SessionID string
}
func (c *Client) MarshalJSON() ([]byte, error) {
m := marshaledClient{
SoapClient: c.Client,
SessionID: c.sessionID,
}
return json.Marshal(m)
}
func (c *Client) UnmarshalJSON(b []byte) error {
var m marshaledClient
err := json.Unmarshal(b, &m)
if err != nil {
return err
}
*c = Client{
Client: m.SoapClient,
sessionID: m.SessionID,
}
return nil
}
// isAPI returns true if path starts with "/api"
// This hack allows helpers to support both endpoints:
// "/rest" - value wrapped responses and structured error responses
// "/api" - raw responses and no structured error responses
func isAPI(path string) bool {
return strings.HasPrefix(path, "/api")
}
// Resource helper for the given path.
func (c *Client) Resource(path string) *Resource {
r := &Resource{u: c.URL()}
if !isAPI(path) {
path = Path + path
}
r.u.Path = path
return r
}
type Signer interface {
@@ -53,24 +135,64 @@ func (c *Client) WithSigner(ctx context.Context, s Signer) context.Context {
return context.WithValue(ctx, signerContext{}, s)
}
type headersContext struct{}
// WithHeader returns a new Context populated with the provided headers map.
// Calls to a VAPI REST client with this context will populate the HTTP headers
// map using the provided headers.
func (c *Client) WithHeader(
ctx context.Context,
headers http.Header) context.Context {
return context.WithValue(ctx, headersContext{}, headers)
}
type statusError struct {
res *http.Response
}
func (e *statusError) Error() string {
return fmt.Sprintf("%s %s: %s", e.res.Request.Method, e.res.Request.URL, e.res.Status)
}
// RawResponse may be used with the Do method as the resBody argument in order
// to capture the raw response data.
type RawResponse struct {
bytes.Buffer
}
// Do sends the http.Request, decoding resBody if provided.
func (c *Client) Do(ctx context.Context, req *http.Request, resBody interface{}) error {
switch req.Method {
case http.MethodPost, http.MethodPatch:
case http.MethodPost, http.MethodPatch, http.MethodPut:
req.Header.Set("Content-Type", "application/json")
}
req.Header.Set("Accept", "application/json")
if id := c.SessionID(); id != "" {
req.Header.Set(internal.SessionCookieName, id)
}
if s, ok := ctx.Value(signerContext{}).(Signer); ok {
if err := s.SignRequest(req); err != nil {
return err
}
}
if headers, ok := ctx.Value(headersContext{}).(http.Header); ok {
for k, v := range headers {
for _, v := range v {
req.Header.Add(k, v)
}
}
}
return c.Client.Do(ctx, req, func(res *http.Response) error {
switch res.StatusCode {
case http.StatusOK:
case http.StatusCreated:
case http.StatusNoContent:
case http.StatusBadRequest:
// TODO: structured error types
detail, err := ioutil.ReadAll(res.Body)
@@ -79,7 +201,7 @@ func (c *Client) Do(ctx context.Context, req *http.Request, resBody interface{})
}
return fmt.Errorf("%s: %s", res.Status, bytes.TrimSpace(detail))
default:
return fmt.Errorf("%s %s: %s", req.Method, req.URL, res.Status)
return &statusError{res}
}
if resBody == nil {
@@ -87,23 +209,68 @@ func (c *Client) Do(ctx context.Context, req *http.Request, resBody interface{})
}
switch b := resBody.(type) {
case *RawResponse:
return res.Write(b)
case io.Writer:
_, err := io.Copy(b, res.Body)
return err
default:
d := json.NewDecoder(res.Body)
if isAPI(req.URL.Path) {
// Responses from the /api endpoint are not wrapped
return d.Decode(resBody)
}
// Responses from the /rest endpoint are wrapped in this structure
val := struct {
Value interface{} `json:"value,omitempty"`
}{
resBody,
}
return json.NewDecoder(res.Body).Decode(&val)
return d.Decode(&val)
}
})
}
// authHeaders ensures the given map contains a REST auth header
func (c *Client) authHeaders(h map[string]string) map[string]string {
if _, exists := h[internal.SessionCookieName]; exists {
return h
}
if h == nil {
h = make(map[string]string)
}
h[internal.SessionCookieName] = c.SessionID()
return h
}
// Download wraps soap.Client.Download, adding the REST authentication header
func (c *Client) Download(ctx context.Context, u *url.URL, param *soap.Download) (io.ReadCloser, int64, error) {
p := *param
p.Headers = c.authHeaders(p.Headers)
return c.Client.Download(ctx, u, &p)
}
// DownloadFile wraps soap.Client.DownloadFile, adding the REST authentication header
func (c *Client) DownloadFile(ctx context.Context, file string, u *url.URL, param *soap.Download) error {
p := *param
p.Headers = c.authHeaders(p.Headers)
return c.Client.DownloadFile(ctx, file, u, &p)
}
// Upload wraps soap.Client.Upload, adding the REST authentication header
func (c *Client) Upload(ctx context.Context, f io.Reader, u *url.URL, param *soap.Upload) error {
p := *param
p.Headers = c.authHeaders(p.Headers)
return c.Client.Upload(ctx, f, u, &p)
}
// Login creates a new session via Basic Authentication with the given url.Userinfo.
func (c *Client) Login(ctx context.Context, user *url.Userinfo) error {
req := internal.URL(c, internal.SessionPath).Request(http.MethodPost)
req := c.Resource(internal.SessionPath).Request(http.MethodPost)
req.Header.Set(internal.UseHeaderAuthn, "true")
if user != nil {
if password, ok := user.Password(); ok {
@@ -111,15 +278,59 @@ func (c *Client) Login(ctx context.Context, user *url.Userinfo) error {
}
}
return c.Do(ctx, req, nil)
var id string
err := c.Do(ctx, req, &id)
if err != nil {
return err
}
c.SessionID(id)
return nil
}
func (c *Client) LoginByToken(ctx context.Context) error {
return c.Login(ctx, nil)
}
// Session returns the user's current session.
// Nil is returned if the session is not authenticated.
func (c *Client) Session(ctx context.Context) (*Session, error) {
var s Session
req := c.Resource(internal.SessionPath).WithAction("get").Request(http.MethodPost)
err := c.Do(ctx, req, &s)
if err != nil {
if e, ok := err.(*statusError); ok {
if e.res.StatusCode == http.StatusUnauthorized {
return nil, nil
}
}
return nil, err
}
return &s, nil
}
// Logout deletes the current session.
func (c *Client) Logout(ctx context.Context) error {
req := internal.URL(c, internal.SessionPath).Request(http.MethodDelete)
req := c.Resource(internal.SessionPath).Request(http.MethodDelete)
return c.Do(ctx, req, nil)
}
// Valid returns whether or not the client is valid and ready for use.
// This should be called after unmarshalling the client.
func (c *Client) Valid() bool {
if c == nil {
return false
}
if c.Client == nil {
return false
}
return true
}
// Path returns rest.Path (see cache.Client)
func (c *Client) Path() string {
return Path
}

114
vendor/github.com/vmware/govmomi/vapi/rest/resource.go generated vendored Normal file
View File

@@ -0,0 +1,114 @@
/*
Copyright (c) 2019 VMware, 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 rest
import (
"bytes"
"encoding/json"
"io"
"net/http"
"net/url"
)
const (
Path = "/rest"
)
// Resource wraps url.URL with helpers
type Resource struct {
u *url.URL
}
func (r *Resource) String() string {
return r.u.String()
}
// WithID appends id to the URL.Path
func (r *Resource) WithID(id string) *Resource {
r.u.Path += "/id:" + id
return r
}
// WithAction sets adds action to the URL.RawQuery
func (r *Resource) WithAction(action string) *Resource {
return r.WithParam("~action", action)
}
// WithParam adds one parameter on the URL.RawQuery
func (r *Resource) WithParam(name string, value string) *Resource {
// ParseQuery handles empty case, and we control access to query string so shouldn't encounter an error case
params, err := url.ParseQuery(r.u.RawQuery)
if err != nil {
panic(err)
}
params[name] = append(params[name], value)
r.u.RawQuery = params.Encode()
return r
}
// WithPathEncodedParam appends a parameter on the URL.RawQuery,
// For special cases where URL Path-style encoding is needed
func (r *Resource) WithPathEncodedParam(name string, value string) *Resource {
t := &url.URL{Path: value}
encodedValue := t.String()
t = &url.URL{Path: name}
encodedName := t.String()
// ParseQuery handles empty case, and we control access to query string so shouldn't encounter an error case
params, err := url.ParseQuery(r.u.RawQuery)
if err != nil {
panic(err)
}
// Values.Encode() doesn't escape exactly how we want, so we need to build the query string ourselves
if len(params) >= 1 {
r.u.RawQuery = r.u.RawQuery + "&" + encodedName + "=" + encodedValue
} else {
r.u.RawQuery = r.u.RawQuery + encodedName + "=" + encodedValue
}
return r
}
// Request returns a new http.Request for the given method.
// An optional body can be provided for POST and PATCH methods.
func (r *Resource) Request(method string, body ...interface{}) *http.Request {
rdr := io.MultiReader() // empty body by default
if len(body) != 0 {
rdr = encode(body[0])
}
req, err := http.NewRequest(method, r.u.String(), rdr)
if err != nil {
panic(err)
}
return req
}
type errorReader struct {
e error
}
func (e errorReader) Read([]byte) (int, error) {
return -1, e.e
}
// encode body as JSON, deferring any errors until io.Reader is used.
func encode(body interface{}) io.Reader {
var b bytes.Buffer
err := json.NewEncoder(&b).Encode(body)
if err != nil {
return errorReader{err}
}
return &b
}