mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-18 04:32:59 -06:00
98d84680b7
* provider/scaleway: extract volume validation helpers * provider/scaleway: add server volume property fixes #9499 * provider/scaleway: update `scaleway_server` docu * provider/scaleway: fix volume handling this actually broken when merging the latest SDK update :( * provider/scaleway: fix volume attachment * provider/scaleway: fix volume expectation
270 lines
6.0 KiB
Go
270 lines
6.0 KiB
Go
package scaleway
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
|
|
"github.com/hashicorp/terraform/helper/schema"
|
|
"github.com/scaleway/scaleway-cli/pkg/api"
|
|
)
|
|
|
|
func resourceScalewayServer() *schema.Resource {
|
|
return &schema.Resource{
|
|
Create: resourceScalewayServerCreate,
|
|
Read: resourceScalewayServerRead,
|
|
Update: resourceScalewayServerUpdate,
|
|
Delete: resourceScalewayServerDelete,
|
|
Importer: &schema.ResourceImporter{
|
|
State: schema.ImportStatePassthrough,
|
|
},
|
|
|
|
Schema: map[string]*schema.Schema{
|
|
"name": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Required: true,
|
|
},
|
|
"image": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Required: true,
|
|
ForceNew: true,
|
|
},
|
|
"type": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Required: true,
|
|
ForceNew: true,
|
|
},
|
|
"bootscript": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
},
|
|
"tags": &schema.Schema{
|
|
Type: schema.TypeList,
|
|
Elem: &schema.Schema{
|
|
Type: schema.TypeString,
|
|
},
|
|
Optional: true,
|
|
},
|
|
"enable_ipv6": &schema.Schema{
|
|
Type: schema.TypeBool,
|
|
Optional: true,
|
|
Default: false,
|
|
},
|
|
"dynamic_ip_required": &schema.Schema{
|
|
Type: schema.TypeBool,
|
|
Optional: true,
|
|
},
|
|
"security_group": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
},
|
|
"volume": &schema.Schema{
|
|
Type: schema.TypeList,
|
|
Optional: true,
|
|
ForceNew: true,
|
|
Elem: &schema.Resource{
|
|
Schema: map[string]*schema.Schema{
|
|
"size_in_gb": {
|
|
Type: schema.TypeInt,
|
|
Required: true,
|
|
ValidateFunc: validateVolumeSize,
|
|
},
|
|
"type": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Required: true,
|
|
ValidateFunc: validateVolumeType,
|
|
},
|
|
"volume_id": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Computed: true,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
"private_ip": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Computed: true,
|
|
},
|
|
"public_ip": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Computed: true,
|
|
},
|
|
"state": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
Computed: true,
|
|
},
|
|
"state_detail": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Computed: true,
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
func resourceScalewayServerCreate(d *schema.ResourceData, m interface{}) error {
|
|
scaleway := m.(*Client).scaleway
|
|
|
|
image := d.Get("image").(string)
|
|
var server = api.ScalewayServerDefinition{
|
|
Name: d.Get("name").(string),
|
|
Image: String(image),
|
|
Organization: scaleway.Organization,
|
|
EnableIPV6: d.Get("enable_ipv6").(bool),
|
|
SecurityGroup: d.Get("security_group").(string),
|
|
}
|
|
|
|
server.DynamicIPRequired = Bool(d.Get("dynamic_ip_required").(bool))
|
|
server.CommercialType = d.Get("type").(string)
|
|
|
|
if bootscript, ok := d.GetOk("bootscript"); ok {
|
|
server.Bootscript = String(bootscript.(string))
|
|
}
|
|
|
|
if vs, ok := d.GetOk("volume"); ok {
|
|
server.Volumes = make(map[string]string)
|
|
|
|
volumes := vs.([]interface{})
|
|
for i, v := range volumes {
|
|
volume := v.(map[string]interface{})
|
|
|
|
volumeID, err := scaleway.PostVolume(api.ScalewayVolumeDefinition{
|
|
Size: uint64(volume["size_in_gb"].(int)) * gb,
|
|
Type: volume["type"].(string),
|
|
Name: fmt.Sprintf("%s-%d", server.Name, volume["size_in_gb"].(int)),
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
volume["volume_id"] = volumeID
|
|
volumes[i] = volume
|
|
server.Volumes[fmt.Sprintf("%d", i+1)] = volumeID
|
|
}
|
|
d.Set("volume", volumes)
|
|
}
|
|
|
|
if raw, ok := d.GetOk("tags"); ok {
|
|
for _, tag := range raw.([]interface{}) {
|
|
server.Tags = append(server.Tags, tag.(string))
|
|
}
|
|
}
|
|
|
|
id, err := scaleway.PostServer(server)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
d.SetId(id)
|
|
if d.Get("state").(string) != "stopped" {
|
|
err = scaleway.PostServerAction(id, "poweron")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = waitForServerState(scaleway, id, "running")
|
|
}
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return resourceScalewayServerRead(d, m)
|
|
}
|
|
|
|
func resourceScalewayServerRead(d *schema.ResourceData, m interface{}) error {
|
|
scaleway := m.(*Client).scaleway
|
|
server, err := scaleway.GetServer(d.Id())
|
|
|
|
if err != nil {
|
|
if serr, ok := err.(api.ScalewayAPIError); ok {
|
|
log.Printf("[DEBUG] Error reading server: %q\n", serr.APIMessage)
|
|
|
|
if serr.StatusCode == 404 {
|
|
d.SetId("")
|
|
return nil
|
|
}
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
d.Set("name", server.Name)
|
|
d.Set("image", server.Image.Identifier)
|
|
d.Set("type", server.CommercialType)
|
|
d.Set("enable_ipv6", server.EnableIPV6)
|
|
d.Set("private_ip", server.PrivateIP)
|
|
d.Set("public_ip", server.PublicAddress.IP)
|
|
|
|
d.Set("state", server.State)
|
|
d.Set("state_detail", server.StateDetail)
|
|
d.Set("tags", server.Tags)
|
|
|
|
d.SetConnInfo(map[string]string{
|
|
"type": "ssh",
|
|
"host": server.PublicAddress.IP,
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
func resourceScalewayServerUpdate(d *schema.ResourceData, m interface{}) error {
|
|
scaleway := m.(*Client).scaleway
|
|
|
|
var req api.ScalewayServerPatchDefinition
|
|
|
|
if d.HasChange("name") {
|
|
name := d.Get("name").(string)
|
|
req.Name = &name
|
|
}
|
|
|
|
if d.HasChange("tags") {
|
|
if raw, ok := d.GetOk("tags"); ok {
|
|
var tags []string
|
|
for _, tag := range raw.([]interface{}) {
|
|
tags = append(tags, tag.(string))
|
|
}
|
|
req.Tags = &tags
|
|
}
|
|
}
|
|
|
|
if d.HasChange("enable_ipv6") {
|
|
req.EnableIPV6 = Bool(d.Get("enable_ipv6").(bool))
|
|
}
|
|
|
|
if d.HasChange("dynamic_ip_required") {
|
|
req.DynamicIPRequired = Bool(d.Get("dynamic_ip_required").(bool))
|
|
}
|
|
|
|
if d.HasChange("security_group") {
|
|
req.SecurityGroup = &api.ScalewaySecurityGroup{
|
|
Identifier: d.Get("security_group").(string),
|
|
}
|
|
}
|
|
|
|
if err := scaleway.PatchServer(d.Id(), req); err != nil {
|
|
return fmt.Errorf("Failed patching scaleway server: %q", err)
|
|
}
|
|
|
|
return resourceScalewayServerRead(d, m)
|
|
}
|
|
|
|
func resourceScalewayServerDelete(d *schema.ResourceData, m interface{}) error {
|
|
scaleway := m.(*Client).scaleway
|
|
|
|
s, err := scaleway.GetServer(d.Id())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if s.State == "stopped" {
|
|
return deleteStoppedServer(scaleway, s)
|
|
}
|
|
|
|
err = deleteRunningServer(scaleway, s)
|
|
|
|
if err == nil {
|
|
d.SetId("")
|
|
}
|
|
|
|
return err
|
|
}
|