open-nomad/vendor/github.com/joyent/triton-go/compute/volumes.go
Seth Hoenig 435c0d9fc8 deps: Switch to Go modules for dependency management
This PR switches the Nomad repository from using govendor to Go modules
for managing dependencies. Aspects of the Nomad workflow remain pretty
much the same. The usual Makefile targets should continue to work as
they always did. The API submodule simply defers to the parent Nomad
version on the repository, keeping the semantics of API versioning that
currently exists.
2020-06-02 14:30:36 -05:00

218 lines
4.8 KiB
Go

//
// Copyright (c) 2018, Joyent, Inc. All rights reserved.
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
//
package compute
import (
"context"
"encoding/json"
"net/http"
"net/url"
"path"
"github.com/joyent/triton-go/client"
"github.com/pkg/errors"
)
type VolumesClient struct {
client *client.Client
}
type Volume struct {
ID string `json:"id"`
Name string `json:"name"`
Owner string `json:"owner_uuid"`
Type string `json:"type"`
FileSystemPath string `json:"filesystem_path"`
Size int64 `json:"size"`
State string `json:"state"`
Networks []string `json:"networks"`
Refs []string `json:"refs"`
}
type ListVolumesInput struct {
Name string
Size string
State string
Type string
}
func (c *VolumesClient) List(ctx context.Context, input *ListVolumesInput) ([]*Volume, error) {
fullPath := path.Join("/", c.client.AccountName, "volumes")
query := &url.Values{}
if input.Name != "" {
query.Set("name", input.Name)
}
if input.Size != "" {
query.Set("size", input.Size)
}
if input.State != "" {
query.Set("state", input.State)
}
if input.Type != "" {
query.Set("type", input.Type)
}
reqInputs := client.RequestInput{
Method: http.MethodGet,
Path: fullPath,
Query: query,
}
resp, err := c.client.ExecuteRequest(ctx, reqInputs)
if resp != nil {
defer resp.Close()
}
if err != nil {
return nil, errors.Wrap(err, "unable to list volumes")
}
var result []*Volume
decoder := json.NewDecoder(resp)
if err = decoder.Decode(&result); err != nil {
return nil, errors.Wrap(err, "unable to decode list volumes response")
}
return result, nil
}
type CreateVolumeInput struct {
Name string
Size int64
Networks []string
Type string
}
func (input *CreateVolumeInput) toAPI() map[string]interface{} {
result := make(map[string]interface{}, 0)
if input.Name != "" {
result["name"] = input.Name
}
if input.Size != 0 {
result["size"] = input.Size
}
if input.Type != "" {
result["type"] = input.Type
}
if len(input.Networks) > 0 {
result["networks"] = input.Networks
}
return result
}
func (c *VolumesClient) Create(ctx context.Context, input *CreateVolumeInput) (*Volume, error) {
fullPath := path.Join("/", c.client.AccountName, "volumes")
reqInputs := client.RequestInput{
Method: http.MethodPost,
Path: fullPath,
Body: input.toAPI(),
}
resp, err := c.client.ExecuteRequest(ctx, reqInputs)
if err != nil {
return nil, errors.Wrap(err, "unable to create volume")
}
if resp != nil {
defer resp.Close()
}
var result *Volume
decoder := json.NewDecoder(resp)
if err = decoder.Decode(&result); err != nil {
return nil, errors.Wrap(err, "unable to decode create volume response")
}
return result, nil
}
type DeleteVolumeInput struct {
ID string
}
func (c *VolumesClient) Delete(ctx context.Context, input *DeleteVolumeInput) error {
fullPath := path.Join("/", c.client.AccountName, "volumes", input.ID)
reqInputs := client.RequestInput{
Method: http.MethodDelete,
Path: fullPath,
}
resp, err := c.client.ExecuteRequestRaw(ctx, reqInputs)
if err != nil {
return errors.Wrap(err, "unable to delete volume")
}
if resp == nil {
return errors.Wrap(err, "unable to delete volume")
}
if resp.Body != nil {
defer resp.Body.Close()
}
if resp.StatusCode == http.StatusNotFound || resp.StatusCode == http.StatusGone {
return nil
}
return nil
}
type GetVolumeInput struct {
ID string
}
func (c *VolumesClient) Get(ctx context.Context, input *GetVolumeInput) (*Volume, error) {
fullPath := path.Join("/", c.client.AccountName, "volumes", input.ID)
reqInputs := client.RequestInput{
Method: http.MethodGet,
Path: fullPath,
}
resp, err := c.client.ExecuteRequestRaw(ctx, reqInputs)
if err != nil {
return nil, errors.Wrap(err, "unable to get volume")
}
if resp == nil {
return nil, errors.Wrap(err, "unable to get volume")
}
if resp.Body != nil {
defer resp.Body.Close()
}
var result *Volume
decoder := json.NewDecoder(resp.Body)
if err = decoder.Decode(&result); err != nil {
return nil, errors.Wrap(err, "unable to decode get volume volume")
}
return result, nil
}
type UpdateVolumeInput struct {
ID string `json:"-"`
Name string `json:"name"`
}
func (c *VolumesClient) Update(ctx context.Context, input *UpdateVolumeInput) error {
fullPath := path.Join("/", c.client.AccountName, "volumes", input.ID)
reqInputs := client.RequestInput{
Method: http.MethodPost,
Path: fullPath,
Body: input,
}
resp, err := c.client.ExecuteRequest(ctx, reqInputs)
if err != nil {
return errors.Wrap(err, "unable to update volume")
}
if resp != nil {
defer resp.Close()
}
return nil
}