package sqlserverflexalpha import ( "context" "fmt" "net/http" "github.com/hashicorp/terraform-plugin-framework/datasource" "github.com/hashicorp/terraform-plugin-framework/datasource/schema" "github.com/hashicorp/terraform-plugin-framework/diag" "github.com/hashicorp/terraform-plugin-framework/types" "github.com/hashicorp/terraform-plugin-log/tflog" "tf-provider.git.onstackit.cloud/stackit-dev-tools/terraform-provider-stackitprivatepreview/pkg_gen/sqlserverflexalpha" "tf-provider.git.onstackit.cloud/stackit-dev-tools/terraform-provider-stackitprivatepreview/stackit/internal/conversion" "tf-provider.git.onstackit.cloud/stackit-dev-tools/terraform-provider-stackitprivatepreview/stackit/internal/core" sqlserverflexalphaGen "tf-provider.git.onstackit.cloud/stackit-dev-tools/terraform-provider-stackitprivatepreview/stackit/internal/services/sqlserverflexalpha/database/datasources_gen" sqlserverflexUtils "tf-provider.git.onstackit.cloud/stackit-dev-tools/terraform-provider-stackitprivatepreview/stackit/internal/services/sqlserverflexalpha/utils" "tf-provider.git.onstackit.cloud/stackit-dev-tools/terraform-provider-stackitprivatepreview/stackit/internal/utils" ) // dataSourceModel maps the data source schema data. type dataSourceModel struct { sqlserverflexalphaGen.DatabaseModel TerraformID types.String `tfsdk:"id"` } var _ datasource.DataSource = (*databaseDataSource)(nil) // NewDatabaseDataSource creates a new database data source. func NewDatabaseDataSource() datasource.DataSource { return &databaseDataSource{} } type databaseDataSource struct { client *sqlserverflexalpha.APIClient providerData core.ProviderData } // Metadata returns the data source type name. func (d *databaseDataSource) Metadata( _ context.Context, req datasource.MetadataRequest, resp *datasource.MetadataResponse, ) { resp.TypeName = req.ProviderTypeName + "_sqlserverflexalpha_database" } // Schema defines the data source schema. func (d *databaseDataSource) Schema(ctx context.Context, _ datasource.SchemaRequest, resp *datasource.SchemaResponse) { s := sqlserverflexalphaGen.DatabaseDataSourceSchema(ctx) s.Attributes["id"] = schema.StringAttribute{ Description: "Terraform's internal resource ID. It is structured as \\\"`project_id`,`region`,`instance_id`," + "`database_id`\\\".\",", Computed: true, } resp.Schema = s } // Configure adds the provider configured client to the data source. func (d *databaseDataSource) Configure( ctx context.Context, req datasource.ConfigureRequest, resp *datasource.ConfigureResponse, ) { var ok bool d.providerData, ok = conversion.ParseProviderData(ctx, req.ProviderData, &resp.Diagnostics) if !ok { return } apiClient := sqlserverflexUtils.ConfigureClient(ctx, &d.providerData, &resp.Diagnostics) if resp.Diagnostics.HasError() { return } d.client = apiClient tflog.Info(ctx, "SQL SERVER Flex alpha database client configured") } // Read retrieves the resource's state from the API. func (d *databaseDataSource) Read(ctx context.Context, req datasource.ReadRequest, resp *datasource.ReadResponse) { var model dataSourceModel diags := req.Config.Get(ctx, &model) resp.Diagnostics.Append(diags...) if resp.Diagnostics.HasError() { return } ctx = core.InitProviderContext(ctx) // Extract identifiers from the plan projectId := model.ProjectId.ValueString() instanceId := model.InstanceId.ValueString() region := d.providerData.GetRegionWithOverride(model.Region) databaseName := model.DatabaseName.ValueString() ctx = tflog.SetField(ctx, "project_id", projectId) ctx = tflog.SetField(ctx, "instance_id", instanceId) ctx = tflog.SetField(ctx, "region", region) ctx = tflog.SetField(ctx, "database_name", databaseName) // Fetch database from the API databaseResp, err := d.client.GetDatabaseRequest(ctx, projectId, region, instanceId, databaseName).Execute() if resp.Diagnostics.HasError() { return } if err != nil { handleReadError(ctx, &resp.Diagnostics, err, projectId, instanceId) resp.State.RemoveResource(ctx) return } ctx = core.LogResponse(ctx) // Map response body to schema and populate Computed attribute values err = mapFields(databaseResp, &model, region) if err != nil { core.LogAndAddError( ctx, &resp.Diagnostics, "Error reading database", fmt.Sprintf("Processing API payload: %v", err), ) return } // Set refreshed state diags = resp.State.Set(ctx, model) resp.Diagnostics.Append(diags...) if resp.Diagnostics.HasError() { return } tflog.Info(ctx, "SQL Server Flex alpha database read") } // handleReadError centralizes API error handling for the Read operation. func handleReadError(ctx context.Context, diags *diag.Diagnostics, err error, projectId, instanceId string) { utils.LogError( ctx, diags, err, "Reading database", fmt.Sprintf( "Could not retrieve database for instance %q in project %q.", instanceId, projectId, ), map[int]string{ http.StatusBadRequest: fmt.Sprintf( "Invalid request parameters for project %q and instance %q.", projectId, instanceId, ), http.StatusNotFound: fmt.Sprintf( "Database, instance %q, or project %q not found.", instanceId, projectId, ), http.StatusForbidden: fmt.Sprintf("Forbidden access to project %q.", projectId), }, ) }