alicloud/resource_alicloud_kvstore_instance.go (1,611 lines of code) (raw):

package alicloud import ( "encoding/json" "fmt" "log" "strconv" "strings" "time" "github.com/aliyun/alibaba-cloud-sdk-go/sdk/requests" r_kvstore "github.com/aliyun/alibaba-cloud-sdk-go/services/r-kvstore" "github.com/aliyun/terraform-provider-alicloud/alicloud/connectivity" "github.com/hashicorp/terraform-plugin-sdk/helper/hashcode" "github.com/hashicorp/terraform-plugin-sdk/helper/resource" "github.com/hashicorp/terraform-plugin-sdk/helper/schema" ) func resourceAliCloudKvstoreInstance() *schema.Resource { return &schema.Resource{ Create: resourceAliCloudKvstoreInstanceCreate, Read: resourceAliCloudKvstoreInstanceRead, Update: resourceAliCloudKvstoreInstanceUpdate, Delete: resourceAliCloudKvstoreInstanceDelete, Importer: &schema.ResourceImporter{ State: schema.ImportStatePassthrough, }, Timeouts: &schema.ResourceTimeout{ Create: schema.DefaultTimeout(20 * time.Minute), Update: schema.DefaultTimeout(40 * time.Minute), Delete: schema.DefaultTimeout(20 * time.Minute), }, Schema: map[string]*schema.Schema{ "auto_renew": { Type: schema.TypeBool, Optional: true, DiffSuppressFunc: redisPostPaidDiffSuppressFunc, }, "auto_renew_period": { Type: schema.TypeInt, Optional: true, Default: 1, ValidateFunc: IntBetween(1, 12), DiffSuppressFunc: redisPostPaidAndRenewDiffSuppressFunc, }, "auto_use_coupon": { Type: schema.TypeBool, Optional: true, ForceNew: true, Default: false, DiffSuppressFunc: func(k, old, new string, d *schema.ResourceData) bool { return !d.IsNewResource() }, }, "backup_id": { Type: schema.TypeString, Optional: true, ForceNew: true, }, "backup_period": { Type: schema.TypeSet, Optional: true, Computed: true, Elem: &schema.Schema{ Type: schema.TypeString, }, }, "backup_time": { Type: schema.TypeString, Optional: true, Computed: true, }, "bandwidth": { Type: schema.TypeInt, Optional: true, Computed: true, }, "business_info": { Type: schema.TypeString, Optional: true, }, "capacity": { Type: schema.TypeInt, Optional: true, ForceNew: true, Computed: true, }, "config": { Type: schema.TypeMap, Optional: true, Computed: true, ConflictsWith: []string{"parameters"}, ValidateFunc: validateRedisConfig, }, "connection_domain": { Type: schema.TypeString, Computed: true, }, "coupon_no": { Type: schema.TypeString, Optional: true, }, "db_instance_name": { Type: schema.TypeString, Optional: true, Computed: true, ConflictsWith: []string{"instance_name"}, }, "instance_name": { Type: schema.TypeString, Optional: true, Computed: true, ConflictsWith: []string{"db_instance_name"}, Deprecated: "Field `instance_name` has been deprecated from version 1.101.0. Use `db_instance_name` instead.", }, "dedicated_host_group_id": { Type: schema.TypeString, Optional: true, ForceNew: true, }, "dry_run": { Type: schema.TypeBool, Optional: true, }, "enable_backup_log": { Type: schema.TypeInt, Optional: true, ValidateFunc: IntInSlice([]int{0, 1}), Default: 0, }, "end_time": { Type: schema.TypeString, Computed: true, }, "engine_version": { Type: schema.TypeString, Optional: true, ValidateFunc: StringInSlice([]string{"2.8", "4.0", "5.0", "6.0", "7.0"}, false), Computed: true, }, "force_upgrade": { Type: schema.TypeBool, Optional: true, Default: true, }, "global_instance": { Type: schema.TypeBool, Optional: true, ForceNew: true, Default: false, DiffSuppressFunc: func(k, old, new string, d *schema.ResourceData) bool { return !d.IsNewResource() }, }, "global_instance_id": { Type: schema.TypeString, Optional: true, }, "instance_class": { Type: schema.TypeString, Optional: true, }, "instance_release_protection": { Type: schema.TypeBool, Optional: true, }, "instance_type": { Type: schema.TypeString, Optional: true, ForceNew: true, ValidateFunc: StringInSlice([]string{"Memcache", "Redis"}, false), Default: "Redis", }, "maintain_end_time": { Type: schema.TypeString, Optional: true, Computed: true, }, "maintain_start_time": { Type: schema.TypeString, Optional: true, Computed: true, }, "effective_time": { Type: schema.TypeString, Optional: true, ValidateFunc: StringInSlice([]string{"Immediately", "MaintainTime"}, false), }, "modify_mode": { Type: schema.TypeInt, Optional: true, Removed: "Field `modify_mode` has been removed from provider version 1.216.0.", }, "node_type": { Type: schema.TypeString, Optional: true, ForceNew: true, Computed: true, ValidateFunc: StringInSlice([]string{"MASTER_SLAVE", "STAND_ALONE", "double", "single"}, false), Deprecated: "Field 'node_type' has been deprecated from version 1.120.1", }, "order_type": { Type: schema.TypeString, Optional: true, ValidateFunc: StringInSlice([]string{"UPGRADE", "DOWNGRADE"}, false), Default: "UPGRADE", }, "password": { Type: schema.TypeString, Optional: true, Sensitive: true, }, "payment_type": { Type: schema.TypeString, Optional: true, Computed: true, ValidateFunc: StringInSlice([]string{"PostPaid", "PrePaid"}, false), ConflictsWith: []string{"instance_charge_type"}, }, "instance_charge_type": { Type: schema.TypeString, Optional: true, Computed: true, ValidateFunc: StringInSlice([]string{"PostPaid", "PrePaid"}, false), Deprecated: "Field 'instance_charge_type' has been deprecated from version 1.101.0. Use 'payment_type' instead.", ConflictsWith: []string{"payment_type"}, }, "period": { Type: schema.TypeString, Optional: true, ValidateFunc: StringInSlice([]string{"1", "12", "2", "24", "3", "36", "4", "5", "6", "7", "8", "9"}, false), DiffSuppressFunc: redisPostPaidDiffSuppressFunc, }, "port": { Type: schema.TypeInt, Optional: true, }, "private_connection_prefix": { Type: schema.TypeString, Optional: true, }, "private_connection_port": { Type: schema.TypeString, Optional: true, Computed: true, }, "private_ip": { Type: schema.TypeString, Optional: true, ForceNew: true, Computed: true, }, "qps": { Type: schema.TypeInt, Computed: true, }, "resource_group_id": { Type: schema.TypeString, Optional: true, Computed: true, }, "restore_time": { Type: schema.TypeString, Optional: true, ForceNew: true, }, "ssl_enable": { Type: schema.TypeString, Optional: true, Computed: true, ValidateFunc: StringInSlice([]string{"Disable", "Enable", "Update"}, false), }, "secondary_zone_id": { Type: schema.TypeString, Optional: true, }, "security_group_id": { Type: schema.TypeString, Optional: true, DiffSuppressFunc: redisSecurityGroupIdDiffSuppressFunc, }, "security_ip_group_attribute": { Type: schema.TypeString, Optional: true, }, "security_ip_group_name": { Type: schema.TypeString, Optional: true, Computed: true, }, "security_ips": { Type: schema.TypeSet, Optional: true, Computed: true, MinItems: 1, Elem: &schema.Schema{ Type: schema.TypeString, ValidateFunc: StringLenAtLeast(1), }, }, "srcdb_instance_id": { Type: schema.TypeString, Optional: true, ForceNew: true, }, "status": { Type: schema.TypeString, Computed: true, }, "tags": tagsSchema(), "vswitch_id": { Type: schema.TypeString, Optional: true, }, "vpc_auth_mode": { Type: schema.TypeString, Optional: true, ValidateFunc: StringInSlice([]string{"Close", "Open"}, false), Default: "Open", }, "zone_id": { Type: schema.TypeString, Optional: true, Computed: true, ConflictsWith: []string{"availability_zone"}, }, "availability_zone": { Type: schema.TypeString, Optional: true, Computed: true, Deprecated: "Field 'availability_zone' has been deprecated from version 1.101.0. Use 'zone_id' instead.", ConflictsWith: []string{"zone_id"}, }, "kms_encrypted_password": { Type: schema.TypeString, Optional: true, DiffSuppressFunc: kmsDiffSuppressFunc, }, "kms_encryption_context": { Type: schema.TypeMap, Optional: true, DiffSuppressFunc: func(k, old, new string, d *schema.ResourceData) bool { return d.Get("kms_encrypted_password").(string) == "" }, Elem: schema.TypeString, }, "enable_public": { Type: schema.TypeBool, Optional: true, Computed: true, Deprecated: "Field 'enable_public' has been deprecated from version 1.101.0. Please use resource 'alicloud_kvstore_connection' instead.", }, "connection_string_prefix": { Type: schema.TypeString, Optional: true, Deprecated: "Field 'connection_string_prefix' has been deprecated from version 1.101.0. Please use resource 'alicloud_kvstore_connection' instead.", }, "connection_string": { Type: schema.TypeString, Computed: true, Deprecated: "Field 'connection_string' has been deprecated from version 1.101.0. Please use resource 'alicloud_kvstore_connection' instead.", }, "parameters": { Type: schema.TypeSet, Elem: &schema.Resource{ Schema: map[string]*schema.Schema{ "name": { Type: schema.TypeString, Optional: true, Deprecated: "Field 'parameters' has been deprecated from version 1.101.0. Use 'config' instead.", }, "value": { Type: schema.TypeString, Optional: true, Deprecated: "Field 'parameters' has been deprecated from version 1.101.0. Use 'config' instead.", }, }, }, Set: func(v interface{}) int { return hashcode.String( v.(map[string]interface{})["name"].(string) + "|" + v.(map[string]interface{})["value"].(string)) }, Optional: true, Computed: true, Deprecated: "Field 'parameters' has been deprecated from version 1.101.0. Use 'config' instead.", ConflictsWith: []string{"config"}, }, "tde_status": { Type: schema.TypeString, Optional: true, Computed: true, ValidateFunc: StringInSlice([]string{"Enabled"}, false), }, "encryption_name": { Type: schema.TypeString, Optional: true, Computed: true, }, "encryption_key": { Type: schema.TypeString, Optional: true, Computed: true, }, "role_arn": { Type: schema.TypeString, Optional: true, Computed: true, }, "shard_count": { Type: schema.TypeInt, Optional: true, Computed: true, }, "read_only_count": { Type: schema.TypeInt, Optional: true, }, "slave_read_only_count": { Type: schema.TypeInt, Optional: true, }, "is_auto_upgrade_open": { Type: schema.TypeString, Optional: true, Computed: true, }, }, } } func resourceAliCloudKvstoreInstanceCreate(d *schema.ResourceData, meta interface{}) error { client := meta.(*connectivity.AliyunClient) r_kvstoreService := R_kvstoreService{client} var response map[string]interface{} var err error action := "CreateInstance" request := make(map[string]interface{}) request["RegionId"] = client.RegionId request["NetworkType"] = "CLASSIC" if v, ok := d.GetOkExists("auto_renew"); ok { request["AutoRenew"] = convertBoolToString(v.(bool)) } if v, ok := d.GetOk("auto_renew_period"); ok { request["AutoRenewPeriod"] = convertIntergerToString(v.(int)) } if v, ok := d.GetOkExists("auto_use_coupon"); ok { request["AutoUseCoupon"] = convertBoolToString(v.(bool)) } if v, ok := d.GetOk("backup_id"); ok { request["BackupId"] = v } if v, ok := d.GetOk("business_info"); ok { request["BusinessInfo"] = v } if v, ok := d.GetOkExists("capacity"); ok { request["Capacity"] = v } if v, ok := d.GetOk("coupon_no"); ok { request["CouponNo"] = v } if v, ok := d.GetOk("db_instance_name"); ok { request["InstanceName"] = v } else if v, ok := d.GetOk("instance_name"); ok { request["InstanceName"] = v } if v, ok := d.GetOk("dedicated_host_group_id"); ok { request["DedicatedHostGroupId"] = v } if v, ok := d.GetOkExists("dry_run"); ok { request["DryRun"] = v } if v, ok := d.GetOk("engine_version"); ok { request["EngineVersion"] = v } if v, ok := d.GetOkExists("global_instance"); ok { request["GlobalInstance"] = v } if v, ok := d.GetOk("global_instance_id"); ok { request["GlobalInstanceId"] = v } if v, ok := d.GetOk("instance_class"); ok { request["InstanceClass"] = v } if v, ok := d.GetOk("instance_type"); ok { request["InstanceType"] = v } if v, ok := d.GetOk("node_type"); ok { request["NodeType"] = v } request["Password"] = d.Get("password") if fmt.Sprint(request["Password"]) == "" { if v, ok := d.GetOk("kms_encrypted_password"); ok && fmt.Sprint(v) != "" { kmsService := KmsService{client} decryptResp, err := kmsService.Decrypt(v.(string), d.Get("kms_encryption_context").(map[string]interface{})) if err != nil { return WrapError(err) } request["Password"] = decryptResp } } if v, ok := d.GetOk("payment_type"); ok { request["ChargeType"] = v } else if v, ok := d.GetOk("instance_charge_type"); ok { request["ChargeType"] = v } if v, ok := d.GetOk("period"); ok { request["Period"] = v } if v, ok := d.GetOk("private_ip"); ok { request["PrivateIpAddress"] = v } if v, ok := d.GetOk("resource_group_id"); ok { request["ResourceGroupId"] = v } if v, ok := d.GetOk("restore_time"); ok { request["RestoreTime"] = v } if v, ok := d.GetOk("srcdb_instance_id"); ok { request["SrcDBInstanceId"] = v } if v, ok := d.GetOk("zone_id"); ok { request["ZoneId"] = v } else if v, ok := d.GetOk("availability_zone"); ok { request["ZoneId"] = v } if v, ok := d.GetOk("secondary_zone_id"); ok { request["SecondaryZoneId"] = v } if v, ok := d.GetOkExists("shard_count"); ok { request["ShardCount"] = v } if v, ok := d.GetOkExists("read_only_count"); ok { request["ReadOnlyCount"] = v } if v, ok := d.GetOkExists("slave_read_only_count"); ok { request["SlaveReadOnlyCount"] = v } vswitchId := Trim(d.Get("vswitch_id").(string)) if vswitchId != "" { vpcService := VpcService{client} vsw, err := vpcService.DescribeVSwitch(vswitchId) if err != nil { return WrapError(err) } request["NetworkType"] = "VPC" request["VpcId"] = vsw.VpcId request["VSwitchId"] = vswitchId if fmt.Sprint(request["ZoneId"]) == "" { request["ZoneId"] = vsw.ZoneId } } wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutCreate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, request, true) if err != nil { if NoCodeRegexRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, request) if err != nil { return WrapErrorf(err, DefaultErrorMsg, "alicloud_kvstore_instance", action, AlibabaCloudSdkGoERROR) } d.SetId(fmt.Sprint(response["InstanceId"])) stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutCreate), 180*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } return resourceAliCloudKvstoreInstanceUpdate(d, meta) } func resourceAliCloudKvstoreInstanceRead(d *schema.ResourceData, meta interface{}) error { client := meta.(*connectivity.AliyunClient) r_kvstoreService := R_kvstoreService{client} object, err := r_kvstoreService.DescribeKvstoreInstance(d.Id()) if err != nil { if !d.IsNewResource() && NotFoundError(err) { log.Printf("[DEBUG] Resource alicloud_kvstore_instance r_kvstoreService.DescribeKvstoreInstance Failed!!! %s", err) d.SetId("") return nil } return WrapError(err) } d.Set("enable_public", false) d.Set("connection_string", "") netInfoList, err := r_kvstoreService.DescribeKvStoreInstanceNetInfo(d.Id()) if err != nil { return WrapError(err) } for _, netInfo := range netInfoList { netInfoArg := netInfo.(map[string]interface{}) if fmt.Sprint(netInfoArg["DBInstanceNetType"]) == "0" { d.Set("enable_public", true) d.Set("connection_string", netInfoArg["ConnectionString"]) } if fmt.Sprint(netInfoArg["DBInstanceNetType"]) == "2" { if _, ok := netInfoArg["IsSlaveProxy"]; !ok { d.Set("private_connection_port", netInfoArg["Port"]) } } } d.Set("bandwidth", object["Bandwidth"]) d.Set("capacity", object["Capacity"]) gotConfigs := make(map[string]interface{}) if v, ok := d.GetOk("config"); ok && v != nil { gotConfigs = v.(map[string]interface{}) } if object["Config"] != "" { configMap := make(map[string]string) config, err := convertJsonStringToMap(fmt.Sprint(object["Config"])) if err != nil { return WrapError(err) } for k, v := range config { // There is an openapi bug that it will return all of configs even through the config does not specified by user. // This workaround is not prefect when user set the configs if _, ok := gotConfigs[k]; !ok && len(gotConfigs) > 0 { continue } configMap[k] = fmt.Sprint(v) } d.Set("config", configMap) } d.Set("connection_domain", object["ConnectionDomain"]) d.Set("db_instance_name", object["InstanceName"]) d.Set("instance_name", object["InstanceName"]) d.Set("end_time", object["EndTime"]) d.Set("engine_version", object["EngineVersion"]) d.Set("instance_class", object["RealInstanceClass"]) d.Set("instance_release_protection", object["InstanceReleaseProtection"]) d.Set("instance_type", object["InstanceType"]) d.Set("maintain_end_time", object["MaintainEndTime"]) d.Set("maintain_start_time", object["MaintainStartTime"]) d.Set("node_type", object["NodeType"]) d.Set("payment_type", object["ChargeType"]) d.Set("instance_charge_type", object["ChargeType"]) d.Set("private_ip", object["PrivateIp"]) d.Set("qps", object["QPS"]) d.Set("resource_group_id", object["ResourceGroupId"]) d.Set("shard_count", object["ShardCount"]) d.Set("read_only_count", object["ReadOnlyCount"]) d.Set("slave_read_only_count", object["SlaveReadOnlyCount"]) d.Set("status", object["InstanceStatus"]) if v, ok := object["Tags"].(map[string]interface{}); ok { d.Set("tags", tagsToMap(v["Tag"])) } d.Set("vswitch_id", object["VSwitchId"]) d.Set("vpc_auth_mode", object["VpcAuthMode"]) d.Set("zone_id", object["ZoneId"]) d.Set("availability_zone", object["ZoneId"]) d.Set("secondary_zone_id", object["SecondaryZoneId"]) describeBackupPolicyObject, err := r_kvstoreService.DescribeBackupPolicy(d.Id()) if err != nil { return WrapError(err) } d.Set("backup_period", strings.Split(describeBackupPolicyObject.PreferredBackupPeriod, ",")) d.Set("backup_time", describeBackupPolicyObject.PreferredBackupTime) describeSecurityGroupConfigurationObject, err := r_kvstoreService.DescribeSecurityGroupConfiguration(d.Id()) if err != nil { return WrapError(err) } sgs := make([]string, 0) for _, sg := range describeSecurityGroupConfigurationObject.EcsSecurityGroupRelation { sgs = append(sgs, sg.SecurityGroupId) } d.Set("security_group_id", strings.Join(sgs, ",")) describeInstanceSSLObject, err := r_kvstoreService.DescribeInstanceSSL(d.Id()) if err != nil { // some engine does not support the ssl if !IsExpectedErrors(err, []string{"IncorrectEngineVersion"}) { return WrapError(err) } } else { d.Set("ssl_enable", describeInstanceSSLObject.SSLEnabled) } var securityIpGroupName string if v, ok := d.GetOk("security_ip_group_name"); ok { securityIpGroupName = v.(string) } describeSecurityIpsObject, err := r_kvstoreService.DescribeSecurityIps(d.Id(), securityIpGroupName) if err != nil { return WrapError(err) } d.Set("security_ip_group_attribute", describeSecurityIpsObject.SecurityIpGroupAttribute) d.Set("security_ip_group_name", describeSecurityIpsObject.SecurityIpGroupName) d.Set("security_ips", strings.Split(describeSecurityIpsObject.SecurityIpList, ",")) if fmt.Sprint(object["ChargeType"]) == string(PrePaid) { describeInstanceAutoRenewalAttributeObject, err := r_kvstoreService.DescribeInstanceAutoRenewalAttribute(d.Id()) if err != nil { return WrapError(err) } autoRenew, err := strconv.ParseBool(describeInstanceAutoRenewalAttributeObject.AutoRenew) if err != nil { // invalid request response return WrapError(err) } d.Set("auto_renew", autoRenew) d.Set("auto_renew_period", describeInstanceAutoRenewalAttributeObject.Duration) } //refresh parameters if err = refreshParameters(d, meta); err != nil { return WrapError(err) } if object["IsSupportTDE"].(bool) { rKvstoreService := RKvstoreService{client} describeTDEStatusObject, err := rKvstoreService.DescribeInstanceTDEStatus(d.Id()) if err != nil { return WrapError(err) } d.Set("tde_status", convertModifyInstanceTDERequest(describeTDEStatusObject["TDEStatus"])) encryptionKeyObject, err := rKvstoreService.DescribeEncryptionKey(d.Id()) if err != nil { return WrapError(err) } d.Set("encryption_key", encryptionKeyObject["EncryptionKey"]) d.Set("encryption_name", encryptionKeyObject["EncryptionName"]) d.Set("role_arn", encryptionKeyObject["RoleArn"]) } engineVersionObject, err := r_kvstoreService.DescribeKvStoreEngineVersion(d.Id()) if err != nil { return WrapError(err) } d.Set("is_auto_upgrade_open", engineVersionObject["IsAutoUpgradeOpen"]) return nil } func resourceAliCloudKvstoreInstanceUpdate(d *schema.ResourceData, meta interface{}) error { client := meta.(*connectivity.AliyunClient) r_kvstoreService := R_kvstoreService{client} var response map[string]interface{} var err error d.Partial(true) if d.HasChange("tags") { if err := r_kvstoreService.SetResourceTags(d, "INSTANCE"); err != nil { return WrapError(err) } d.SetPartial("tags") } if d.HasChange("config") { request := r_kvstore.CreateModifyInstanceConfigRequest() request.InstanceId = d.Id() respJson, err := convertMaptoJsonString(d.Get("config").(map[string]interface{})) if err != nil { return WrapErrorf(err, DefaultErrorMsg, "alicloud_kvstore_instance", request.GetActionName(), AlibabaCloudSdkGoERROR) } request.Config = respJson raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceConfig(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("config") } if d.HasChange("vpc_auth_mode") && d.Get("vswitch_id") != "" { request := r_kvstore.CreateModifyInstanceVpcAuthModeRequest() request.InstanceId = d.Id() request.VpcAuthMode = d.Get("vpc_auth_mode").(string) raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceVpcAuthMode(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("vpc_auth_mode") } if d.HasChange("security_group_id") { request := r_kvstore.CreateModifySecurityGroupConfigurationRequest() request.DBInstanceId = d.Id() request.SecurityGroupId = d.Get("security_group_id").(string) raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifySecurityGroupConfiguration(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("security_group_id") } update := false transformInstanceChargeTypeReq := map[string]interface{}{ "AutoPay": true, "InstanceId": d.Id(), } if !d.IsNewResource() && d.HasChange("payment_type") { update = true if v, ok := d.GetOk("payment_type"); ok { transformInstanceChargeTypeReq["ChargeType"] = v } } if !d.IsNewResource() && d.HasChange("instance_charge_type") { update = true if v, ok := d.GetOk("instance_charge_type"); ok { transformInstanceChargeTypeReq["ChargeType"] = v } } if v, ok := d.GetOk("period"); ok { transformInstanceChargeTypeReq["Period"] = v } if v, ok := d.GetOkExists("auto_renew"); ok { transformInstanceChargeTypeReq["AutoRenew"] = convertBoolToString(v.(bool)) } if v, ok := d.GetOkExists("auto_renew_period"); ok { transformInstanceChargeTypeReq["AutoRenewPeriod"] = v } if update { action := "TransformInstanceChargeType" wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, transformInstanceChargeTypeReq, false) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, transformInstanceChargeTypeReq) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } instanceStatusConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "InstanceStatus")) if _, err := instanceStatusConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } stateConf := BuildStateConf([]string{}, []string{"true"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "IsOrderCompleted")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("payment_type") d.SetPartial("instance_charge_type") } update = false request := r_kvstore.CreateModifyInstanceAutoRenewalAttributeRequest() request.DBInstanceId = d.Id() if !d.IsNewResource() && d.HasChange("auto_renew") { update = true } request.AutoRenew = convertBoolToString(d.Get("auto_renew").(bool)) if !d.IsNewResource() && d.HasChange("auto_renew_period") { update = true } request.Duration = convertIntergerToString(d.Get("auto_renew_period").(int)) if update { raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceAutoRenewalAttribute(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } d.SetPartial("auto_renew") d.SetPartial("auto_renew_period") } update = false modifyInstanceMaintainTimeReq := r_kvstore.CreateModifyInstanceMaintainTimeRequest() modifyInstanceMaintainTimeReq.InstanceId = d.Id() if d.HasChange("maintain_end_time") { update = true } modifyInstanceMaintainTimeReq.MaintainEndTime = d.Get("maintain_end_time").(string) if d.HasChange("maintain_start_time") { update = true } modifyInstanceMaintainTimeReq.MaintainStartTime = d.Get("maintain_start_time").(string) if update { raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceMaintainTime(modifyInstanceMaintainTimeReq) }) addDebug(modifyInstanceMaintainTimeReq.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), modifyInstanceMaintainTimeReq.GetActionName(), AlibabaCloudSdkGoERROR) } d.SetPartial("maintain_end_time") d.SetPartial("maintain_start_time") } if d.HasChange("ssl_enable") { request := r_kvstore.CreateModifyInstanceSSLRequest() request.InstanceId = d.Id() request.SSLEnabled = d.Get("ssl_enable").(string) raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceSSL(request) }) addDebug(request.GetActionName(), raw) if err != nil && !IsExpectedErrors(err, []string{"SSLDisableStateExistsFault"}) { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 5*time.Minute, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("ssl_enable") } if !d.IsNewResource() && d.HasChange("resource_group_id") { request := r_kvstore.CreateModifyResourceGroupRequest() request.InstanceId = d.Id() request.RegionId = client.RegionId request.ResourceGroupId = d.Get("resource_group_id").(string) raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyResourceGroup(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 60*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("resource_group_id") } update = false migrateToOtherZoneReq := map[string]interface{}{ "DBInstanceId": d.Id(), } if !d.IsNewResource() && d.HasChange("zone_id") { update = true if v, ok := d.GetOk("zone_id"); ok { migrateToOtherZoneReq["ZoneId"] = v } } if !d.IsNewResource() && d.HasChange("vswitch_id") { update = true } if v, ok := d.GetOk("vswitch_id"); ok { migrateToOtherZoneReq["VSwitchId"] = v } if !d.IsNewResource() && d.HasChange("secondary_zone_id") { update = true } if v, ok := d.GetOk("secondary_zone_id"); ok { migrateToOtherZoneReq["SecondaryZoneId"] = v } if v, ok := d.GetOkExists("read_only_count"); ok { migrateToOtherZoneReq["ReadOnlyCount"] = v } if v, ok := d.GetOkExists("slave_read_only_count"); ok { migrateToOtherZoneReq["SlaveReadOnlyCount"] = v } if !d.IsNewResource() && d.HasChange("availability_zone") { update = true if v, ok := d.GetOk("availability_zone"); ok { migrateToOtherZoneReq["ZoneId"] = v } } if v, ok := d.GetOk("effective_time"); ok { migrateToOtherZoneReq["EffectiveTime"] = v } if update { action := "MigrateToOtherZone" wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, migrateToOtherZoneReq, true) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, migrateToOtherZoneReq) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 300*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("vswitch_id") d.SetPartial("zone_id") d.SetPartial("availability_zone") d.SetPartial("secondary_zone_id") } update = false modifyBackupPolicyReq := r_kvstore.CreateModifyBackupPolicyRequest() modifyBackupPolicyReq.InstanceId = d.Id() if d.HasChange("backup_period") { update = true } modifyBackupPolicyReq.PreferredBackupPeriod = convertListToCommaSeparate(d.Get("backup_period").(*schema.Set).List()) if d.HasChange("backup_time") { update = true } modifyBackupPolicyReq.PreferredBackupTime = d.Get("backup_time").(string) if update { if _, ok := d.GetOk("enable_backup_log"); ok { modifyBackupPolicyReq.EnableBackupLog = requests.NewInteger(d.Get("enable_backup_log").(int)) } raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyBackupPolicy(modifyBackupPolicyReq) }) addDebug(modifyBackupPolicyReq.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), modifyBackupPolicyReq.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 60*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("backup_period") d.SetPartial("backup_time") } update = false modifyInstanceAttributeReq := r_kvstore.CreateModifyInstanceAttributeRequest() modifyInstanceAttributeReq.InstanceId = d.Id() if !d.IsNewResource() && (d.HasChange("db_instance_name") || d.HasChange("instance_name")) { update = true } if v, ok := d.GetOk("db_instance_name"); ok { modifyInstanceAttributeReq.InstanceName = v.(string) } else if v, ok := d.GetOk("instance_name"); ok { modifyInstanceAttributeReq.InstanceName = v.(string) } if d.HasChange("instance_release_protection") { update = true modifyInstanceAttributeReq.InstanceReleaseProtection = requests.NewBoolean(d.Get("instance_release_protection").(bool)) } if !d.IsNewResource() && (d.HasChange("password") || d.HasChange("kms_encrypted_password")) { update = true password := d.Get("password").(string) kmsPassword := d.Get("kms_encrypted_password").(string) if password == "" && kmsPassword == "" { return WrapError(Error("One of the 'password' and 'kms_encrypted_password' should be set.")) } if password != "" { modifyInstanceAttributeReq.NewPassword = password } else { kmsService := KmsService{meta.(*connectivity.AliyunClient)} decryptResp, err := kmsService.Decrypt(kmsPassword, d.Get("kms_encryption_context").(map[string]interface{})) if err != nil { return WrapError(err) } modifyInstanceAttributeReq.NewPassword = decryptResp } } if update { raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceAttribute(modifyInstanceAttributeReq) }) addDebug(modifyInstanceAttributeReq.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), modifyInstanceAttributeReq.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("instance_name") d.SetPartial("db_instance_name") d.SetPartial("instance_release_protection") d.SetPartial("kms_encrypted_password") d.SetPartial("kms_encryption_context") d.SetPartial("password") } update = false modifyDBInstanceConnectionStringReq := r_kvstore.CreateModifyDBInstanceConnectionStringRequest() modifyDBInstanceConnectionStringReq.DBInstanceId = d.Id() if d.HasChange("private_connection_prefix") { update = true modifyDBInstanceConnectionStringReq.NewConnectionString = d.Get("private_connection_prefix").(string) } if d.HasChange("private_connection_port") { update = true modifyDBInstanceConnectionStringReq.Port = d.Get("private_connection_port").(string) } modifyDBInstanceConnectionStringReq.IPType = "Private" if update { object, err := r_kvstoreService.DescribeKvstoreInstance(d.Id()) modifyDBInstanceConnectionStringReq.CurrentConnectionString = fmt.Sprint(object["ConnectionDomain"]) raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyDBInstanceConnectionString(modifyDBInstanceConnectionStringReq) }) addDebug(modifyDBInstanceConnectionStringReq.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), modifyDBInstanceConnectionStringReq.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 60*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("private_connection_prefix") d.SetPartial("private_connection_port") } update = false modifySecurityIpsReq := r_kvstore.CreateModifySecurityIpsRequest() modifySecurityIpsReq.InstanceId = d.Id() modifySecurityIpsReq.ModifyMode = "Cover" if d.HasChange("security_ips") { update = true } modifySecurityIpsReq.SecurityIps = convertListToCommaSeparate(d.Get("security_ips").(*schema.Set).List()) if d.HasChange("security_ip_group_attribute") { update = true } if v, ok := d.GetOk("security_ip_group_attribute"); ok { modifySecurityIpsReq.SecurityIpGroupAttribute = v.(string) } if d.HasChange("security_ip_group_name") { update = true } if v, ok := d.GetOk("security_ip_group_name"); ok { modifySecurityIpsReq.SecurityIpGroupName = v.(string) } if update { raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifySecurityIps(modifySecurityIpsReq) }) addDebug(modifySecurityIpsReq.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), modifySecurityIpsReq.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 1*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("security_ips") d.SetPartial("security_ip_group_attribute") d.SetPartial("security_ip_group_name") } update = false modifyInstanceSpecReq := map[string]interface{}{ "RegionId": client.RegionId, "AutoPay": true, "InstanceId": d.Id(), } if !d.IsNewResource() && d.HasChange("instance_class") { update = true } if v, ok := d.GetOk("instance_class"); ok { modifyInstanceSpecReq["InstanceClass"] = v } // read_only_count and slave_read_only_count may be changed after other attributes changed, like secondary_zone_id // and ReadOnlyCount and SlaveReadOnlyCount can not be changed together if !d.IsNewResource() && d.HasChange("read_only_count") { update = true if v, ok := d.GetOkExists("read_only_count"); ok { modifyInstanceSpecReq["ReadOnlyCount"] = v } } if v, ok := d.GetOk("effective_time"); ok { modifyInstanceSpecReq["EffectiveTime"] = v } if v, ok := d.GetOk("business_info"); ok { modifyInstanceSpecReq["BusinessInfo"] = v } if v, ok := d.GetOk("coupon_no"); ok { modifyInstanceSpecReq["CouponNo"] = v } if v, ok := d.GetOkExists("force_upgrade"); ok { modifyInstanceSpecReq["ForceUpgrade"] = v } if v, ok := d.GetOk("order_type"); ok { modifyInstanceSpecReq["OrderType"] = v } if update { action := "ModifyInstanceSpec" wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, modifyInstanceSpecReq, false) if err != nil { if IsExpectedErrors(err, []string{"MissingRedisUsedmemoryUnsupportPerfItem", "Task.Conflict"}) || NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, modifyInstanceSpecReq) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } instanceStatusConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 1*time.Minute, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "InstanceStatus")) if _, err := instanceStatusConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } stateConf := BuildStateConf([]string{}, []string{"true"}, d.Timeout(schema.TimeoutUpdate), 1*time.Minute, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "IsOrderCompleted")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } if fmt.Sprint(modifyInstanceSpecReq["EffectiveTime"]) != "MaintainTime" && d.HasChange("instance_class") { stateConf := BuildStateConf([]string{}, []string{modifyInstanceSpecReq["InstanceClass"].(string)}, d.Timeout(schema.TimeoutUpdate), 1*time.Minute, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "RealInstanceClass")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } } } // read_only_count and slave_read_only_count may be changed after other attributes changed, like secondary_zone_id // and ReadOnlyCount and SlaveReadOnlyCount can not be changed together if !d.IsNewResource() && d.HasChange("slave_read_only_count") { update = true if v, ok := d.GetOkExists("slave_read_only_count"); ok { modifyInstanceSpecReq["SlaveReadOnlyCount"] = v } delete(modifyInstanceSpecReq, "ReadOnlyCount") action := "ModifyInstanceSpec" wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, modifyInstanceSpecReq, false) if err != nil { if IsExpectedErrors(err, []string{"MissingRedisUsedmemoryUnsupportPerfItem", "Task.Conflict"}) || NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, modifyInstanceSpecReq) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } instanceStatusConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 1*time.Minute, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "InstanceStatus")) if _, err := instanceStatusConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } stateConf := BuildStateConf([]string{}, []string{"true"}, d.Timeout(schema.TimeoutUpdate), 1*time.Minute, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "IsOrderCompleted")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } } update = false modifyInstanceMajorVersionReq := r_kvstore.CreateModifyInstanceMajorVersionRequest() modifyInstanceMajorVersionReq.InstanceId = d.Id() if !d.IsNewResource() && d.HasChange("engine_version") { update = true } modifyInstanceMajorVersionReq.MajorVersion = d.Get("engine_version").(string) if v, ok := d.GetOk("effective_time"); ok { modifyInstanceMajorVersionReq.EffectiveTime = v.(string) } if update { raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceMajorVersion(modifyInstanceMajorVersionReq) }) addDebug(modifyInstanceMajorVersionReq.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), modifyInstanceMajorVersionReq.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 300*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("engine_version") } if d.HasChange("parameters") { request := r_kvstore.CreateModifyInstanceConfigRequest() request.InstanceId = d.Id() config := make(map[string]interface{}) documented := d.Get("parameters").(*schema.Set).List() if len(documented) > 0 { for _, i := range documented { key := i.(map[string]interface{})["name"].(string) value := i.(map[string]interface{})["value"] config[key] = value } cfg, _ := convertMaptoJsonString(config) request.Config = cfg raw, err := client.WithRKvstoreClient(func(r_kvstoreClient *r_kvstore.Client) (interface{}, error) { return r_kvstoreClient.ModifyInstanceConfig(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } } d.SetPartial("parameters") } if d.HasChange("enable_public") { prefix := d.Get("connection_string_prefix").(string) port := fmt.Sprintf("%s", d.Get("port")) target := d.Get("enable_public").(bool) if target { request := r_kvstore.CreateAllocateInstancePublicConnectionRequest() request.InstanceId = d.Id() request.ConnectionStringPrefix = prefix request.Port = port raw, err := client.WithRKvstoreClient(func(client *r_kvstore.Client) (interface{}, error) { return client.AllocateInstancePublicConnection(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } } if !target && d.Get("connection_string") != "" { request := r_kvstore.CreateReleaseInstancePublicConnectionRequest() request.InstanceId = d.Id() request.CurrentConnectionString = d.Get("connection_string").(string) raw, err := client.WithRKvstoreClient(func(client *r_kvstore.Client) (interface{}, error) { return client.ReleaseInstancePublicConnection(request) }) addDebug(request.GetActionName(), raw) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), request.GetActionName(), AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 120*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } } d.SetPartial("enable_public") } update = false modifyInstanceTDERequest := map[string]interface{}{ "InstanceId": d.Id(), } if d.HasChange("tde_status") { update = true if v, ok := d.GetOk("tde_status"); ok { modifyInstanceTDERequest["TDEStatus"] = v } } if d.HasChange("encryption_name") { update = true if v, ok := d.GetOk("encryption_name"); ok { modifyInstanceTDERequest["EncryptionName"] = v } } if d.HasChange("encryption_key") { update = true if v, ok := d.GetOk("encryption_key"); ok { modifyInstanceTDERequest["EncryptionKey"] = v } } if d.HasChange("role_arn") { update = true if v, ok := d.GetOk("role_arn"); ok { modifyInstanceTDERequest["RoleArn"] = v } } if update { action := "ModifyInstanceTDE" wait := incrementalWait(3*time.Second, 3*time.Second) err := resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, modifyInstanceTDERequest, false) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, request) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 5*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("tde_status") d.SetPartial("encryption_name") d.SetPartial("encryption_key") d.SetPartial("role_arn") } if !d.IsNewResource() && d.HasChange("shard_count") { if instanceClass, ok := d.GetOk("instance_class"); ok { if !isCloudDiskSpec(fmt.Sprint(instanceClass)) { return WrapErrorf(err, "The instance_class(%s) is not cloud disk specification, if you want to modify the shard_count, you can do this by modifying instance_class.", instanceClass) } } oldEntry, newEntry := d.GetChange("shard_count") oldEntryValue := oldEntry.(int) newEntryValue := newEntry.(int) removed := oldEntryValue - newEntryValue added := newEntryValue - oldEntryValue if removed > 0 { action := "DeleteShardingNode" request := make(map[string]interface{}) request["InstanceId"] = d.Id() request["ShardCount"] = removed wait := incrementalWait(3*time.Second, 5*time.Second) err = resource.Retry(d.Timeout(schema.TimeoutUpdate), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, request, false) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } addDebug(action, response, request) return nil }) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } instanceStatusConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "InstanceStatus")) if _, err := instanceStatusConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } stateConf := BuildStateConf([]string{}, []string{"true"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "IsOrderCompleted")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("shard_count") } if added > 0 { action := "AddShardingNode" request := make(map[string]interface{}) request["InstanceId"] = d.Id() request["ClientToken"] = buildClientToken(action) request["ShardCount"] = added wait := incrementalWait(3*time.Second, 5*time.Second) err = resource.Retry(d.Timeout(schema.TimeoutUpdate), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, request, true) request["ClientToken"] = buildClientToken(action) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, request) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } instanceStatusConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "InstanceStatus")) if _, err := instanceStatusConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } stateConf := BuildStateConf([]string{}, []string{"true"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "IsOrderCompleted")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("shard_count") } } update = false modifyDBInstanceAutoUpgradeReq := map[string]interface{}{ "DBInstanceId": d.Id(), } if d.HasChange("is_auto_upgrade_open") { update = true } if v, ok := d.GetOk("is_auto_upgrade_open"); ok { modifyDBInstanceAutoUpgradeReq["Value"] = v } if update { action := "ModifyDBInstanceAutoUpgrade" wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, modifyDBInstanceAutoUpgradeReq, false) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, modifyDBInstanceAutoUpgradeReq) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } d.SetPartial("is_auto_upgrade_open") } update = false modifyInstanceBandwidthReq := map[string]interface{}{ "InstanceId": d.Id(), } if d.HasChange("bandwidth") { update = true if v, ok := d.GetOkExists("bandwidth"); ok { modifyInstanceBandwidthReq["TargetIntranetBandwidth"] = v } } if update { action := "ModifyInstanceBandwidth" wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutUpdate)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, modifyInstanceBandwidthReq, false) if err != nil { if NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, modifyInstanceBandwidthReq) if err != nil { return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } instanceStatusConf := BuildStateConf([]string{}, []string{"Normal"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "InstanceStatus")) if _, err := instanceStatusConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } stateConf := BuildStateConf([]string{}, []string{"true"}, d.Timeout(schema.TimeoutUpdate), 10*time.Second, r_kvstoreService.KvstoreInstanceAttributeRefreshFunc(d.Id(), "IsOrderCompleted")) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } d.SetPartial("bandwidth") } d.Partial(false) return resourceAliCloudKvstoreInstanceRead(d, meta) } func resourceAliCloudKvstoreInstanceDelete(d *schema.ResourceData, meta interface{}) error { client := meta.(*connectivity.AliyunClient) r_kvstoreService := R_kvstoreService{client} instance, err := r_kvstoreService.DescribeKvstoreInstance(d.Id()) if err != nil { if NotFoundError(err) { return nil } return WrapError(err) } if fmt.Sprint(instance["ChargeType"]) == string(PrePaid) { log.Printf("[WARN] Cannot destroy Subscription resource: alicloud_kvstore_instance. Terraform will remove this resource from the state file, however resources may remain.") return nil } var response map[string]interface{} action := "DeleteInstance" request := map[string]interface{}{ "InstanceId": d.Id(), "RegionId": client.RegionId, } if v, ok := d.GetOk("global_instance_id"); ok { request["GlobalInstanceId"] = v.(string) } wait := incrementalWait(3*time.Second, 3*time.Second) err = resource.Retry(client.GetRetryTimeout(d.Timeout(schema.TimeoutDelete)), func() *resource.RetryError { response, err = client.RpcPost("R-kvstore", "2015-01-01", action, nil, request, true) if err != nil { if NoCodeRegexRetry(err) || NeedRetry(err) { wait() return resource.RetryableError(err) } return resource.NonRetryableError(err) } return nil }) addDebug(action, response, request) if err != nil { if IsExpectedErrors(err, []string{"InvalidInstanceId.NotFound"}) { return nil } return WrapErrorf(err, DefaultErrorMsg, d.Id(), action, AlibabaCloudSdkGoERROR) } stateConf := BuildStateConf([]string{}, []string{}, d.Timeout(schema.TimeoutDelete), 60*time.Second, r_kvstoreService.KvstoreInstanceStateRefreshFunc(d.Id(), []string{})) if _, err := stateConf.WaitForState(); err != nil { return WrapErrorf(err, IdMsg, d.Id()) } return nil } func refreshParameters(d *schema.ResourceData, meta interface{}) error { client := meta.(*connectivity.AliyunClient) r_kvstoreService := R_kvstoreService{client} var param []map[string]interface{} _, ok := d.GetOk("parameters") if !ok { d.Set("parameters", param) return nil } object, err := r_kvstoreService.DescribeKvstoreInstance(d.Id()) if err != nil { return WrapError(err) } m := make(map[string]interface{}) err = json.Unmarshal([]byte(fmt.Sprint(object["Config"])), &m) if err != nil { return WrapError(err) } for k, v := range m { parameter := map[string]interface{}{ "name": k, "value": v, } param = append(param, parameter) } d.Set("parameters", param) return nil } func convertModifyInstanceTDERequest(source interface{}) interface{} { switch source { case "enabled": return "Enabled" case "disabled": return "Disabled" } return source } func isCloudDiskSpec(instanceClass string) bool { if strings.HasSuffix(instanceClass, ".ce") || strings.HasSuffix(instanceClass, ".ee") { return true } return false }