github.com/boyvanduuren/terraform@v0.7.0-rc2.0.20160805175930-de822d909c40/builtin/providers/aws/resource_aws_rds_cluster_instance.go (about) 1 package aws 2 3 import ( 4 "fmt" 5 "log" 6 "time" 7 8 "github.com/aws/aws-sdk-go/aws" 9 "github.com/aws/aws-sdk-go/service/rds" 10 "github.com/hashicorp/terraform/helper/resource" 11 "github.com/hashicorp/terraform/helper/schema" 12 ) 13 14 func resourceAwsRDSClusterInstance() *schema.Resource { 15 return &schema.Resource{ 16 Create: resourceAwsRDSClusterInstanceCreate, 17 Read: resourceAwsRDSClusterInstanceRead, 18 Update: resourceAwsRDSClusterInstanceUpdate, 19 Delete: resourceAwsRDSClusterInstanceDelete, 20 Importer: &schema.ResourceImporter{ 21 State: schema.ImportStatePassthrough, 22 }, 23 24 Schema: map[string]*schema.Schema{ 25 "identifier": &schema.Schema{ 26 Type: schema.TypeString, 27 Optional: true, 28 ForceNew: true, 29 ValidateFunc: validateRdsId, 30 }, 31 32 "db_subnet_group_name": &schema.Schema{ 33 Type: schema.TypeString, 34 Optional: true, 35 ForceNew: true, 36 Computed: true, 37 }, 38 39 "writer": &schema.Schema{ 40 Type: schema.TypeBool, 41 Computed: true, 42 }, 43 44 "cluster_identifier": &schema.Schema{ 45 Type: schema.TypeString, 46 Required: true, 47 ForceNew: true, 48 }, 49 50 "endpoint": &schema.Schema{ 51 Type: schema.TypeString, 52 Computed: true, 53 }, 54 55 "port": &schema.Schema{ 56 Type: schema.TypeInt, 57 Computed: true, 58 }, 59 60 "publicly_accessible": &schema.Schema{ 61 Type: schema.TypeBool, 62 Optional: true, 63 Default: false, 64 }, 65 66 "instance_class": &schema.Schema{ 67 Type: schema.TypeString, 68 Required: true, 69 }, 70 71 "db_parameter_group_name": &schema.Schema{ 72 Type: schema.TypeString, 73 Optional: true, 74 Computed: true, 75 }, 76 77 // apply_immediately is used to determine when the update modifications 78 // take place. 79 // See http://docs.aws.amazon.com/AmazonRDS/latest/UserGuide/Overview.DBInstance.Modifying.html 80 "apply_immediately": &schema.Schema{ 81 Type: schema.TypeBool, 82 Optional: true, 83 Computed: true, 84 }, 85 86 "kms_key_id": &schema.Schema{ 87 Type: schema.TypeString, 88 Optional: true, 89 Computed: true, 90 ForceNew: true, 91 }, 92 93 "storage_encrypted": &schema.Schema{ 94 Type: schema.TypeBool, 95 Optional: true, 96 Default: false, 97 ForceNew: true, 98 }, 99 100 "tags": tagsSchema(), 101 }, 102 } 103 } 104 105 func resourceAwsRDSClusterInstanceCreate(d *schema.ResourceData, meta interface{}) error { 106 conn := meta.(*AWSClient).rdsconn 107 tags := tagsFromMapRDS(d.Get("tags").(map[string]interface{})) 108 109 createOpts := &rds.CreateDBInstanceInput{ 110 DBInstanceClass: aws.String(d.Get("instance_class").(string)), 111 DBClusterIdentifier: aws.String(d.Get("cluster_identifier").(string)), 112 Engine: aws.String("aurora"), 113 PubliclyAccessible: aws.Bool(d.Get("publicly_accessible").(bool)), 114 Tags: tags, 115 } 116 117 if attr, ok := d.GetOk("db_parameter_group_name"); ok { 118 createOpts.DBParameterGroupName = aws.String(attr.(string)) 119 } 120 121 if v := d.Get("identifier").(string); v != "" { 122 createOpts.DBInstanceIdentifier = aws.String(v) 123 } else { 124 createOpts.DBInstanceIdentifier = aws.String(resource.UniqueId()) 125 } 126 127 if attr, ok := d.GetOk("db_subnet_group_name"); ok { 128 createOpts.DBSubnetGroupName = aws.String(attr.(string)) 129 } 130 131 log.Printf("[DEBUG] Creating RDS DB Instance opts: %s", createOpts) 132 resp, err := conn.CreateDBInstance(createOpts) 133 if err != nil { 134 return err 135 } 136 137 d.SetId(*resp.DBInstance.DBInstanceIdentifier) 138 139 // reuse db_instance refresh func 140 stateConf := &resource.StateChangeConf{ 141 Pending: []string{"creating", "backing-up", "modifying"}, 142 Target: []string{"available"}, 143 Refresh: resourceAwsDbInstanceStateRefreshFunc(d, meta), 144 Timeout: 40 * time.Minute, 145 MinTimeout: 10 * time.Second, 146 Delay: 10 * time.Second, 147 } 148 149 // Wait, catching any errors 150 _, err = stateConf.WaitForState() 151 if err != nil { 152 return err 153 } 154 155 return resourceAwsRDSClusterInstanceRead(d, meta) 156 } 157 158 func resourceAwsRDSClusterInstanceRead(d *schema.ResourceData, meta interface{}) error { 159 db, err := resourceAwsDbInstanceRetrieve(d, meta) 160 // Errors from this helper are always reportable 161 if err != nil { 162 return fmt.Errorf("[WARN] Error on retrieving RDS Cluster Instance (%s): %s", d.Id(), err) 163 } 164 // A nil response means "not found" 165 if db == nil { 166 log.Printf("[WARN] RDS Cluster Instance (%s): not found, removing from state.", d.Id()) 167 d.SetId("") 168 return nil 169 } 170 171 // Retreive DB Cluster information, to determine if this Instance is a writer 172 conn := meta.(*AWSClient).rdsconn 173 resp, err := conn.DescribeDBClusters(&rds.DescribeDBClustersInput{ 174 DBClusterIdentifier: db.DBClusterIdentifier, 175 }) 176 177 var dbc *rds.DBCluster 178 for _, c := range resp.DBClusters { 179 if *c.DBClusterIdentifier == *db.DBClusterIdentifier { 180 dbc = c 181 } 182 } 183 184 if dbc == nil { 185 return fmt.Errorf("[WARN] Error finding RDS Cluster (%s) for Cluster Instance (%s): %s", 186 *db.DBClusterIdentifier, *db.DBInstanceIdentifier, err) 187 } 188 189 for _, m := range dbc.DBClusterMembers { 190 if *db.DBInstanceIdentifier == *m.DBInstanceIdentifier { 191 if *m.IsClusterWriter == true { 192 d.Set("writer", true) 193 } else { 194 d.Set("writer", false) 195 } 196 } 197 } 198 199 if db.Endpoint != nil { 200 d.Set("endpoint", db.Endpoint.Address) 201 d.Set("port", db.Endpoint.Port) 202 } 203 204 d.Set("publicly_accessible", db.PubliclyAccessible) 205 d.Set("cluster_identifier", db.DBClusterIdentifier) 206 d.Set("instance_class", db.DBInstanceClass) 207 d.Set("identifier", db.DBInstanceIdentifier) 208 d.Set("storage_encrypted", db.StorageEncrypted) 209 210 if len(db.DBParameterGroups) > 0 { 211 d.Set("db_parameter_group_name", db.DBParameterGroups[0].DBParameterGroupName) 212 } 213 214 // Fetch and save tags 215 arn, err := buildRDSARN(d.Id(), meta) 216 if err != nil { 217 log.Printf("[DEBUG] Error building ARN for RDS Cluster Instance (%s), not setting Tags", *db.DBInstanceIdentifier) 218 } else { 219 if err := saveTagsRDS(conn, d, arn); err != nil { 220 log.Printf("[WARN] Failed to save tags for RDS Cluster Instance (%s): %s", *db.DBClusterIdentifier, err) 221 } 222 } 223 224 return nil 225 } 226 227 func resourceAwsRDSClusterInstanceUpdate(d *schema.ResourceData, meta interface{}) error { 228 conn := meta.(*AWSClient).rdsconn 229 requestUpdate := false 230 231 req := &rds.ModifyDBInstanceInput{ 232 ApplyImmediately: aws.Bool(d.Get("apply_immediately").(bool)), 233 DBInstanceIdentifier: aws.String(d.Id()), 234 } 235 236 if d.HasChange("db_parameter_group_name") { 237 req.DBParameterGroupName = aws.String(d.Get("db_parameter_group_name").(string)) 238 requestUpdate = true 239 240 } 241 242 if d.HasChange("instance_class") { 243 req.DBInstanceClass = aws.String(d.Get("instance_class").(string)) 244 requestUpdate = true 245 246 } 247 248 log.Printf("[DEBUG] Send DB Instance Modification request: %#v", requestUpdate) 249 if requestUpdate { 250 log.Printf("[DEBUG] DB Instance Modification request: %#v", req) 251 _, err := conn.ModifyDBInstance(req) 252 if err != nil { 253 return fmt.Errorf("Error modifying DB Instance %s: %s", d.Id(), err) 254 } 255 256 // reuse db_instance refresh func 257 stateConf := &resource.StateChangeConf{ 258 Pending: []string{"creating", "backing-up", "modifying"}, 259 Target: []string{"available"}, 260 Refresh: resourceAwsDbInstanceStateRefreshFunc(d, meta), 261 Timeout: 40 * time.Minute, 262 MinTimeout: 10 * time.Second, 263 Delay: 10 * time.Second, 264 } 265 266 // Wait, catching any errors 267 _, err = stateConf.WaitForState() 268 if err != nil { 269 return err 270 } 271 272 } 273 274 if arn, err := buildRDSARN(d.Id(), meta); err == nil { 275 if err := setTagsRDS(conn, d, arn); err != nil { 276 return err 277 } 278 } 279 280 return resourceAwsRDSClusterInstanceRead(d, meta) 281 } 282 283 func resourceAwsRDSClusterInstanceDelete(d *schema.ResourceData, meta interface{}) error { 284 conn := meta.(*AWSClient).rdsconn 285 286 log.Printf("[DEBUG] RDS Cluster Instance destroy: %v", d.Id()) 287 288 opts := rds.DeleteDBInstanceInput{DBInstanceIdentifier: aws.String(d.Id())} 289 290 log.Printf("[DEBUG] RDS Cluster Instance destroy configuration: %s", opts) 291 if _, err := conn.DeleteDBInstance(&opts); err != nil { 292 return err 293 } 294 295 // re-uses db_instance refresh func 296 log.Println("[INFO] Waiting for RDS Cluster Instance to be destroyed") 297 stateConf := &resource.StateChangeConf{ 298 Pending: []string{"modifying", "deleting"}, 299 Target: []string{}, 300 Refresh: resourceAwsDbInstanceStateRefreshFunc(d, meta), 301 Timeout: 40 * time.Minute, 302 MinTimeout: 10 * time.Second, 303 } 304 305 if _, err := stateConf.WaitForState(); err != nil { 306 return err 307 } 308 309 return nil 310 311 }