2019-04-15 20:42:02 -04:00
|
|
|
package linodego
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"time"
|
2020-06-12 05:36:54 -04:00
|
|
|
|
|
|
|
"github.com/linode/linodego/internal/parseabletime"
|
2019-04-15 20:42:02 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// InstanceDisk represents an Instance Disk object
|
|
|
|
type InstanceDisk struct {
|
|
|
|
ID int `json:"id"`
|
|
|
|
Label string `json:"label"`
|
|
|
|
Status DiskStatus `json:"status"`
|
|
|
|
Size int `json:"size"`
|
|
|
|
Filesystem DiskFilesystem `json:"filesystem"`
|
|
|
|
Created time.Time `json:"-"`
|
|
|
|
Updated time.Time `json:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// DiskFilesystem constants start with Filesystem and include Linode API Filesystems
|
|
|
|
type DiskFilesystem string
|
|
|
|
|
|
|
|
// DiskFilesystem constants represent the filesystems types an Instance Disk may use
|
|
|
|
const (
|
|
|
|
FilesystemRaw DiskFilesystem = "raw"
|
|
|
|
FilesystemSwap DiskFilesystem = "swap"
|
|
|
|
FilesystemExt3 DiskFilesystem = "ext3"
|
|
|
|
FilesystemExt4 DiskFilesystem = "ext4"
|
|
|
|
FilesystemInitrd DiskFilesystem = "initrd"
|
|
|
|
)
|
|
|
|
|
|
|
|
// DiskStatus constants have the prefix "Disk" and include Linode API Instance Disk Status
|
|
|
|
type DiskStatus string
|
|
|
|
|
|
|
|
// DiskStatus constants represent the status values an Instance Disk may have
|
|
|
|
const (
|
|
|
|
DiskReady DiskStatus = "ready"
|
|
|
|
DiskNotReady DiskStatus = "not ready"
|
|
|
|
DiskDeleting DiskStatus = "deleting"
|
|
|
|
)
|
|
|
|
|
|
|
|
// InstanceDisksPagedResponse represents a paginated InstanceDisk API response
|
|
|
|
type InstanceDisksPagedResponse struct {
|
|
|
|
*PageOptions
|
|
|
|
Data []InstanceDisk `json:"data"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// InstanceDiskCreateOptions are InstanceDisk settings that can be used at creation
|
|
|
|
type InstanceDiskCreateOptions struct {
|
|
|
|
Label string `json:"label"`
|
|
|
|
Size int `json:"size"`
|
|
|
|
|
|
|
|
// Image is optional, but requires RootPass if provided
|
|
|
|
Image string `json:"image,omitempty"`
|
|
|
|
RootPass string `json:"root_pass,omitempty"`
|
|
|
|
|
|
|
|
Filesystem string `json:"filesystem,omitempty"`
|
|
|
|
AuthorizedKeys []string `json:"authorized_keys,omitempty"`
|
|
|
|
AuthorizedUsers []string `json:"authorized_users,omitempty"`
|
|
|
|
ReadOnly bool `json:"read_only,omitempty"`
|
|
|
|
StackscriptID int `json:"stackscript_id,omitempty"`
|
|
|
|
StackscriptData map[string]string `json:"stackscript_data,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// InstanceDiskUpdateOptions are InstanceDisk settings that can be used in updates
|
|
|
|
type InstanceDiskUpdateOptions struct {
|
|
|
|
Label string `json:"label"`
|
|
|
|
ReadOnly bool `json:"read_only"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// endpointWithID gets the endpoint URL for InstanceDisks of a given Instance
|
|
|
|
func (InstanceDisksPagedResponse) endpointWithID(c *Client, id int) string {
|
|
|
|
endpoint, err := c.InstanceDisks.endpointWithID(id)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return endpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
// appendData appends InstanceDisks when processing paginated InstanceDisk responses
|
|
|
|
func (resp *InstanceDisksPagedResponse) appendData(r *InstanceDisksPagedResponse) {
|
|
|
|
resp.Data = append(resp.Data, r.Data...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListInstanceDisks lists InstanceDisks
|
|
|
|
func (c *Client) ListInstanceDisks(ctx context.Context, linodeID int, opts *ListOptions) ([]InstanceDisk, error) {
|
|
|
|
response := InstanceDisksPagedResponse{}
|
|
|
|
err := c.listHelperWithID(ctx, &response, linodeID, opts)
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return response.Data, nil
|
|
|
|
}
|
|
|
|
|
2020-06-12 05:36:54 -04:00
|
|
|
// UnmarshalJSON implements the json.Unmarshaler interface
|
|
|
|
func (i *InstanceDisk) UnmarshalJSON(b []byte) error {
|
|
|
|
type Mask InstanceDisk
|
|
|
|
|
|
|
|
p := struct {
|
|
|
|
*Mask
|
|
|
|
Created *parseabletime.ParseableTime `json:"created"`
|
|
|
|
Updated *parseabletime.ParseableTime `json:"updated"`
|
|
|
|
}{
|
|
|
|
Mask: (*Mask)(i),
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(b, &p); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Created != nil {
|
|
|
|
i.Created = time.Time(*p.Created)
|
2019-04-15 20:42:02 -04:00
|
|
|
}
|
2020-06-12 05:36:54 -04:00
|
|
|
if p.Updated != nil {
|
|
|
|
i.Updated = time.Time(*p.Updated)
|
2019-04-15 20:42:02 -04:00
|
|
|
}
|
2020-06-12 05:36:54 -04:00
|
|
|
|
|
|
|
return nil
|
2019-04-15 20:42:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetInstanceDisk gets the template with the provided ID
|
|
|
|
func (c *Client) GetInstanceDisk(ctx context.Context, linodeID int, configID int) (*InstanceDisk, error) {
|
|
|
|
e, err := c.InstanceDisks.endpointWithID(linodeID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
e = fmt.Sprintf("%s/%d", e, configID)
|
|
|
|
r, err := coupleAPIErrors(c.R(ctx).SetResult(&InstanceDisk{}).Get(e))
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-12 05:36:54 -04:00
|
|
|
return r.Result().(*InstanceDisk), nil
|
2019-04-15 20:42:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateInstanceDisk creates a new InstanceDisk for the given Instance
|
|
|
|
func (c *Client) CreateInstanceDisk(ctx context.Context, linodeID int, createOpts InstanceDiskCreateOptions) (*InstanceDisk, error) {
|
|
|
|
var body string
|
|
|
|
e, err := c.InstanceDisks.endpointWithID(linodeID)
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
req := c.R(ctx).SetResult(&InstanceDisk{})
|
|
|
|
|
|
|
|
if bodyData, err := json.Marshal(createOpts); err == nil {
|
|
|
|
body = string(bodyData)
|
|
|
|
} else {
|
|
|
|
return nil, NewError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := coupleAPIErrors(req.
|
|
|
|
SetBody(body).
|
|
|
|
Post(e))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-06-12 05:36:54 -04:00
|
|
|
return r.Result().(*InstanceDisk), nil
|
2019-04-15 20:42:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateInstanceDisk creates a new InstanceDisk for the given Instance
|
|
|
|
func (c *Client) UpdateInstanceDisk(ctx context.Context, linodeID int, diskID int, updateOpts InstanceDiskUpdateOptions) (*InstanceDisk, error) {
|
|
|
|
var body string
|
|
|
|
e, err := c.InstanceDisks.endpointWithID(linodeID)
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-06-12 05:36:54 -04:00
|
|
|
e = fmt.Sprintf("%s/%d", e, diskID)
|
2019-04-15 20:42:02 -04:00
|
|
|
req := c.R(ctx).SetResult(&InstanceDisk{})
|
|
|
|
|
|
|
|
if bodyData, err := json.Marshal(updateOpts); err == nil {
|
|
|
|
body = string(bodyData)
|
|
|
|
} else {
|
|
|
|
return nil, NewError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := coupleAPIErrors(req.
|
|
|
|
SetBody(body).
|
|
|
|
Put(e))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-06-12 05:36:54 -04:00
|
|
|
return r.Result().(*InstanceDisk), nil
|
2019-04-15 20:42:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// RenameInstanceDisk renames an InstanceDisk
|
|
|
|
func (c *Client) RenameInstanceDisk(ctx context.Context, linodeID int, diskID int, label string) (*InstanceDisk, error) {
|
|
|
|
return c.UpdateInstanceDisk(ctx, linodeID, diskID, InstanceDiskUpdateOptions{Label: label})
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResizeInstanceDisk resizes the size of the Instance disk
|
|
|
|
func (c *Client) ResizeInstanceDisk(ctx context.Context, linodeID int, diskID int, size int) error {
|
|
|
|
var body string
|
|
|
|
e, err := c.InstanceDisks.endpointWithID(linodeID)
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
e = fmt.Sprintf("%s/%d/resize", e, diskID)
|
|
|
|
|
|
|
|
req := c.R(ctx).SetResult(&InstanceDisk{})
|
|
|
|
updateOpts := map[string]interface{}{
|
|
|
|
"size": size,
|
|
|
|
}
|
|
|
|
|
|
|
|
if bodyData, err := json.Marshal(updateOpts); err == nil {
|
|
|
|
body = string(bodyData)
|
|
|
|
} else {
|
|
|
|
return NewError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = coupleAPIErrors(req.
|
|
|
|
SetBody(body).
|
|
|
|
Post(e))
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// PasswordResetInstanceDisk resets the "root" account password on the Instance disk
|
|
|
|
func (c *Client) PasswordResetInstanceDisk(ctx context.Context, linodeID int, diskID int, password string) error {
|
|
|
|
var body string
|
|
|
|
e, err := c.InstanceDisks.endpointWithID(linodeID)
|
2020-06-12 05:36:54 -04:00
|
|
|
|
2019-04-15 20:42:02 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
e = fmt.Sprintf("%s/%d/password", e, diskID)
|
|
|
|
|
|
|
|
req := c.R(ctx).SetResult(&InstanceDisk{})
|
|
|
|
updateOpts := map[string]interface{}{
|
|
|
|
"password": password,
|
|
|
|
}
|
|
|
|
|
|
|
|
if bodyData, err := json.Marshal(updateOpts); err == nil {
|
|
|
|
body = string(bodyData)
|
|
|
|
} else {
|
|
|
|
return NewError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = coupleAPIErrors(req.
|
|
|
|
SetBody(body).
|
|
|
|
Post(e))
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteInstanceDisk deletes a Linode Instance Disk
|
|
|
|
func (c *Client) DeleteInstanceDisk(ctx context.Context, linodeID int, diskID int) error {
|
|
|
|
e, err := c.InstanceDisks.endpointWithID(linodeID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
e = fmt.Sprintf("%s/%d", e, diskID)
|
|
|
|
|
|
|
|
_, err = coupleAPIErrors(c.R(ctx).Delete(e))
|
|
|
|
return err
|
|
|
|
}
|