| package bootfromvolume |
| |
| import ( |
| "errors" |
| "strconv" |
| |
| "github.com/gophercloud/gophercloud" |
| "github.com/gophercloud/gophercloud/openstack/compute/v2/servers" |
| ) |
| |
| // SourceType represents the type of medium being used to create the volume. |
| type SourceType string |
| |
| const ( |
| Volume SourceType = "volume" |
| Snapshot SourceType = "snapshot" |
| Image SourceType = "image" |
| Blank SourceType = "blank" |
| ) |
| |
| // BlockDevice is a structure with options for booting a server instance |
| // from a volume. The volume may be created from an image, snapshot, or another |
| // volume. |
| type BlockDevice struct { |
| // BootIndex [optional] is the boot index. It defaults to 0. |
| BootIndex int `json:"boot_index"` |
| |
| // DeleteOnTermination [optional] specifies whether or not to delete the attached volume |
| // when the server is deleted. Defaults to `false`. |
| DeleteOnTermination bool `json:"delete_on_termination"` |
| |
| // DestinationType [optional] is the type that gets created. Possible values are "volume" |
| // and "local". |
| DestinationType string `json:"destination_type"` |
| |
| // GuestFormat [optional] specifies the format of the block device. |
| GuestFormat string `json:"guest_format"` |
| |
| // SourceType [required] must be one of: "volume", "snapshot", "image". |
| SourceType SourceType `json:"source_type"` |
| |
| // UUID [required] is the unique identifier for the volume, snapshot, or image (see above) |
| UUID string `json:"uuid"` |
| |
| // VolumeSize [optional] is the size of the volume to create (in gigabytes). |
| VolumeSize int `json:"volume_size"` |
| } |
| |
| // CreateOptsExt is a structure that extends the server `CreateOpts` structure |
| // by allowing for a block device mapping. |
| type CreateOptsExt struct { |
| servers.CreateOptsBuilder |
| BlockDevice []BlockDevice `json:"block_device_mapping_v2,omitempty"` |
| } |
| |
| // ToServerCreateMap adds the block device mapping option to the base server |
| // creation options. |
| func (opts CreateOptsExt) ToServerCreateMap() (map[string]interface{}, error) { |
| base, err := opts.CreateOptsBuilder.ToServerCreateMap() |
| if err != nil { |
| return nil, err |
| } |
| |
| if len(opts.BlockDevice) == 0 { |
| return nil, errors.New("Required fields UUID and SourceType not set.") |
| } |
| |
| serverMap := base["server"].(map[string]interface{}) |
| |
| blockDevice := make([]map[string]interface{}, len(opts.BlockDevice)) |
| |
| for i, bd := range opts.BlockDevice { |
| if string(bd.SourceType) == "" { |
| return nil, errors.New("SourceType must be one of: volume, image, snapshot.") |
| } |
| |
| blockDevice[i] = make(map[string]interface{}) |
| |
| blockDevice[i]["source_type"] = bd.SourceType |
| blockDevice[i]["boot_index"] = strconv.Itoa(bd.BootIndex) |
| blockDevice[i]["delete_on_termination"] = strconv.FormatBool(bd.DeleteOnTermination) |
| blockDevice[i]["volume_size"] = strconv.Itoa(bd.VolumeSize) |
| if bd.UUID != "" { |
| blockDevice[i]["uuid"] = bd.UUID |
| } |
| if bd.DestinationType != "" { |
| blockDevice[i]["destination_type"] = bd.DestinationType |
| } |
| if bd.GuestFormat != "" { |
| blockDevice[i]["guest_format"] = bd.GuestFormat |
| } |
| |
| } |
| serverMap["block_device_mapping_v2"] = blockDevice |
| |
| return base, nil |
| } |
| |
| // Create requests the creation of a server from the given block device mapping. |
| func Create(client *gophercloud.ServiceClient, opts servers.CreateOptsBuilder) servers.CreateResult { |
| var res servers.CreateResult |
| |
| reqBody, err := opts.ToServerCreateMap() |
| if err != nil { |
| res.Err = err |
| return res |
| } |
| |
| // Delete imageName and flavorName that come from ToServerCreateMap(). |
| // As of Liberty, Boot From Volume is failing if they are passed. |
| delete(reqBody["server"].(map[string]interface{}), "imageName") |
| delete(reqBody["server"].(map[string]interface{}), "flavorName") |
| |
| _, res.Err = client.Post(createURL(client), reqBody, &res.Body, &gophercloud.RequestOpts{ |
| OkCodes: []int{200, 202}, |
| }) |
| return res |
| } |