diff --git a/.github/workflows/mysql.yml b/.github/workflows/mysql8_0.yml similarity index 89% rename from .github/workflows/mysql.yml rename to .github/workflows/mysql8_0.yml index 604bb9c9c..d3b2f96bb 100644 --- a/.github/workflows/mysql.yml +++ b/.github/workflows/mysql8_0.yml @@ -1,4 +1,4 @@ -name: Test MySQL rs +name: Test MySQL8_0 rs on: workflow_dispatch: @@ -39,5 +39,5 @@ jobs: - name: Check out code into the Go module directory uses: actions/checkout@v2 - - name: Run TestAccResourceGridscaleMySQL_Basic - run: make testacc TEST=./gridscale TESTARGS='-run=TestAccResourceGridscaleMySQL_Basic' + - name: Run TestAccResourceGridscaleMySQL8_0_Basic + run: make testacc TEST=./gridscale TESTARGS='-run=TestAccResourceGridscaleMySQL8_0_Basic' diff --git a/CHANGELOG.md b/CHANGELOG.md index baa6dc4a8..63c521117 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,14 @@ # Changelog + +## 1.23.0 (Dec 12, 2023) + +FEATURES: +- Add `griscale_mysql8_0` resource [PR #246](https://github.com/gridscale/terraform-provider-gridscale/issues/246) + +IMPROVEMENTS: +- Add deprecation notice for `griscale_mysql` (MySQL 5.7) + ## 1.22.0 (Sept 26, 2023) FEATURES: diff --git a/README.md b/README.md index c82866f1d..2ec25c4bf 100644 --- a/README.md +++ b/README.md @@ -88,6 +88,45 @@ To run a specific acceptance test, use `TESTARGS`. TEST=./gridscale \ TESTARGS='-run=TestAccResourceGridscaleLoadBalancerBasic' +# Override local terraform provider for development + +Create `local-dev.tfrc` with this content and change YOUR_USERNAME: + +``` +provider_installation { + # Use /Users/YOUR_USERNAME/github/terraform-provider-gridscale as an overridden package directory + # for the hashicorp/null provider. This disables the version and checksum + # verifications for this provider and forces Terraform to look for the + # null provider plugin in the given directory. + dev_overrides { + "gridscale/gridscale" = "/Users/YOUR_USERNAME/github/terraform-provider-gridscale/" + } + # For all other providers, install them directly from their origin provider + # registries as normal. If you omit this, Terraform will _only_ use + # the dev_overrides block, and so no other providers will be available. + direct {} +} +``` + +`export TF_CLI_CONFIG_FILE=./local-dev.tfrc` + +Build the binary: `go build -o terraform-provider-gridscale` + +Now you can run `terraform plan/apply` and terraform will tell you that you are using a local version of that provider: + +``` +> terraform apply +╷ +│ Warning: Provider development overrides are in effect +│ +│ The following provider development overrides are set in the CLI configuration: +│ - gridscale/gridscale in /Users/YOUR_USERNAME/github/terraform-provider-gridscale +│ +│ The behavior may therefore not match any released version of the provider and applying changes may cause the state to become incompatible with published +│ releases. +╵ +``` + ## Releasing the Provider A [GoReleaser](https://goreleaser.com/) configuration is provided that produces build artifacts matching the [layout required](https://www.terraform.io/docs/registry/providers/publishing.html#manually-preparing-a-release) to publish the provider in the Terraform Registry. diff --git a/gridscale/provider.go b/gridscale/provider.go index 2eb4d8173..af1b48872 100644 --- a/gridscale/provider.go +++ b/gridscale/provider.go @@ -96,6 +96,7 @@ func Provider() *schema.Provider { "gridscale_postgresql": resourceGridscalePostgreSQL(), "gridscale_sqlserver": resourceGridscaleMSSQLServer(), "gridscale_mysql": resourceGridscaleMySQL(), + "gridscale_mysql8_0": resourceGridscaleMySQL8_0(), "gridscale_mariadb": resourceGridscaleMariaDB(), "gridscale_memcached": resourceGridscaleMemcached(), "gridscale_filesystem": resourceGridscaleFilesystem(), diff --git a/gridscale/resource_gridscale_mysql8_0.go b/gridscale/resource_gridscale_mysql8_0.go new file mode 100644 index 000000000..64a3d60ed --- /dev/null +++ b/gridscale/resource_gridscale_mysql8_0.go @@ -0,0 +1,519 @@ +package gridscale + +import ( + "context" + "errors" + "fmt" + "net/http" + "regexp" + "strings" + "time" + + "github.com/gridscale/gsclient-go/v3" + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/validation" + errHandler "github.com/terraform-providers/terraform-provider-gridscale/gridscale/error-handler" + + "log" +) + +const mysql8TemplateFlavourName = "mysql" + +func resourceGridscaleMySQL8_0() *schema.Resource { + return &schema.Resource{ + Create: resourceGridscaleMySQL8_0Create, + Read: resourceGridscaleMySQL8_0Read, + Delete: resourceGridscaleMySQL8_0Delete, + Update: resourceGridscaleMySQL8_0Update, + Importer: &schema.ResourceImporter{ + State: schema.ImportStatePassthrough, + }, + CustomizeDiff: func(ctx context.Context, d *schema.ResourceDiff, meta interface{}) error { + client := meta.(*gsclient.Client) + paasTemplates, err := client.GetPaaSTemplateList(ctx) + if err != nil { + return err + } + + releaseVal := d.Get("release").(string) + perfClassVal := d.Get("performance_class").(string) + var chosenTemplate gsclient.PaaSTemplate + var isReleasePerfClassValid bool + releaseWPerfClasess := make(map[string][]string) + for _, template := range paasTemplates { + if template.Properties.Flavour == mysql8TemplateFlavourName { + perfClasses := releaseWPerfClasess[template.Properties.Release] + releaseWPerfClasess[template.Properties.Release] = append(perfClasses, template.Properties.PerformanceClass) + if template.Properties.Release == releaseVal && template.Properties.PerformanceClass == perfClassVal { + isReleasePerfClassValid = true + chosenTemplate = template + } + } + } + if !isReleasePerfClassValid { + errMess := fmt.Sprintf("release %v with performance class %s is not a valid MySQL release/performance class. Valid releases with corresponding performance classes are:\n\t", releaseVal, perfClassVal) + for release, perfClasses := range releaseWPerfClasess { + errMess += fmt.Sprintf("release %s has following perfomance classes: %s\n\t", release, strings.Join(perfClasses, ", ")) + } + return errors.New(errMess) + } + return validateMySQL8_0Parameters(d, chosenTemplate) + }, + Schema: map[string]*schema.Schema{ + "name": { + Type: schema.TypeString, + Description: "The human-readable name of the object. It supports the full UTF-8 character set, with a maximum of 64 characters.", + Required: true, + ValidateFunc: validation.NoZeroValues, + }, + "release": { + Type: schema.TypeString, + Description: `The MySQL release of this instance.\n + For convenience, please use gscloud https://github.com/gridscale/gscloud to get the list of available MySQL service releases.`, + Required: true, + ValidateFunc: validation.NoZeroValues, + }, + "performance_class": { + Type: schema.TypeString, + Description: "Performance class of MySQL service.", + Required: true, + ValidateFunc: validation.NoZeroValues, + }, + "mysql_sql_mode": { + Type: schema.TypeString, + Description: "SQL Mode.", + Optional: true, + Default: "ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION", + }, + "mysql_max_connections": { + Type: schema.TypeInt, + Description: "Max Connections.", + Optional: true, + Default: 4000, + }, + "mysql_default_time_zone": { + Type: schema.TypeString, + Description: "Server Timezone.", + Optional: true, + Default: "UTC", + }, + "mysql_max_allowed_packet": { + Type: schema.TypeString, + Description: "Max Allowed Packet Size. Format: xM (where x is an integer, M stands for unit: k(kB), M(MB), G(GB)).", + Optional: true, + Default: "64M", + }, + "username": { + Type: schema.TypeString, + Description: "Username for MySQL service. It is used to connect to the MySQL instance.", + Computed: true, + Sensitive: true, + }, + "password": { + Type: schema.TypeString, + Description: "Password for MySQL service. It is used to connect to the MySQL instance.", + Computed: true, + Sensitive: true, + }, + "listen_port": { + Type: schema.TypeSet, + Description: "The port numbers where this MySQL service accepts connections.", + Computed: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "name": { + Type: schema.TypeString, + Computed: true, + }, + "host": { + Type: schema.TypeString, + Computed: true, + }, + "port": { + Type: schema.TypeInt, + Computed: true, + }, + }, + }, + }, + "security_zone_uuid": { + Type: schema.TypeString, + Description: "Security zone UUID linked to MySQL service.", + Deprecated: "Security zone is deprecated for gridSQL, gridStore, and gridFs. Please consider to use private network instead.", + Optional: true, + ForceNew: true, + Computed: true, + }, + "network_uuid": { + Type: schema.TypeString, + Description: "The UUID of the network that the service is attached to.", + Optional: true, + Computed: true, + }, + "service_template_uuid": { + Type: schema.TypeString, + Description: "PaaS service template that MySQL service uses.", + Computed: true, + }, + "service_template_category": { + Type: schema.TypeString, + Computed: true, + Description: "The template service's category used to create the service.", + }, + "usage_in_minutes": { + Type: schema.TypeInt, + Description: "Number of minutes that MySQL service is in use.", + Computed: true, + }, + "change_time": { + Type: schema.TypeString, + Description: "Time of the last change.", + Computed: true, + }, + "create_time": { + Type: schema.TypeString, + Description: "Date time this service has been created.", + Computed: true, + }, + "status": { + Type: schema.TypeString, + Description: "Current status of MySQL service.", + Computed: true, + }, + "max_core_count": { + Type: schema.TypeInt, + Description: "Maximum CPU core count. The MySQL instance's CPU core count will be autoscaled based on the workload. The number of cores stays between 1 and `max_core_count`.", + Optional: true, + Computed: true, + ValidateFunc: validation.NoZeroValues, + }, + "labels": { + Type: schema.TypeSet, + Description: "List of labels.", + Optional: true, + Elem: &schema.Schema{Type: schema.TypeString}, + }, + }, + Timeouts: &schema.ResourceTimeout{ + Create: schema.DefaultTimeout(15 * time.Minute), + Update: schema.DefaultTimeout(15 * time.Minute), + Delete: schema.DefaultTimeout(15 * time.Minute), + }, + } +} + +func resourceGridscaleMySQL8_0Read(d *schema.ResourceData, meta interface{}) error { + client := meta.(*gsclient.Client) + errorPrefix := fmt.Sprintf("read mysql (%s) resource -", d.Id()) + paas, err := client.GetPaaSService(context.Background(), d.Id()) + if err != nil { + if requestError, ok := err.(gsclient.RequestError); ok { + if requestError.StatusCode == 404 { + d.SetId("") + return nil + } + } + return fmt.Errorf("%s error: %v", errorPrefix, err) + } + props := paas.Properties + creds := props.Credentials + if err = d.Set("name", props.Name); err != nil { + return fmt.Errorf("%s error setting name: %v", errorPrefix, err) + } + if len(creds) > 0 { + if err = d.Set("username", creds[0].Username); err != nil { + return fmt.Errorf("%s error setting username: %v", errorPrefix, err) + } + if err = d.Set("password", creds[0].Password); err != nil { + return fmt.Errorf("%s error setting password: %v", errorPrefix, err) + } + } + if err = d.Set("network_uuid", props.NetworkUUID); err != nil { + return fmt.Errorf("%s error setting network_uuid: %v", errorPrefix, err) + } + if err = d.Set("security_zone_uuid", props.SecurityZoneUUID); err != nil { + return fmt.Errorf("%s error setting security_zone_uuid: %v", errorPrefix, err) + } + if err = d.Set("service_template_category", props.ServiceTemplateCategory); err != nil { + return fmt.Errorf("%s error setting service_template_category: %v", errorPrefix, err) + } + if err = d.Set("service_template_uuid", props.ServiceTemplateUUID); err != nil { + return fmt.Errorf("%s error setting service_template_uuid: %v", errorPrefix, err) + } + if err = d.Set("usage_in_minutes", props.UsageInMinutes); err != nil { + return fmt.Errorf("%s error setting usage_in_minutes: %v", errorPrefix, err) + } + if err = d.Set("change_time", props.ChangeTime.String()); err != nil { + return fmt.Errorf("%s error setting change_time: %v", errorPrefix, err) + } + if err = d.Set("create_time", props.CreateTime.String()); err != nil { + return fmt.Errorf("%s error setting create_time: %v", errorPrefix, err) + } + if err = d.Set("status", props.Status); err != nil { + return fmt.Errorf("%s error setting status: %v", errorPrefix, err) + } + + // Set MySQL parameters + if err = d.Set("mysql_sql_mode", props.Parameters["mysql_sql_mode"]); err != nil { + return fmt.Errorf("%s error setting mysql_sql_mode: %v", errorPrefix, err) + } + if err = d.Set("mysql_max_connections", props.Parameters["mysql_max_connections"]); err != nil { + return fmt.Errorf("%s error setting mysql_max_connections: %v", errorPrefix, err) + } + if err = d.Set("mysql_default_time_zone", props.Parameters["mysql_default_time_zone"]); err != nil { + return fmt.Errorf("%s error setting mysql_default_time_zone: %v", errorPrefix, err) + } + if err = d.Set("mysql_max_allowed_packet", props.Parameters["mysql_max_allowed_packet"]); err != nil { + return fmt.Errorf("%s error setting mysql_max_allowed_packet: %v", errorPrefix, err) + } + //Get listen ports + listenPorts := make([]interface{}, 0) + for host, value := range props.ListenPorts { + for k, portValue := range value { + port := map[string]interface{}{ + "name": k, + "host": host, + "port": portValue, + } + listenPorts = append(listenPorts, port) + } + } + if err = d.Set("listen_port", listenPorts); err != nil { + return fmt.Errorf("%s error setting listen ports: %v", errorPrefix, err) + } + + //Get core count's limit + for _, value := range props.ResourceLimits { + if value.Resource == "cores" { + if err = d.Set("max_core_count", value.Limit); err != nil { + return fmt.Errorf("%s error setting max_core_count: %v", errorPrefix, err) + } + } + } + + //Set labels + if err = d.Set("labels", props.Labels); err != nil { + return fmt.Errorf("%s error setting labels: %v", errorPrefix, err) + } + + // Look for security zone's network that the PaaS service is connected to + // (if the paas is connected to security zone. O.w skip) + if props.SecurityZoneUUID == "" { + return nil + } + networks, err := client.GetNetworkList(context.Background()) + if err != nil { + return fmt.Errorf("%s error getting networks: %v", errorPrefix, err) + } + //look for a network that the MySQL service is in + for _, network := range networks { + securityZones := network.Properties.Relations.PaaSSecurityZones + //Each network can contain only one security zone + if len(securityZones) >= 1 { + if securityZones[0].ObjectUUID == props.SecurityZoneUUID { + if err = d.Set("network_uuid", network.Properties.ObjectUUID); err != nil { + return fmt.Errorf("%s error setting network_uuid: %v", errorPrefix, err) + } + } + } + } + return nil +} + +func resourceGridscaleMySQL8_0Create(d *schema.ResourceData, meta interface{}) error { + client := meta.(*gsclient.Client) + errorPrefix := fmt.Sprintf("create mysql (%s) resource -", d.Id()) + + // Get mysql template UUID + release := d.Get("release").(string) + performanceClass := d.Get("performance_class").(string) + templateUUID, err := getMySQL8_0TemplateUUID(client, release, performanceClass) + if err != nil { + return fmt.Errorf("%s error: %v", errorPrefix, err) + } + + requestBody := gsclient.PaaSServiceCreateRequest{ + Name: d.Get("name").(string), + PaaSServiceTemplateUUID: templateUUID, + Labels: convSOStrings(d.Get("labels").(*schema.Set).List()), + } + networkUUIDInf, isNetworkSet := d.GetOk("network_uuid") + if isNetworkSet { + requestBody.NetworkUUID = networkUUIDInf.(string) + } + // If network_uuid is set, skip setting security_zone_uuid. + if secZoneUUIDInf, ok := d.GetOk("security_zone_uuid"); ok && !isNetworkSet { + requestBody.PaaSSecurityZoneUUID = secZoneUUIDInf.(string) + } + if val, ok := d.GetOk("max_core_count"); ok { + limits := []gsclient.ResourceLimit{ + { + Resource: "cores", + Limit: val.(int), + }, + } + requestBody.ResourceLimits = limits + } + + params := make(map[string]interface{}) + params["mysql_sql_mode"] = d.Get("mysql_sql_mode") + params["mysql_max_connections"] = d.Get("mysql_max_connections") + params["mysql_default_time_zone"] = d.Get("mysql_default_time_zone") + params["mysql_max_allowed_packet"] = d.Get("mysql_max_allowed_packet") + requestBody.Parameters = params + + ctx, cancel := context.WithTimeout(context.Background(), d.Timeout(schema.TimeoutCreate)) + defer cancel() + response, err := client.CreatePaaSService(ctx, requestBody) + if err != nil { + return err + } + d.SetId(response.ObjectUUID) + log.Printf("The id for MySQL service %s has been set to %v", requestBody.Name, response.ObjectUUID) + return resourceGridscaleMySQL8_0Read(d, meta) +} + +func resourceGridscaleMySQL8_0Update(d *schema.ResourceData, meta interface{}) error { + client := meta.(*gsclient.Client) + errorPrefix := fmt.Sprintf("update mysql (%s) resource -", d.Id()) + + labels := convSOStrings(d.Get("labels").(*schema.Set).List()) + requestBody := gsclient.PaaSServiceUpdateRequest{ + Name: d.Get("name").(string), + Labels: &labels, + } + if d.HasChange("network_uuid") { + requestBody.NetworkUUID = d.Get("network_uuid").(string) + } + // Only update templateUUID, when `release` or `performance_class` is changed + if d.HasChange("release") || d.HasChange("performance_class") { + // Get mysql template UUID + release := d.Get("release").(string) + performanceClass := d.Get("performance_class").(string) + templateUUID, err := getMySQL8_0TemplateUUID(client, release, performanceClass) + if err != nil { + return fmt.Errorf("%s error: %v", errorPrefix, err) + } + requestBody.PaaSServiceTemplateUUID = templateUUID + } + + if val, ok := d.GetOk("max_core_count"); ok { + limits := []gsclient.ResourceLimit{ + { + Resource: "cores", + Limit: val.(int), + }, + } + requestBody.ResourceLimits = limits + } + + params := make(map[string]interface{}) + params["mysql_sql_mode"] = d.Get("mysql_sql_mode") + params["mysql_max_connections"] = d.Get("mysql_max_connections") + params["mysql_default_time_zone"] = d.Get("mysql_default_time_zone") + params["mysql_max_allowed_packet"] = d.Get("mysql_max_allowed_packet") + requestBody.Parameters = params + + ctx, cancel := context.WithTimeout(context.Background(), d.Timeout(schema.TimeoutUpdate)) + defer cancel() + err := client.UpdatePaaSService(ctx, d.Id(), requestBody) + if err != nil { + return fmt.Errorf("%s error: %v", errorPrefix, err) + } + return resourceGridscaleMySQL8_0Read(d, meta) +} + +func resourceGridscaleMySQL8_0Delete(d *schema.ResourceData, meta interface{}) error { + client := meta.(*gsclient.Client) + errorPrefix := fmt.Sprintf("delete mysql (%s) resource -", d.Id()) + + ctx, cancel := context.WithTimeout(context.Background(), d.Timeout(schema.TimeoutDelete)) + defer cancel() + err := errHandler.SuppressHTTPErrorCodes( + client.DeletePaaSService(ctx, d.Id()), + http.StatusNotFound, + ) + if err != nil { + return fmt.Errorf("%s error: %v", errorPrefix, err) + } + return nil +} + +// getMySQL8_0TemplateUUID returns the UUID of the mysql service template. +func getMySQL8_0TemplateUUID(client *gsclient.Client, release, performanceClass string) (string, error) { + paasTemplates, err := client.GetPaaSTemplateList(context.Background()) + if err != nil { + return "", err + } + var isReleaseValid bool + var releases []string + var uTemplate gsclient.PaaSTemplate + for _, template := range paasTemplates { + if template.Properties.Flavour == mysql8TemplateFlavourName { + releases = append(releases, template.Properties.Release) + if template.Properties.Release == release && template.Properties.PerformanceClass == performanceClass { + isReleaseValid = true + uTemplate = template + } + } + } + if !isReleaseValid { + return "", fmt.Errorf("%v is not a valid MySQL release. Valid releases are: %v\n", release, strings.Join(releases, ", ")) + } + + return uTemplate.Properties.ObjectUUID, nil +} + +func validateMySQL8_0Parameters(d *schema.ResourceDiff, template gsclient.PaaSTemplate) error { + var errorMessages []string + if sqlMode, ok := d.GetOk("mysql_sql_mode"); ok { + if scheme, ok := template.Properties.ParametersSchema["mysql_sql_mode"]; ok { + validMode := regexp.MustCompile(scheme.Regex) + if !validMode.MatchString(sqlMode.(string)) { + errorMessages = append(errorMessages, fmt.Sprintf("Invalid 'mysql_sql_mode' value. Example value: '%s'\n", scheme.Default)) + } + } + } + if maxNConn, ok := d.GetOk("mysql_max_connections"); ok { + if scheme, ok := template.Properties.ParametersSchema["mysql_max_connections"]; ok { + if scheme.Min > maxNConn.(int) || maxNConn.(int) > scheme.Max { + errorMessages = append(errorMessages, fmt.Sprintf("Invalid 'mysql_max_connections' value. Value must stays between %d and %d\n", scheme.Min, scheme.Max)) + } + } + } + if defaultTimeZone, ok := d.GetOk("mysql_default_time_zone"); ok { + if scheme, ok := template.Properties.ParametersSchema["mysql_default_time_zone"]; ok { + var isValidFormat bool + for _, allowedValue := range scheme.Allowed { + if defaultTimeZone.(string) == allowedValue { + isValidFormat = true + } + } + if !isValidFormat { + errorMessages = append(errorMessages, + fmt.Sprintf("Invalid 'mysql_default_time_zone' value. Value must be one of these:\n\t%s", + strings.Join(scheme.Allowed, "\n\t"), + ), + ) + } + } + } + if maxAllowedPacket, ok := d.GetOk("mysql_max_allowed_packet"); ok { + if scheme, ok := template.Properties.ParametersSchema["mysql_max_allowed_packet"]; ok { + validMode := regexp.MustCompile(scheme.Regex) + if !validMode.MatchString(maxAllowedPacket.(string)) { + errorMessages = append(errorMessages, fmt.Sprintf("Invalid 'mysql_max_allowed_packet' value. Example value: '%s'\n", scheme.Default)) + } + } + } + if maxNCore, ok := d.GetOk("max_core_count"); ok { + autoscalingNCore := template.Properties.Autoscaling.Cores + if autoscalingNCore.Min > maxNCore.(int) || maxNCore.(int) > autoscalingNCore.Max { + errorMessages = append(errorMessages, fmt.Sprintf("Invalid 'max_core_count' value. Value must stays between %d and %d\n", autoscalingNCore.Min, autoscalingNCore.Max)) + } + } + if len(errorMessages) != 0 { + return errors.New(strings.Join(errorMessages, "")) + } + return nil +} diff --git a/gridscale/resource_gridscale_mysql8_0_test.go b/gridscale/resource_gridscale_mysql8_0_test.go new file mode 100644 index 000000000..8f57f4e03 --- /dev/null +++ b/gridscale/resource_gridscale_mysql8_0_test.go @@ -0,0 +1,63 @@ +package gridscale + +import ( + "fmt" + + "github.com/gridscale/gsclient-go/v3" + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/acctest" + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/resource" + + "testing" +) + +func TestAccResourceGridscaleMySQL8_0_Basic(t *testing.T) { + var object gsclient.PaaSService + name := fmt.Sprintf("MySQL-%s", acctest.RandString(10)) + + resource.Test(t, resource.TestCase{ + PreCheck: func() { testAccPreCheck(t) }, + Providers: testAccProviders, + CheckDestroy: testAccCheckResourceGridscalePaaSDestroyCheck, + Steps: []resource.TestStep{ + { + Config: testAccCheckResourceGridscaleMySQL8_0Config_basic(name), + Check: resource.ComposeTestCheckFunc( + testAccCheckResourceGridscalePaaSExists("gridscale_mysql8_0.test", &object), + resource.TestCheckResourceAttr( + "gridscale_mysql8_0.test", "name", name), + ), + }, + { + Config: testAccCheckResourceGridscaleMySQL8_0Config_basic_update(), + Check: resource.ComposeTestCheckFunc( + testAccCheckResourceGridscalePaaSExists("gridscale_mysql8_0.test", &object), + resource.TestCheckResourceAttr( + "gridscale_mysql8_0.test", "name", "newname"), + ), + }, + }, + }) +} + +func testAccCheckResourceGridscaleMySQL8_0Config_basic(name string) string { + return fmt.Sprintf(` +resource "gridscale_mysql8_0" "test" { + name = "%s" + release = "8.0" + performance_class = "standard" +} +`, name) +} + +func testAccCheckResourceGridscaleMySQL8_0Config_basic_update() string { + return fmt.Sprintf(` +resource "gridscale_mysql8_0" "test" { + name = "newname" + release = "8.0" + performance_class = "standard" + max_core_count = 20 + mysql_max_connections = 2000 + labels = ["test"] +} +`) +} diff --git a/website/docs/r/mysql.html.md b/website/docs/r/mysql.html.md index a7cd8ba2b..689618955 100644 --- a/website/docs/r/mysql.html.md +++ b/website/docs/r/mysql.html.md @@ -8,6 +8,8 @@ description: |- # gridscale_mysql +*DEPRECATED* - We keep this for existing customers who still use MySQL 5.7. Please migrate to MySQL 8.0. + Provides a MySQL resource. This can be used to create, modify, and delete MySQL instances. ## Example diff --git a/website/docs/r/mysql8_0.html.md b/website/docs/r/mysql8_0.html.md new file mode 100644 index 000000000..7bf537690 --- /dev/null +++ b/website/docs/r/mysql8_0.html.md @@ -0,0 +1,89 @@ +--- +layout: "gridscale" +page_title: "gridscale: gridscale_mysql8_0" +sidebar_current: "docs-gridscale-resource-mysql8_0" +description: |- + Manage a MySQL 8.0 service in gridscale. +--- + +# gridscale_mysql8_0 + + +Provides a MySQL 8.0 resource. This can be used to create, modify, and delete MySQL 8.0 instances. + +## Example + +The following example shows how one might use this resource to add a MySQL 8.0 service to gridscale: + +```terraform +resource "gridscale_mysql8_0" "terra-mysql-test" { + name = "my mysql" + release = "8.0" + performance_class = "insane" + max_core_count = 20 + mysql_default_time_zone = "Europe/Berlin" +} +``` + +## Argument Reference + +The following arguments are supported: + +* `name` - (Required) The human-readable name of the object. It supports the full UTF-8 character set, with a maximum of 64 characters. + +* `release` - (Required) The mysql release of this instance. For convenience, please use [gscloud](https://github.com/gridscale/gscloud) to get the list of available mysql service releases. + +* `performance_class` - (Required) Performance class of mysql service. Available performance classes at the time of writing: `standard`, `high`, `insane`, `ultra`. + +* `mysql_sql_mode` - (Optional) mysql parameter: SQL Mode. Default: "ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION". + +* `mysql_max_connections` - (Optional) mysql parameter: Max Connections. Default: 4000. + +* `mysql_default_time_zone` - (Optional) mysql parameter: Server Timezone. Default: UTC. + +* `mysql_max_allowed_packet` - (Optional) mysql parameter: Max Allowed Packet Size. Format: xM (where x is an integer, M stands for unit: k(kB), M(MB), G(GB)). Default: 64M. + +* `labels` - (Optional) List of labels in the format [ "label1", "label2" ]. + +* `network_uuid` - (Optional) The UUID of the network that the service is attached to. + +* `security_zone_uuid` - *DEPRECATED* (Optional, Forcenew) The UUID of the security zone that the service is attached to. + +* `max_core_count` - (Optional) Maximum CPU core count. The mysql instance's CPU core count will be autoscaled based on the workload. The number of cores stays between 1 and `max_core_count`. + +## Timeouts + +Timeouts configuration options (in seconds): +More info: [terraform.io/docs/configuration/resources.html#operation-timeouts](https://www.terraform.io/docs/configuration/resources.html#operation-timeouts) + +* `create` - (Default value is "15m" - 15 minutes) Used for creating a resource. +* `update` - (Default value is "15m" - 15 minutes) Used for updating a resource. +* `delete` - (Default value is "15m" - 15 minutes) Used for deleting a resource. + +## Attributes + +This resource exports the following attributes: + +* `name` - See Argument Reference above. +* `release` - See Argument Reference above. +* `performance_class` - See Argument Reference above. +* `mysql_sql_mode` - See Argument Reference above. +* `mysql_max_connections` - See Argument Reference above. +* `mysql_default_time_zone` - See Argument Reference above. +* `mysql_max_allowed_packet` - See Argument Reference above. +* `username` - Username for PaaS service. It is used to connect to the mysql instance. +* `password` - Password for PaaS service. It is used to connect to the mysql instance. +* `listen_port` - The port numbers where this mysql service accepts connections. + * `name` - Name of a port. + * `host` - Host address. + * `listen_port` - Port number. +* `security_zone_uuid` - See Argument Reference above. +* `network_uuid` - The UUID of the network that the service is attached to or network UUID containing security zone. +* `service_template_uuid` - PaaS service template that mysql service uses. +* `service_template_category` - The template service's category used to create the service. +* `usage_in_minutes` - Number of minutes that PaaS service is in use. +* `change_time` - Time of the last change. +* `create_time` - Date time this service has been created. +* `status` - Current status of PaaS service. +* `max_core_count` - See Argument Reference above. +* `labels` - See Argument Reference above. diff --git a/website/gridscale.erb b/website/gridscale.erb index 841c31ef5..cb73ea882 100644 --- a/website/gridscale.erb +++ b/website/gridscale.erb @@ -118,6 +118,9 @@