393 lines
10 KiB
Go
393 lines
10 KiB
Go
package hcloud
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"net/url"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/hetznercloud/hcloud-go/hcloud/schema"
|
|
)
|
|
|
|
// Volume represents a volume in the Hetzner Cloud.
|
|
type Volume struct {
|
|
ID int
|
|
Name string
|
|
Server *Server
|
|
Location *Location
|
|
Size int
|
|
Protection VolumeProtection
|
|
Labels map[string]string
|
|
LinuxDevice string
|
|
Created time.Time
|
|
}
|
|
|
|
// VolumeProtection represents the protection level of a volume.
|
|
type VolumeProtection struct {
|
|
Delete bool
|
|
}
|
|
|
|
// VolumeClient is a client for the volume API.
|
|
type VolumeClient struct {
|
|
client *Client
|
|
}
|
|
|
|
// VolumeStatus specifies a volume's status.
|
|
type VolumeStatus string
|
|
|
|
const (
|
|
// VolumeStatusCreating is the status when a volume is being created.
|
|
VolumeStatusCreating VolumeStatus = "creating"
|
|
|
|
// VolumeStatusAvailable is the status when a volume is available.
|
|
VolumeStatusAvailable VolumeStatus = "available"
|
|
)
|
|
|
|
// GetByID retrieves a volume by its ID. If the volume does not exist, nil is returned.
|
|
func (c *VolumeClient) GetByID(ctx context.Context, id int) (*Volume, *Response, error) {
|
|
req, err := c.client.NewRequest(ctx, "GET", fmt.Sprintf("/volumes/%d", id), nil)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
var body schema.VolumeGetResponse
|
|
resp, err := c.client.Do(req, &body)
|
|
if err != nil {
|
|
if IsError(err, ErrorCodeNotFound) {
|
|
return nil, resp, nil
|
|
}
|
|
return nil, nil, err
|
|
}
|
|
return VolumeFromSchema(body.Volume), resp, nil
|
|
}
|
|
|
|
// GetByName retrieves a volume by its name. If the volume does not exist, nil is returned.
|
|
func (c *VolumeClient) GetByName(ctx context.Context, name string) (*Volume, *Response, error) {
|
|
volumes, response, err := c.List(ctx, VolumeListOpts{Name: name})
|
|
if len(volumes) == 0 {
|
|
return nil, response, err
|
|
}
|
|
return volumes[0], response, err
|
|
}
|
|
|
|
// Get retrieves a volume by its ID if the input can be parsed as an integer, otherwise it
|
|
// retrieves a volume by its name. If the volume does not exist, nil is returned.
|
|
func (c *VolumeClient) Get(ctx context.Context, idOrName string) (*Volume, *Response, error) {
|
|
if id, err := strconv.Atoi(idOrName); err == nil {
|
|
return c.GetByID(ctx, int(id))
|
|
}
|
|
return c.GetByName(ctx, idOrName)
|
|
}
|
|
|
|
// VolumeListOpts specifies options for listing volumes.
|
|
type VolumeListOpts struct {
|
|
ListOpts
|
|
Name string
|
|
Status []VolumeStatus
|
|
}
|
|
|
|
func (l VolumeListOpts) values() url.Values {
|
|
vals := l.ListOpts.values()
|
|
if l.Name != "" {
|
|
vals.Add("name", l.Name)
|
|
}
|
|
for _, status := range l.Status {
|
|
vals.Add("status", string(status))
|
|
}
|
|
return vals
|
|
}
|
|
|
|
// List returns a list of volumes for a specific page.
|
|
func (c *VolumeClient) List(ctx context.Context, opts VolumeListOpts) ([]*Volume, *Response, error) {
|
|
path := "/volumes?" + opts.values().Encode()
|
|
req, err := c.client.NewRequest(ctx, "GET", path, nil)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
var body schema.VolumeListResponse
|
|
resp, err := c.client.Do(req, &body)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
volumes := make([]*Volume, 0, len(body.Volumes))
|
|
for _, s := range body.Volumes {
|
|
volumes = append(volumes, VolumeFromSchema(s))
|
|
}
|
|
return volumes, resp, nil
|
|
}
|
|
|
|
// All returns all volumes.
|
|
func (c *VolumeClient) All(ctx context.Context) ([]*Volume, error) {
|
|
return c.AllWithOpts(ctx, VolumeListOpts{ListOpts: ListOpts{PerPage: 50}})
|
|
}
|
|
|
|
// AllWithOpts returns all volumes with the given options.
|
|
func (c *VolumeClient) AllWithOpts(ctx context.Context, opts VolumeListOpts) ([]*Volume, error) {
|
|
allVolumes := []*Volume{}
|
|
|
|
_, err := c.client.all(func(page int) (*Response, error) {
|
|
opts.Page = page
|
|
volumes, resp, err := c.List(ctx, opts)
|
|
if err != nil {
|
|
return resp, err
|
|
}
|
|
allVolumes = append(allVolumes, volumes...)
|
|
return resp, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return allVolumes, nil
|
|
}
|
|
|
|
// VolumeCreateOpts specifies parameters for creating a volume.
|
|
type VolumeCreateOpts struct {
|
|
Name string
|
|
Size int
|
|
Server *Server
|
|
Location *Location
|
|
Labels map[string]string
|
|
Automount *bool
|
|
Format *string
|
|
}
|
|
|
|
// Validate checks if options are valid.
|
|
func (o VolumeCreateOpts) Validate() error {
|
|
if o.Name == "" {
|
|
return errors.New("missing name")
|
|
}
|
|
if o.Size <= 0 {
|
|
return errors.New("size must be greater than 0")
|
|
}
|
|
if o.Server == nil && o.Location == nil {
|
|
return errors.New("one of server or location must be provided")
|
|
}
|
|
if o.Server != nil && o.Location != nil {
|
|
return errors.New("only one of server or location must be provided")
|
|
}
|
|
if o.Server == nil && (o.Automount != nil && *o.Automount) {
|
|
return errors.New("server must be provided when automount is true")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// VolumeCreateResult is the result of creating a volume.
|
|
type VolumeCreateResult struct {
|
|
Volume *Volume
|
|
Action *Action
|
|
NextActions []*Action
|
|
}
|
|
|
|
// Create creates a new volume with the given options.
|
|
func (c *VolumeClient) Create(ctx context.Context, opts VolumeCreateOpts) (VolumeCreateResult, *Response, error) {
|
|
if err := opts.Validate(); err != nil {
|
|
return VolumeCreateResult{}, nil, err
|
|
}
|
|
reqBody := schema.VolumeCreateRequest{
|
|
Name: opts.Name,
|
|
Size: opts.Size,
|
|
Automount: opts.Automount,
|
|
Format: opts.Format,
|
|
}
|
|
if opts.Labels != nil {
|
|
reqBody.Labels = &opts.Labels
|
|
}
|
|
if opts.Server != nil {
|
|
reqBody.Server = Int(opts.Server.ID)
|
|
}
|
|
if opts.Location != nil {
|
|
if opts.Location.ID != 0 {
|
|
reqBody.Location = opts.Location.ID
|
|
} else {
|
|
reqBody.Location = opts.Location.Name
|
|
}
|
|
}
|
|
|
|
reqBodyData, err := json.Marshal(reqBody)
|
|
if err != nil {
|
|
return VolumeCreateResult{}, nil, err
|
|
}
|
|
|
|
req, err := c.client.NewRequest(ctx, "POST", "/volumes", bytes.NewReader(reqBodyData))
|
|
if err != nil {
|
|
return VolumeCreateResult{}, nil, err
|
|
}
|
|
|
|
var respBody schema.VolumeCreateResponse
|
|
resp, err := c.client.Do(req, &respBody)
|
|
if err != nil {
|
|
return VolumeCreateResult{}, resp, err
|
|
}
|
|
|
|
var action *Action
|
|
if respBody.Action != nil {
|
|
action = ActionFromSchema(*respBody.Action)
|
|
}
|
|
|
|
return VolumeCreateResult{
|
|
Volume: VolumeFromSchema(respBody.Volume),
|
|
Action: action,
|
|
NextActions: ActionsFromSchema(respBody.NextActions),
|
|
}, resp, nil
|
|
}
|
|
|
|
// Delete deletes a volume.
|
|
func (c *VolumeClient) Delete(ctx context.Context, volume *Volume) (*Response, error) {
|
|
req, err := c.client.NewRequest(ctx, "DELETE", fmt.Sprintf("/volumes/%d", volume.ID), nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return c.client.Do(req, nil)
|
|
}
|
|
|
|
// VolumeUpdateOpts specifies options for updating a volume.
|
|
type VolumeUpdateOpts struct {
|
|
Name string
|
|
Labels map[string]string
|
|
}
|
|
|
|
// Update updates a volume.
|
|
func (c *VolumeClient) Update(ctx context.Context, volume *Volume, opts VolumeUpdateOpts) (*Volume, *Response, error) {
|
|
reqBody := schema.VolumeUpdateRequest{
|
|
Name: opts.Name,
|
|
}
|
|
if opts.Labels != nil {
|
|
reqBody.Labels = &opts.Labels
|
|
}
|
|
reqBodyData, err := json.Marshal(reqBody)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
path := fmt.Sprintf("/volumes/%d", volume.ID)
|
|
req, err := c.client.NewRequest(ctx, "PUT", path, bytes.NewReader(reqBodyData))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
respBody := schema.VolumeUpdateResponse{}
|
|
resp, err := c.client.Do(req, &respBody)
|
|
if err != nil {
|
|
return nil, resp, err
|
|
}
|
|
return VolumeFromSchema(respBody.Volume), resp, nil
|
|
}
|
|
|
|
// VolumeAttachOpts specifies options for attaching a volume.
|
|
type VolumeAttachOpts struct {
|
|
Server *Server
|
|
Automount *bool
|
|
}
|
|
|
|
// AttachWithOpts attaches a volume to a server.
|
|
func (c *VolumeClient) AttachWithOpts(ctx context.Context, volume *Volume, opts VolumeAttachOpts) (*Action, *Response, error) {
|
|
reqBody := schema.VolumeActionAttachVolumeRequest{
|
|
Server: opts.Server.ID,
|
|
Automount: opts.Automount,
|
|
}
|
|
|
|
reqBodyData, err := json.Marshal(reqBody)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
path := fmt.Sprintf("/volumes/%d/actions/attach", volume.ID)
|
|
req, err := c.client.NewRequest(ctx, "POST", path, bytes.NewReader(reqBodyData))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
var respBody schema.VolumeActionAttachVolumeResponse
|
|
resp, err := c.client.Do(req, &respBody)
|
|
if err != nil {
|
|
return nil, resp, err
|
|
}
|
|
return ActionFromSchema(respBody.Action), resp, nil
|
|
}
|
|
|
|
// Attach attaches a volume to a server.
|
|
func (c *VolumeClient) Attach(ctx context.Context, volume *Volume, server *Server) (*Action, *Response, error) {
|
|
return c.AttachWithOpts(ctx, volume, VolumeAttachOpts{Server: server})
|
|
}
|
|
|
|
// Detach detaches a volume from a server.
|
|
func (c *VolumeClient) Detach(ctx context.Context, volume *Volume) (*Action, *Response, error) {
|
|
var reqBody schema.VolumeActionDetachVolumeRequest
|
|
reqBodyData, err := json.Marshal(reqBody)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
path := fmt.Sprintf("/volumes/%d/actions/detach", volume.ID)
|
|
req, err := c.client.NewRequest(ctx, "POST", path, bytes.NewReader(reqBodyData))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
var respBody schema.VolumeActionDetachVolumeResponse
|
|
resp, err := c.client.Do(req, &respBody)
|
|
if err != nil {
|
|
return nil, resp, err
|
|
}
|
|
return ActionFromSchema(respBody.Action), resp, nil
|
|
}
|
|
|
|
// VolumeChangeProtectionOpts specifies options for changing the resource protection level of a volume.
|
|
type VolumeChangeProtectionOpts struct {
|
|
Delete *bool
|
|
}
|
|
|
|
// ChangeProtection changes the resource protection level of a volume.
|
|
func (c *VolumeClient) ChangeProtection(ctx context.Context, volume *Volume, opts VolumeChangeProtectionOpts) (*Action, *Response, error) {
|
|
reqBody := schema.VolumeActionChangeProtectionRequest{
|
|
Delete: opts.Delete,
|
|
}
|
|
reqBodyData, err := json.Marshal(reqBody)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
path := fmt.Sprintf("/volumes/%d/actions/change_protection", volume.ID)
|
|
req, err := c.client.NewRequest(ctx, "POST", path, bytes.NewReader(reqBodyData))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
respBody := schema.VolumeActionChangeProtectionResponse{}
|
|
resp, err := c.client.Do(req, &respBody)
|
|
if err != nil {
|
|
return nil, resp, err
|
|
}
|
|
return ActionFromSchema(respBody.Action), resp, err
|
|
}
|
|
|
|
// Resize changes the size of a volume.
|
|
func (c *VolumeClient) Resize(ctx context.Context, volume *Volume, size int) (*Action, *Response, error) {
|
|
reqBody := schema.VolumeActionResizeVolumeRequest{
|
|
Size: size,
|
|
}
|
|
reqBodyData, err := json.Marshal(reqBody)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
path := fmt.Sprintf("/volumes/%d/actions/resize", volume.ID)
|
|
req, err := c.client.NewRequest(ctx, "POST", path, bytes.NewReader(reqBodyData))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
respBody := schema.VolumeActionResizeVolumeResponse{}
|
|
resp, err := c.client.Do(req, &respBody)
|
|
if err != nil {
|
|
return nil, resp, err
|
|
}
|
|
return ActionFromSchema(respBody.Action), resp, err
|
|
}
|