357 lines
9.3 KiB
Go
357 lines
9.3 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
|
|
}
|
|
|
|
// GetByID retrieves a volume by its ID.
|
|
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.
|
|
func (c *VolumeClient) GetByName(ctx context.Context, name string) (*Volume, *Response, error) {
|
|
path := "/volumes?name=" + url.QueryEscape(name)
|
|
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
|
|
}
|
|
|
|
if len(body.Volumes) == 0 {
|
|
return nil, resp, nil
|
|
}
|
|
return VolumeFromSchema(body.Volumes[0]), resp, nil
|
|
}
|
|
|
|
// Get retrieves a volume by its ID if the input can be parsed as an integer,
|
|
// otherwise it retrieves a volume by its name.
|
|
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
|
|
}
|
|
|
|
// List returns a list of volumes for a specific page.
|
|
func (c *VolumeClient) List(ctx context.Context, opts VolumeListOpts) ([]*Volume, *Response, error) {
|
|
path := "/volumes?" + valuesForListOpts(opts.ListOpts).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{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
|
|
}
|
|
|
|
// 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")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// VolumeCreateResult is the result of creating a volume.
|
|
type VolumeCreateResult struct {
|
|
Volume *Volume
|
|
Action *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,
|
|
}
|
|
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,
|
|
}, 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
|
|
}
|
|
|
|
// Attach attaches a volume to a server.
|
|
func (c *VolumeClient) Attach(ctx context.Context, volume *Volume, server *Server) (*Action, *Response, error) {
|
|
reqBody := schema.VolumeActionAttachVolumeRequest{
|
|
Server: server.ID,
|
|
}
|
|
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
|
|
}
|
|
|
|
// 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
|
|
}
|