github.com/kwoods/terraform@v0.6.11-0.20160809170336-13497db7138e/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 "monitoring_role_arn": &schema.Schema{ 101 Type: schema.TypeString, 102 Optional: true, 103 Computed: true, 104 }, 105 106 "monitoring_interval": &schema.Schema{ 107 Type: schema.TypeInt, 108 Optional: true, 109 Default: 0, 110 }, 111 112 "tags": tagsSchema(), 113 }, 114 } 115 } 116 117 func resourceAwsRDSClusterInstanceCreate(d *schema.ResourceData, meta interface{}) error { 118 conn := meta.(*AWSClient).rdsconn 119 tags := tagsFromMapRDS(d.Get("tags").(map[string]interface{})) 120 121 createOpts := &rds.CreateDBInstanceInput{ 122 DBInstanceClass: aws.String(d.Get("instance_class").(string)), 123 DBClusterIdentifier: aws.String(d.Get("cluster_identifier").(string)), 124 Engine: aws.String("aurora"), 125 PubliclyAccessible: aws.Bool(d.Get("publicly_accessible").(bool)), 126 Tags: tags, 127 } 128 129 if attr, ok := d.GetOk("db_parameter_group_name"); ok { 130 createOpts.DBParameterGroupName = aws.String(attr.(string)) 131 } 132 133 if v := d.Get("identifier").(string); v != "" { 134 createOpts.DBInstanceIdentifier = aws.String(v) 135 } else { 136 createOpts.DBInstanceIdentifier = aws.String(resource.UniqueId()) 137 } 138 139 if attr, ok := d.GetOk("db_subnet_group_name"); ok { 140 createOpts.DBSubnetGroupName = aws.String(attr.(string)) 141 } 142 143 if attr, ok := d.GetOk("monitoring_role_arn"); ok { 144 createOpts.MonitoringRoleArn = aws.String(attr.(string)) 145 } 146 147 if attr, ok := d.GetOk("monitoring_interval"); ok { 148 createOpts.MonitoringInterval = aws.Int64(int64(attr.(int))) 149 } 150 151 log.Printf("[DEBUG] Creating RDS DB Instance opts: %s", createOpts) 152 resp, err := conn.CreateDBInstance(createOpts) 153 if err != nil { 154 return err 155 } 156 157 d.SetId(*resp.DBInstance.DBInstanceIdentifier) 158 159 // reuse db_instance refresh func 160 stateConf := &resource.StateChangeConf{ 161 Pending: []string{"creating", "backing-up", "modifying"}, 162 Target: []string{"available"}, 163 Refresh: resourceAwsDbInstanceStateRefreshFunc(d, meta), 164 Timeout: 40 * time.Minute, 165 MinTimeout: 10 * time.Second, 166 Delay: 10 * time.Second, 167 } 168 169 // Wait, catching any errors 170 _, err = stateConf.WaitForState() 171 if err != nil { 172 return err 173 } 174 175 return resourceAwsRDSClusterInstanceRead(d, meta) 176 } 177 178 func resourceAwsRDSClusterInstanceRead(d *schema.ResourceData, meta interface{}) error { 179 db, err := resourceAwsDbInstanceRetrieve(d, meta) 180 // Errors from this helper are always reportable 181 if err != nil { 182 return fmt.Errorf("[WARN] Error on retrieving RDS Cluster Instance (%s): %s", d.Id(), err) 183 } 184 // A nil response means "not found" 185 if db == nil { 186 log.Printf("[WARN] RDS Cluster Instance (%s): not found, removing from state.", d.Id()) 187 d.SetId("") 188 return nil 189 } 190 191 // Retreive DB Cluster information, to determine if this Instance is a writer 192 conn := meta.(*AWSClient).rdsconn 193 resp, err := conn.DescribeDBClusters(&rds.DescribeDBClustersInput{ 194 DBClusterIdentifier: db.DBClusterIdentifier, 195 }) 196 197 var dbc *rds.DBCluster 198 for _, c := range resp.DBClusters { 199 if *c.DBClusterIdentifier == *db.DBClusterIdentifier { 200 dbc = c 201 } 202 } 203 204 if dbc == nil { 205 return fmt.Errorf("[WARN] Error finding RDS Cluster (%s) for Cluster Instance (%s): %s", 206 *db.DBClusterIdentifier, *db.DBInstanceIdentifier, err) 207 } 208 209 for _, m := range dbc.DBClusterMembers { 210 if *db.DBInstanceIdentifier == *m.DBInstanceIdentifier { 211 if *m.IsClusterWriter == true { 212 d.Set("writer", true) 213 } else { 214 d.Set("writer", false) 215 } 216 } 217 } 218 219 if db.Endpoint != nil { 220 d.Set("endpoint", db.Endpoint.Address) 221 d.Set("port", db.Endpoint.Port) 222 } 223 224 d.Set("publicly_accessible", db.PubliclyAccessible) 225 d.Set("cluster_identifier", db.DBClusterIdentifier) 226 d.Set("instance_class", db.DBInstanceClass) 227 d.Set("identifier", db.DBInstanceIdentifier) 228 d.Set("storage_encrypted", db.StorageEncrypted) 229 230 if db.MonitoringInterval != nil { 231 d.Set("monitoring_interval", db.MonitoringInterval) 232 } 233 234 if db.MonitoringRoleArn != nil { 235 d.Set("monitoring_role_arn", db.MonitoringRoleArn) 236 } 237 238 if len(db.DBParameterGroups) > 0 { 239 d.Set("db_parameter_group_name", db.DBParameterGroups[0].DBParameterGroupName) 240 } 241 242 // Fetch and save tags 243 arn, err := buildRDSARN(d.Id(), meta.(*AWSClient).accountid, meta.(*AWSClient).region) 244 if err != nil { 245 log.Printf("[DEBUG] Error building ARN for RDS Cluster Instance (%s), not setting Tags", *db.DBInstanceIdentifier) 246 } else { 247 if err := saveTagsRDS(conn, d, arn); err != nil { 248 log.Printf("[WARN] Failed to save tags for RDS Cluster Instance (%s): %s", *db.DBClusterIdentifier, err) 249 } 250 } 251 252 return nil 253 } 254 255 func resourceAwsRDSClusterInstanceUpdate(d *schema.ResourceData, meta interface{}) error { 256 conn := meta.(*AWSClient).rdsconn 257 requestUpdate := false 258 259 req := &rds.ModifyDBInstanceInput{ 260 ApplyImmediately: aws.Bool(d.Get("apply_immediately").(bool)), 261 DBInstanceIdentifier: aws.String(d.Id()), 262 } 263 264 if d.HasChange("db_parameter_group_name") { 265 req.DBParameterGroupName = aws.String(d.Get("db_parameter_group_name").(string)) 266 requestUpdate = true 267 268 } 269 270 if d.HasChange("instance_class") { 271 req.DBInstanceClass = aws.String(d.Get("instance_class").(string)) 272 requestUpdate = true 273 274 } 275 276 if d.HasChange("monitoring_role_arn") { 277 d.SetPartial("monitoring_role_arn") 278 req.MonitoringRoleArn = aws.String(d.Get("monitoring_role_arn").(string)) 279 requestUpdate = true 280 } 281 282 if d.HasChange("monitoring_interval") { 283 d.SetPartial("monitoring_interval") 284 req.MonitoringInterval = aws.Int64(int64(d.Get("monitoring_interval").(int))) 285 requestUpdate = true 286 } 287 288 log.Printf("[DEBUG] Send DB Instance Modification request: %#v", requestUpdate) 289 if requestUpdate { 290 log.Printf("[DEBUG] DB Instance Modification request: %#v", req) 291 _, err := conn.ModifyDBInstance(req) 292 if err != nil { 293 return fmt.Errorf("Error modifying DB Instance %s: %s", d.Id(), err) 294 } 295 296 // reuse db_instance refresh func 297 stateConf := &resource.StateChangeConf{ 298 Pending: []string{"creating", "backing-up", "modifying"}, 299 Target: []string{"available"}, 300 Refresh: resourceAwsDbInstanceStateRefreshFunc(d, meta), 301 Timeout: 40 * time.Minute, 302 MinTimeout: 10 * time.Second, 303 Delay: 10 * time.Second, 304 } 305 306 // Wait, catching any errors 307 _, err = stateConf.WaitForState() 308 if err != nil { 309 return err 310 } 311 312 } 313 314 if arn, err := buildRDSARN(d.Id(), meta.(*AWSClient).accountid, meta.(*AWSClient).region); err == nil { 315 if err := setTagsRDS(conn, d, arn); err != nil { 316 return err 317 } 318 } 319 320 return resourceAwsRDSClusterInstanceRead(d, meta) 321 } 322 323 func resourceAwsRDSClusterInstanceDelete(d *schema.ResourceData, meta interface{}) error { 324 conn := meta.(*AWSClient).rdsconn 325 326 log.Printf("[DEBUG] RDS Cluster Instance destroy: %v", d.Id()) 327 328 opts := rds.DeleteDBInstanceInput{DBInstanceIdentifier: aws.String(d.Id())} 329 330 log.Printf("[DEBUG] RDS Cluster Instance destroy configuration: %s", opts) 331 if _, err := conn.DeleteDBInstance(&opts); err != nil { 332 return err 333 } 334 335 // re-uses db_instance refresh func 336 log.Println("[INFO] Waiting for RDS Cluster Instance to be destroyed") 337 stateConf := &resource.StateChangeConf{ 338 Pending: []string{"modifying", "deleting"}, 339 Target: []string{}, 340 Refresh: resourceAwsDbInstanceStateRefreshFunc(d, meta), 341 Timeout: 40 * time.Minute, 342 MinTimeout: 10 * time.Second, 343 } 344 345 if _, err := stateConf.WaitForState(); err != nil { 346 return err 347 } 348 349 return nil 350 351 }