diff --git a/.changes/unreleased/added-20241223-094847.yaml b/.changes/unreleased/added-20241223-094847.yaml
new file mode 100644
index 00000000..3e7c6a5c
--- /dev/null
+++ b/.changes/unreleased/added-20241223-094847.yaml
@@ -0,0 +1,7 @@
+kind: added
+body: |
+ Added support for multi-format Notebook Resource/Data-Source.
+ By using `format` attribute, you can now define the format of the Notebook Resource/Data-Source. Accepted values are `jpynb`, and `py`.
+time: 2024-12-23T09:48:47.1324573-08:00
+custom:
+ Issue: "168"
diff --git a/.changes/unreleased/breaking-20241223-094847.yaml b/.changes/unreleased/breaking-20241223-094847.yaml
new file mode 100644
index 00000000..050da5f0
--- /dev/null
+++ b/.changes/unreleased/breaking-20241223-094847.yaml
@@ -0,0 +1,7 @@
+kind: breaking
+body: |
+ The `format` attribute is now REQUIRED for Resources/Data-Sources with definition support.
+ Currently applicable to the following Resources/Data-Sources: Report, Notebook, Semantic Model, and Spark Job Definition.
+time: 2024-12-23T09:48:47.1324573-08:00
+custom:
+ Issue: "111"
diff --git a/docs/data-sources/notebook.md b/docs/data-sources/notebook.md
index bcbc6358..5d17e2d3 100644
--- a/docs/data-sources/notebook.md
+++ b/docs/data-sources/notebook.md
@@ -36,6 +36,7 @@ data "fabric_notebook" "example_by_id" {
data "fabric_notebook" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "ipynb"
output_definition = true
}
@@ -68,6 +69,7 @@ output "example_definition_content_object" {
### Optional
- `display_name` (String) The Notebook display name.
+- `format` (String) The Notebook format. Possible values: `ipynb`, `py`. Default: `ipynb`
- `id` (String) The Notebook ID.
- `output_definition` (Boolean) Output definition parts as gzip base64 content? Default: `false`
@@ -77,9 +79,8 @@ output "example_definition_content_object" {
### Read-Only
-- `definition` (Attributes Map) Definition parts. Possible path keys: `notebook-content.ipynb`. (see [below for nested schema](#nestedatt--definition))
+- `definition` (Attributes Map) Definition parts. Possible path keys: `notebook-content.ipynb`, `notebook-content.py`. (see [below for nested schema](#nestedatt--definition))
- `description` (String) The Notebook description.
-- `format` (String) The Notebook format. Possible values: `ipynb`.
diff --git a/docs/data-sources/report.md b/docs/data-sources/report.md
index d1a463b1..38fd00b9 100644
--- a/docs/data-sources/report.md
+++ b/docs/data-sources/report.md
@@ -29,6 +29,7 @@ data "fabric_report" "example" {
data "fabric_report" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "PBIR-Legacy"
output_definition = true
}
@@ -56,6 +57,7 @@ output "example_definition_report_object" {
### Optional
+- `format` (String) The Report format. Possible values: `PBIR-Legacy`. Default: `PBIR-Legacy`
- `output_definition` (Boolean) Output definition parts as gzip base64 content? Default: `false`
!> Your terraform state file may grow a lot if you output definition content. Only use it when you must use data from the definition.
@@ -67,7 +69,6 @@ output "example_definition_report_object" {
- `definition` (Attributes Map) Definition parts. Possible path keys: `report.json`, `definition.pbir`, `StaticResources/RegisteredResources/*`, `StaticResources/SharedResources/*`. (see [below for nested schema](#nestedatt--definition))
- `description` (String) The Report description.
- `display_name` (String) The Report display name.
-- `format` (String) The Report format. Possible values: `PBIR-Legacy`.
diff --git a/docs/data-sources/semantic_model.md b/docs/data-sources/semantic_model.md
index ddd44af7..caa01a27 100644
--- a/docs/data-sources/semantic_model.md
+++ b/docs/data-sources/semantic_model.md
@@ -29,6 +29,7 @@ data "fabric_semantic_model" "example" {
data "fabric_semantic_model" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "TMSL"
output_definition = true
}
@@ -53,6 +54,7 @@ output "example_definition_bim_object" {
### Optional
+- `format` (String) The Semantic Model format. Possible values: `TMSL`. Default: `TMSL`
- `output_definition` (Boolean) Output definition parts as gzip base64 content? Default: `false`
!> Your terraform state file may grow a lot if you output definition content. Only use it when you must use data from the definition.
@@ -64,7 +66,6 @@ output "example_definition_bim_object" {
- `definition` (Attributes Map) Definition parts. Possible path keys: `model.bim`, `definition.pbism`, `diagramLayout.json`. (see [below for nested schema](#nestedatt--definition))
- `description` (String) The Semantic Model description.
- `display_name` (String) The Semantic Model display name.
-- `format` (String) The Semantic Model format. Possible values: `TMSL`.
diff --git a/docs/data-sources/spark_job_definition.md b/docs/data-sources/spark_job_definition.md
index 935ea756..ec5524b5 100644
--- a/docs/data-sources/spark_job_definition.md
+++ b/docs/data-sources/spark_job_definition.md
@@ -36,6 +36,7 @@ data "fabric_spark_job_definition" "example_by_name" {
data "fabric_spark_job_definition" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "SparkJobDefinitionV1"
output_definition = true
}
@@ -68,6 +69,7 @@ output "example_definition_content_object" {
### Optional
- `display_name` (String) The Spark Job Definition display name.
+- `format` (String) The Spark Job Definition format. Possible values: `SparkJobDefinitionV1`. Default: `SparkJobDefinitionV1`
- `id` (String) The Spark Job Definition ID.
- `output_definition` (Boolean) Output definition parts as gzip base64 content? Default: `false`
@@ -79,7 +81,6 @@ output "example_definition_content_object" {
- `definition` (Attributes Map) Definition parts. Possible path keys: `SparkJobDefinitionV1.json`. (see [below for nested schema](#nestedatt--definition))
- `description` (String) The Spark Job Definition description.
-- `format` (String) The Spark Job Definition format. Possible values: `SparkJobDefinitionV1`.
- `properties` (Attributes) The Spark Job Definition properties. (see [below for nested schema](#nestedatt--properties))
diff --git a/docs/resources/notebook.md b/docs/resources/notebook.md
index ae607466..b831cd3b 100644
--- a/docs/resources/notebook.md
+++ b/docs/resources/notebook.md
@@ -31,6 +31,7 @@ resource "fabric_notebook" "example_definition_bootstrap" {
description = "example with definition bootstrapping"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "ipynb"
definition = {
"notebook-content.ipynb" = {
source = "${local.path}/notebook.ipynb.tmpl"
@@ -43,6 +44,7 @@ resource "fabric_notebook" "example_definition_update" {
display_name = "example"
description = "example with definition update when source or tokens changed"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "ipynb"
definition = {
"notebook-content.ipynb" = {
source = "${local.path}/notebook.ipynb.tmpl"
@@ -65,14 +67,14 @@ resource "fabric_notebook" "example_definition_update" {
### Optional
-- `definition` (Attributes Map) Definition parts. Accepted path keys: `notebook-content.ipynb`. Read more about [Notebook definition part paths](https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/notebook-definition). (see [below for nested schema](#nestedatt--definition))
+- `definition` (Attributes Map) Definition parts. Accepted path keys: `notebook-content.ipynb`, `notebook-content.py`. Read more about [Notebook definition part paths](https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/notebook-definition). (see [below for nested schema](#nestedatt--definition))
- `definition_update_enabled` (Boolean) Update definition on change of source content. Default: `true`.
- `description` (String) The Notebook description.
+- `format` (String) The Notebook format. Possible values: `ipynb`, `py`
- `timeouts` (Attributes) (see [below for nested schema](#nestedatt--timeouts))
### Read-Only
-- `format` (String) The Notebook format. Possible values: `ipynb`.
- `id` (String) The Notebook ID.
diff --git a/docs/resources/report.md b/docs/resources/report.md
index 84123323..b3aa1ea1 100644
--- a/docs/resources/report.md
+++ b/docs/resources/report.md
@@ -24,6 +24,7 @@ resource "fabric_report" "example_bootstrap" {
display_name = "example"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "PBIR-Legacy"
definition = {
"report.json" = {
source = "${local.path}/report.json"
@@ -44,6 +45,7 @@ resource "fabric_report" "example_bootstrap" {
resource "fabric_report" "example_update" {
display_name = "example with update"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "PBIR-Legacy"
definition = {
"report.json" = {
source = "${local.path}/report.json"
@@ -68,6 +70,7 @@ resource "fabric_report" "example_update" {
- `definition` (Attributes Map) Definition parts. Accepted path keys: `report.json`, `definition.pbir`, `StaticResources/RegisteredResources/*`, `StaticResources/SharedResources/*`. Read more about [Report definition part paths](https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/report-definition). (see [below for nested schema](#nestedatt--definition))
- `display_name` (String) The Report display name.
+- `format` (String) The Report format. Possible values: `PBIR-Legacy`
- `workspace_id` (String) The Workspace ID.
### Optional
@@ -78,7 +81,6 @@ resource "fabric_report" "example_update" {
### Read-Only
-- `format` (String) The Report format. Possible values: `PBIR-Legacy`.
- `id` (String) The Report ID.
diff --git a/docs/resources/semantic_model.md b/docs/resources/semantic_model.md
index f3aa4e58..6394f199 100644
--- a/docs/resources/semantic_model.md
+++ b/docs/resources/semantic_model.md
@@ -24,6 +24,7 @@ resource "fabric_semantic_model" "example_bootstrap" {
display_name = "example"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "TMSL"
definition = {
"model.bim" = {
source = "${local.path}/model.bim.tmpl"
@@ -38,6 +39,7 @@ resource "fabric_semantic_model" "example_bootstrap" {
resource "fabric_semantic_model" "example_update" {
display_name = "example with update"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "TMSL"
definition = {
"model.bim" = {
source = "${local.path}/model.bim.tmpl"
@@ -59,6 +61,7 @@ resource "fabric_semantic_model" "example_update" {
- `definition` (Attributes Map) Definition parts. Accepted path keys: `model.bim`, `definition.pbism`, `diagramLayout.json`. Read more about [Semantic Model definition part paths](https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/semantic-model-definition). (see [below for nested schema](#nestedatt--definition))
- `display_name` (String) The Semantic Model display name.
+- `format` (String) The Semantic Model format. Possible values: `TMSL`
- `workspace_id` (String) The Workspace ID.
### Optional
@@ -69,7 +72,6 @@ resource "fabric_semantic_model" "example_update" {
### Read-Only
-- `format` (String) The Semantic Model format. Possible values: `TMSL`.
- `id` (String) The Semantic Model ID.
diff --git a/docs/resources/spark_job_definition.md b/docs/resources/spark_job_definition.md
index c08ab096..b38ac416 100644
--- a/docs/resources/spark_job_definition.md
+++ b/docs/resources/spark_job_definition.md
@@ -31,6 +31,7 @@ resource "fabric_spark_job_definition" "example_definition_bootstrap" {
description = "example with definition bootstrapping"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "SparkJobDefinitionV1"
definition = {
"SparkJobDefinitionV1.json" = {
source = "${local.path}/SparkJobDefinitionV1.json.tmpl"
@@ -43,6 +44,7 @@ resource "fabric_spark_job_definition" "example_definition_update" {
display_name = "example3"
description = "example with definition update when source or tokens changed"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "SparkJobDefinitionV1"
definition = {
"SparkJobDefinitionV1.json" = {
source = "${local.path}/SparkJobDefinitionV1.json.tmpl"
@@ -68,11 +70,11 @@ resource "fabric_spark_job_definition" "example_definition_update" {
- `definition` (Attributes Map) Definition parts. Accepted path keys: `SparkJobDefinitionV1.json`. Read more about [Spark Job Definition definition part paths](https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/spark-job-definition). (see [below for nested schema](#nestedatt--definition))
- `definition_update_enabled` (Boolean) Update definition on change of source content. Default: `true`.
- `description` (String) The Spark Job Definition description.
+- `format` (String) The Spark Job Definition format. Possible values: `SparkJobDefinitionV1`
- `timeouts` (Attributes) (see [below for nested schema](#nestedatt--timeouts))
### Read-Only
-- `format` (String) The Spark Job Definition format. Possible values: `SparkJobDefinitionV1`.
- `id` (String) The Spark Job Definition ID.
- `properties` (Attributes) The Spark Job Definition properties. (see [below for nested schema](#nestedatt--properties))
diff --git a/examples/data-sources/fabric_notebook/data-source.tf b/examples/data-sources/fabric_notebook/data-source.tf
index f3faf46b..e7210117 100644
--- a/examples/data-sources/fabric_notebook/data-source.tf
+++ b/examples/data-sources/fabric_notebook/data-source.tf
@@ -15,6 +15,7 @@ data "fabric_notebook" "example_by_id" {
data "fabric_notebook" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "ipynb"
output_definition = true
}
diff --git a/examples/data-sources/fabric_report/data-source.tf b/examples/data-sources/fabric_report/data-source.tf
index 677ba505..6fa4b5e6 100644
--- a/examples/data-sources/fabric_report/data-source.tf
+++ b/examples/data-sources/fabric_report/data-source.tf
@@ -8,6 +8,7 @@ data "fabric_report" "example" {
data "fabric_report" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "PBIR-Legacy"
output_definition = true
}
diff --git a/examples/data-sources/fabric_semantic_model/data-source.tf b/examples/data-sources/fabric_semantic_model/data-source.tf
index bc1d2b3a..a965feed 100644
--- a/examples/data-sources/fabric_semantic_model/data-source.tf
+++ b/examples/data-sources/fabric_semantic_model/data-source.tf
@@ -8,6 +8,7 @@ data "fabric_semantic_model" "example" {
data "fabric_semantic_model" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "TMSL"
output_definition = true
}
diff --git a/examples/data-sources/fabric_spark_job_definition/data-source.tf b/examples/data-sources/fabric_spark_job_definition/data-source.tf
index 0540e7a1..a3aaa5f6 100644
--- a/examples/data-sources/fabric_spark_job_definition/data-source.tf
+++ b/examples/data-sources/fabric_spark_job_definition/data-source.tf
@@ -15,6 +15,7 @@ data "fabric_spark_job_definition" "example_by_name" {
data "fabric_spark_job_definition" "example_definition" {
id = "11111111-1111-1111-1111-111111111111"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "SparkJobDefinitionV1"
output_definition = true
}
diff --git a/examples/resources/fabric_notebook/resource.tf b/examples/resources/fabric_notebook/resource.tf
index 2f6e58dd..80f6fc2f 100644
--- a/examples/resources/fabric_notebook/resource.tf
+++ b/examples/resources/fabric_notebook/resource.tf
@@ -10,6 +10,7 @@ resource "fabric_notebook" "example_definition_bootstrap" {
description = "example with definition bootstrapping"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "ipynb"
definition = {
"notebook-content.ipynb" = {
source = "${local.path}/notebook.ipynb.tmpl"
@@ -22,6 +23,7 @@ resource "fabric_notebook" "example_definition_update" {
display_name = "example"
description = "example with definition update when source or tokens changed"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "ipynb"
definition = {
"notebook-content.ipynb" = {
source = "${local.path}/notebook.ipynb.tmpl"
diff --git a/examples/resources/fabric_report/resource.tf b/examples/resources/fabric_report/resource.tf
index 846b03f5..4faf413e 100644
--- a/examples/resources/fabric_report/resource.tf
+++ b/examples/resources/fabric_report/resource.tf
@@ -3,6 +3,7 @@ resource "fabric_report" "example_bootstrap" {
display_name = "example"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "PBIR-Legacy"
definition = {
"report.json" = {
source = "${local.path}/report.json"
@@ -23,6 +24,7 @@ resource "fabric_report" "example_bootstrap" {
resource "fabric_report" "example_update" {
display_name = "example with update"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "PBIR-Legacy"
definition = {
"report.json" = {
source = "${local.path}/report.json"
diff --git a/examples/resources/fabric_semantic_model/resource.tf b/examples/resources/fabric_semantic_model/resource.tf
index 8d536ef3..abe5e017 100644
--- a/examples/resources/fabric_semantic_model/resource.tf
+++ b/examples/resources/fabric_semantic_model/resource.tf
@@ -3,6 +3,7 @@ resource "fabric_semantic_model" "example_bootstrap" {
display_name = "example"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "TMSL"
definition = {
"model.bim" = {
source = "${local.path}/model.bim.tmpl"
@@ -17,6 +18,7 @@ resource "fabric_semantic_model" "example_bootstrap" {
resource "fabric_semantic_model" "example_update" {
display_name = "example with update"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "TMSL"
definition = {
"model.bim" = {
source = "${local.path}/model.bim.tmpl"
diff --git a/examples/resources/fabric_spark_job_definition/resource.tf b/examples/resources/fabric_spark_job_definition/resource.tf
index 619d28b3..eace3561 100644
--- a/examples/resources/fabric_spark_job_definition/resource.tf
+++ b/examples/resources/fabric_spark_job_definition/resource.tf
@@ -10,6 +10,7 @@ resource "fabric_spark_job_definition" "example_definition_bootstrap" {
description = "example with definition bootstrapping"
workspace_id = "00000000-0000-0000-0000-000000000000"
definition_update_enabled = false
+ format = "SparkJobDefinitionV1"
definition = {
"SparkJobDefinitionV1.json" = {
source = "${local.path}/SparkJobDefinitionV1.json.tmpl"
@@ -22,6 +23,7 @@ resource "fabric_spark_job_definition" "example_definition_update" {
display_name = "example3"
description = "example with definition update when source or tokens changed"
workspace_id = "00000000-0000-0000-0000-000000000000"
+ format = "SparkJobDefinitionV1"
definition = {
"SparkJobDefinitionV1.json" = {
source = "${local.path}/SparkJobDefinitionV1.json.tmpl"
diff --git a/internal/pkg/fabricitem/data_item_definition.go b/internal/pkg/fabricitem/data_item_definition.go
index ceb63cce..44221c39 100644
--- a/internal/pkg/fabricitem/data_item_definition.go
+++ b/internal/pkg/fabricitem/data_item_definition.go
@@ -36,8 +36,7 @@ type DataSourceFabricItemDefinition struct {
MarkdownDescription string
IsDisplayNameUnique bool
FormatTypeDefault string
- FormatTypes []string
- DefinitionPathKeys []string
+ DefinitionFormats []DefinitionFormat
}
func NewDataSourceFabricItemDefinition(config DataSourceFabricItemDefinition) datasource.DataSource {
@@ -120,10 +119,12 @@ func (d *DataSourceFabricItemDefinition) Read(ctx context.Context, req datasourc
return
}
- data.Format = types.StringNull()
-
- if d.FormatTypeDefault != "" {
- data.Format = types.StringValue(d.FormatTypeDefault)
+ if data.Format.IsNull() || data.Format.IsUnknown() {
+ if d.FormatTypeDefault != "" {
+ data.Format = types.StringValue(d.FormatTypeDefault)
+ } else {
+ data.Format = types.StringNull()
+ }
}
resp.Diagnostics.Append(resp.State.Set(ctx, data)...)
@@ -203,7 +204,11 @@ func (d *DataSourceFabricItemDefinition) getDefinition(ctx context.Context, mode
respGetOpts := &fabcore.ItemsClientBeginGetItemDefinitionOptions{}
if !model.Format.IsNull() {
- respGetOpts.Format = model.Format.ValueStringPointer()
+ apiFormat := GetDefinitionFormatAPI(d.DefinitionFormats, model.Format.ValueString())
+
+ if apiFormat != "" {
+ respGetOpts.Format = azto.Ptr(apiFormat)
+ }
}
respGet, err := d.client.GetItemDefinition(ctx, model.WorkspaceID.ValueString(), model.ID.ValueString(), respGetOpts)
diff --git a/internal/pkg/fabricitem/data_schema.go b/internal/pkg/fabricitem/data_schema.go
index 7cdbb7b1..8e30a92a 100644
--- a/internal/pkg/fabricitem/data_schema.go
+++ b/internal/pkg/fabricitem/data_schema.go
@@ -8,8 +8,14 @@ import (
"fmt"
supertypes "github.com/FrangipaneTeam/terraform-plugin-framework-supertypes"
+ superstringvalidator "github.com/FrangipaneTeam/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework-timeouts/datasource/timeouts"
+ "github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
+ "github.com/hashicorp/terraform-plugin-framework/attr"
"github.com/hashicorp/terraform-plugin-framework/datasource/schema"
+ "github.com/hashicorp/terraform-plugin-framework/path"
+ "github.com/hashicorp/terraform-plugin-framework/schema/validator"
+ "github.com/hashicorp/terraform-plugin-framework/types"
"github.com/microsoft/terraform-provider-fabric/internal/framework/customtypes"
"github.com/microsoft/terraform-provider-fabric/internal/pkg/utils"
@@ -27,7 +33,7 @@ func GetDataSourceFabricItemSchema(ctx context.Context, d DataSourceFabricItem)
func GetDataSourceFabricItemDefinitionSchema(ctx context.Context, d DataSourceFabricItemDefinition) schema.Schema {
attributes := getDataSourceFabricItemBaseAttributes(ctx, d.Name, d.IsDisplayNameUnique)
- for key, value := range getDataSourceFabricItemDefinitionAttributes(ctx, d.Name, d.FormatTypes, d.DefinitionPathKeys) {
+ for key, value := range getDataSourceFabricItemDefinitionAttributes(ctx, d.Name, d.FormatTypeDefault, d.DefinitionFormats) {
attributes[key] = value
}
@@ -51,7 +57,7 @@ func GetDataSourceFabricItemDefinitionPropertiesSchema(ctx context.Context, d Da
attributes := getDataSourceFabricItemBaseAttributes(ctx, d.Name, d.IsDisplayNameUnique)
attributes["properties"] = properties
- for key, value := range getDataSourceFabricItemDefinitionAttributes(ctx, d.Name, d.FormatTypes, d.DefinitionPathKeys) {
+ for key, value := range getDataSourceFabricItemDefinitionAttributes(ctx, d.Name, d.FormatTypeDefault, d.DefinitionFormats) {
attributes[key] = value
}
@@ -65,7 +71,7 @@ func GetDataSourceFabricItemDefinitionPropertiesSchema1[Ttfprop, Titemprop any](
attributes := getDataSourceFabricItemBaseAttributes(ctx, d.Name, d.IsDisplayNameUnique)
attributes["properties"] = d.PropertiesSchema
- for key, value := range getDataSourceFabricItemDefinitionAttributes(ctx, d.Name, d.FormatTypes, d.DefinitionPathKeys) {
+ for key, value := range getDataSourceFabricItemDefinitionAttributes(ctx, d.Name, d.FormatTypeDefault, d.DefinitionFormats) {
attributes[key] = value
}
@@ -118,13 +124,21 @@ func getDataSourceFabricItemBaseAttributes(ctx context.Context, itemName string,
}
// Helper function to get Fabric Item data-source definition attributes.
-func getDataSourceFabricItemDefinitionAttributes(ctx context.Context, name string, formatTypes, definitionPathKeys []string) map[string]schema.Attribute {
+func getDataSourceFabricItemDefinitionAttributes(ctx context.Context, name, formatTypeDefault string, definitionFormatTypes []DefinitionFormat) map[string]schema.Attribute {
attributes := make(map[string]schema.Attribute)
- if len(formatTypes) > 0 {
+ formatTypes := GetDefinitionFormats(definitionFormatTypes)
+ definitionPathKeys := GetDefinitionFormatsPaths(definitionFormatTypes)
+
+ if len(formatTypes) > 1 || (len(formatTypes) == 1 && formatTypes[0] != "") {
attributes["format"] = schema.StringAttribute{
- MarkdownDescription: fmt.Sprintf("The %s format. Possible values: %s.", name, utils.ConvertStringSlicesToString(formatTypes, true, false)),
+ MarkdownDescription: fmt.Sprintf("The %s format. Possible values: %s. Default: `%s`", name, utils.ConvertStringSlicesToString(formatTypes, true, false), formatTypeDefault),
+ Optional: true,
Computed: true,
+ Validators: []validator.String{
+ stringvalidator.OneOf(formatTypes...),
+ superstringvalidator.RequireIfAttributeIsOneOf(path.MatchRoot("output_definition"), []attr.Value{types.BoolValue(true)}),
+ },
}
} else {
attributes["format"] = schema.StringAttribute{
diff --git a/internal/pkg/fabricitem/definition.go b/internal/pkg/fabricitem/definition.go
new file mode 100644
index 00000000..b16a1bc0
--- /dev/null
+++ b/internal/pkg/fabricitem/definition.go
@@ -0,0 +1,83 @@
+// Copyright (c) Microsoft Corporation
+// SPDX-License-Identifier: MPL-2.0
+
+package fabricitem
+
+import (
+ superstringvalidator "github.com/FrangipaneTeam/terraform-plugin-framework-validators/stringvalidator"
+ "github.com/hashicorp/terraform-plugin-framework/attr"
+ "github.com/hashicorp/terraform-plugin-framework/path"
+ "github.com/hashicorp/terraform-plugin-framework/schema/validator"
+ "github.com/hashicorp/terraform-plugin-framework/types"
+)
+
+type DefinitionFormat struct {
+ Type string
+ API string
+ Paths []string
+}
+
+func GetDefinitionFormats(values []DefinitionFormat) []string {
+ results := make([]string, len(values))
+
+ for i, value := range values {
+ results[i] = value.Type
+ }
+
+ return results
+}
+
+func GetDefinitionFormatsPaths(values []DefinitionFormat) []string {
+ var results []string
+
+ for _, value := range values {
+ results = append(results, value.Paths...)
+ }
+
+ return results
+}
+
+func GetDefinitionFormatPaths(values []DefinitionFormat, format string) []string {
+ for _, value := range values {
+ if value.Type == format {
+ return value.Paths
+ }
+ }
+
+ return nil
+}
+
+func GetDefinitionFormatAPI(values []DefinitionFormat, format string) string {
+ for _, value := range values {
+ if value.Type == format {
+ return value.API
+ }
+ }
+
+ return ""
+}
+
+func DefinitionPathKeysValidator(values []DefinitionFormat) []validator.String {
+ results := make([]validator.String, 0, len(values))
+
+ for _, value := range values {
+ paths := []superstringvalidator.OneOfWithDescriptionIfAttributeIsOneOfValues{}
+
+ for _, p := range value.Paths {
+ paths = append(paths, superstringvalidator.OneOfWithDescriptionIfAttributeIsOneOfValues{
+ Value: p,
+ Description: p,
+ })
+ }
+
+ stringValidator := superstringvalidator.OneOfWithDescriptionIfAttributeIsOneOf(
+ path.MatchRoot("format"),
+ []attr.Value{types.StringValue(value.Type)},
+ paths...,
+ )
+
+ results = append(results, stringValidator)
+ }
+
+ return results
+}
diff --git a/internal/pkg/fabricitem/models_resource_item_definition.go b/internal/pkg/fabricitem/models_resource_item_definition.go
index 2cfc7427..f72a744b 100644
--- a/internal/pkg/fabricitem/models_resource_item_definition.go
+++ b/internal/pkg/fabricitem/models_resource_item_definition.go
@@ -35,9 +35,13 @@ type fabricItemDefinition struct {
fabcore.ItemDefinition
}
-func (to *fabricItemDefinition) set(ctx context.Context, from resourceFabricItemDefinitionModel, update bool, definitionEmpty string, definitionPaths []string) diag.Diagnostics { //revive:disable-line:flag-parameter
+func (to *fabricItemDefinition) set(ctx context.Context, from resourceFabricItemDefinitionModel, update bool, definitionEmpty string, definitionPaths []string, definitionFormats []DefinitionFormat) diag.Diagnostics { //revive:disable-line:flag-parameter
if from.Format.ValueString() != DefinitionFormatNotApplicable {
- to.Format = from.Format.ValueStringPointer()
+ apiFormat := GetDefinitionFormatAPI(definitionFormats, from.Format.ValueString())
+
+ if apiFormat != "" {
+ to.Format = azto.Ptr(apiFormat)
+ }
}
to.Parts = []fabcore.ItemDefinitionPart{}
@@ -90,7 +94,7 @@ type requestCreateFabricItemDefinition struct {
fabcore.CreateItemRequest
}
-func (to *requestCreateFabricItemDefinition) set(ctx context.Context, from resourceFabricItemDefinitionModel, itemType fabcore.ItemType) diag.Diagnostics {
+func (to *requestCreateFabricItemDefinition) set(ctx context.Context, from resourceFabricItemDefinitionModel, itemType fabcore.ItemType, definitionFormats []DefinitionFormat) diag.Diagnostics {
to.DisplayName = from.DisplayName.ValueStringPointer()
to.Description = from.Description.ValueStringPointer()
to.Type = azto.Ptr(itemType)
@@ -98,7 +102,7 @@ func (to *requestCreateFabricItemDefinition) set(ctx context.Context, from resou
if !from.Definition.IsNull() && !from.Definition.IsUnknown() {
var def fabricItemDefinition
- if diags := def.set(ctx, from, false, "", []string{}); diags.HasError() {
+ if diags := def.set(ctx, from, false, "", []string{}, definitionFormats); diags.HasError() {
return diags
}
@@ -121,10 +125,10 @@ type requestUpdateFabricItemDefinitionDefinition struct {
fabcore.UpdateItemDefinitionRequest
}
-func (to *requestUpdateFabricItemDefinitionDefinition) set(ctx context.Context, from resourceFabricItemDefinitionModel, definitionEmpty string, definitionPaths []string) diag.Diagnostics {
+func (to *requestUpdateFabricItemDefinitionDefinition) set(ctx context.Context, from resourceFabricItemDefinitionModel, definitionEmpty string, definitionPaths []string, definitionFormats []DefinitionFormat) diag.Diagnostics {
var def fabricItemDefinition
- if diags := def.set(ctx, from, true, definitionEmpty, definitionPaths); diags.HasError() {
+ if diags := def.set(ctx, from, true, definitionEmpty, definitionPaths, definitionFormats); diags.HasError() {
return diags
}
diff --git a/internal/pkg/fabricitem/models_resource_item_definition_properties.go b/internal/pkg/fabricitem/models_resource_item_definition_properties.go
index 37abc206..622a5209 100644
--- a/internal/pkg/fabricitem/models_resource_item_definition_properties.go
+++ b/internal/pkg/fabricitem/models_resource_item_definition_properties.go
@@ -39,9 +39,13 @@ type FabricItemDefinitionProperties[Ttfprop, Titemprop any] struct { //revive:di
fabcore.ItemDefinition
}
-func (to *FabricItemDefinitionProperties[Ttfprop, Titemprop]) set(ctx context.Context, from ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], update bool, definitionEmpty string, definitionPaths []string) diag.Diagnostics { //revive:disable-line:flag-parameter,confusing-naming
+func (to *FabricItemDefinitionProperties[Ttfprop, Titemprop]) set(ctx context.Context, from ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], update bool, definitionEmpty string, definitionPaths []string, definitionFormats []DefinitionFormat) diag.Diagnostics { //revive:disable-line:flag-parameter,confusing-naming
if from.Format.ValueString() != DefinitionFormatNotApplicable {
- to.Format = from.Format.ValueStringPointer()
+ apiFormat := GetDefinitionFormatAPI(definitionFormats, from.Format.ValueString())
+
+ if apiFormat != "" {
+ to.Format = azto.Ptr(apiFormat)
+ }
}
to.Parts = []fabcore.ItemDefinitionPart{}
@@ -94,7 +98,7 @@ type requestCreateFabricItemDefinitionProperties[Ttfprop, Titemprop any] struct
fabcore.CreateItemRequest
}
-func (to *requestCreateFabricItemDefinitionProperties[Ttfprop, Titemprop]) set(ctx context.Context, from ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], itemType fabcore.ItemType) diag.Diagnostics { //revive:disable-line:confusing-naming
+func (to *requestCreateFabricItemDefinitionProperties[Ttfprop, Titemprop]) set(ctx context.Context, from ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], itemType fabcore.ItemType, definitionFormats []DefinitionFormat) diag.Diagnostics { //revive:disable-line:confusing-naming
to.DisplayName = from.DisplayName.ValueStringPointer()
to.Description = from.Description.ValueStringPointer()
to.Type = azto.Ptr(itemType)
@@ -102,7 +106,7 @@ func (to *requestCreateFabricItemDefinitionProperties[Ttfprop, Titemprop]) set(c
if !from.Definition.IsNull() && !from.Definition.IsUnknown() {
var def FabricItemDefinitionProperties[Ttfprop, Titemprop]
- if diags := def.set(ctx, from, false, "", []string{}); diags.HasError() {
+ if diags := def.set(ctx, from, false, "", []string{}, definitionFormats); diags.HasError() {
return diags
}
@@ -125,10 +129,10 @@ type requestUpdateFabricItemDefinitionPropertiesDefinition[Ttfprop, Titemprop an
fabcore.UpdateItemDefinitionRequest
}
-func (to *requestUpdateFabricItemDefinitionPropertiesDefinition[Ttfprop, Titemprop]) set(ctx context.Context, from ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], definitionEmpty string, definitionPaths []string) diag.Diagnostics { //revive:disable-line:confusing-naming
+func (to *requestUpdateFabricItemDefinitionPropertiesDefinition[Ttfprop, Titemprop]) set(ctx context.Context, from ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], definitionEmpty string, definitionPaths []string, definitionFormats []DefinitionFormat) diag.Diagnostics { //revive:disable-line:confusing-naming
var def FabricItemDefinitionProperties[Ttfprop, Titemprop]
- if diags := def.set(ctx, from, true, definitionEmpty, definitionPaths); diags.HasError() {
+ if diags := def.set(ctx, from, true, definitionEmpty, definitionPaths, definitionFormats); diags.HasError() {
return diags
}
diff --git a/internal/pkg/fabricitem/resource_item_definition.go b/internal/pkg/fabricitem/resource_item_definition.go
index 97c4672e..f867e238 100644
--- a/internal/pkg/fabricitem/resource_item_definition.go
+++ b/internal/pkg/fabricitem/resource_item_definition.go
@@ -43,12 +43,11 @@ type ResourceFabricItemDefinition struct {
DisplayNameMaxLength int
DescriptionMaxLength int
FormatTypeDefault string
- FormatTypes []string
DefinitionPathDocsURL string
- DefinitionPathKeys []string
DefinitionPathKeysValidator []validator.Map
DefinitionRequired bool
DefinitionEmpty string
+ DefinitionFormats []DefinitionFormat
}
func NewResourceFabricItemDefinition(config ResourceFabricItemDefinition) resource.Resource {
@@ -147,7 +146,7 @@ func (r *ResourceFabricItemDefinition) Create(ctx context.Context, req resource.
var reqCreate requestCreateFabricItemDefinition
- if resp.Diagnostics.Append(reqCreate.set(ctx, plan, r.Type)...); resp.Diagnostics.HasError() {
+ if resp.Diagnostics.Append(reqCreate.set(ctx, plan, r.Type, r.DefinitionFormats)...); resp.Diagnostics.HasError() {
return
}
@@ -411,7 +410,9 @@ func (r *ResourceFabricItemDefinition) checkUpdateItem(plan, state resourceFabri
func (r *ResourceFabricItemDefinition) checkUpdateDefinition(ctx context.Context, plan, state resourceFabricItemDefinitionModel, reqUpdate *requestUpdateFabricItemDefinitionDefinition) (bool, diag.Diagnostics) {
if !plan.Definition.Equal(state.Definition) && plan.DefinitionUpdateEnabled.ValueBool() {
- if diags := reqUpdate.set(ctx, plan, r.DefinitionEmpty, r.DefinitionPathKeys); diags.HasError() {
+ definitionPathKeys := GetDefinitionFormatsPaths(r.DefinitionFormats)
+
+ if diags := reqUpdate.set(ctx, plan, r.DefinitionEmpty, definitionPathKeys, r.DefinitionFormats); diags.HasError() {
return false, diags
}
diff --git a/internal/pkg/fabricitem/resource_item_definition_properties.go b/internal/pkg/fabricitem/resource_item_definition_properties.go
index affc75b0..a97a5d1d 100644
--- a/internal/pkg/fabricitem/resource_item_definition_properties.go
+++ b/internal/pkg/fabricitem/resource_item_definition_properties.go
@@ -136,7 +136,7 @@ func (r *ResourceFabricItemDefinitionProperties[Ttfprop, Titemprop]) Create(ctx
var reqCreate requestCreateFabricItemDefinitionProperties[Ttfprop, Titemprop]
- if resp.Diagnostics.Append(reqCreate.set(ctx, plan, r.Type)...); resp.Diagnostics.HasError() {
+ if resp.Diagnostics.Append(reqCreate.set(ctx, plan, r.Type, r.DefinitionFormats)...); resp.Diagnostics.HasError() {
return
}
@@ -414,7 +414,9 @@ func (r *ResourceFabricItemDefinitionProperties[Ttfprop, Titemprop]) checkUpdate
func (r *ResourceFabricItemDefinitionProperties[Ttfprop, Titemprop]) checkUpdateDefinition(ctx context.Context, plan, state ResourceFabricItemDefinitionPropertiesModel[Ttfprop, Titemprop], reqUpdate *requestUpdateFabricItemDefinitionPropertiesDefinition[Ttfprop, Titemprop]) (bool, diag.Diagnostics) {
if !plan.Definition.Equal(state.Definition) && plan.DefinitionUpdateEnabled.ValueBool() {
- if diags := reqUpdate.set(ctx, plan, r.DefinitionEmpty, r.DefinitionPathKeys); diags.HasError() {
+ definitionPathKeys := GetDefinitionFormatsPaths(r.DefinitionFormats)
+
+ if diags := reqUpdate.set(ctx, plan, r.DefinitionEmpty, definitionPathKeys, r.DefinitionFormats); diags.HasError() {
return false, diags
}
diff --git a/internal/pkg/fabricitem/resource_schema.go b/internal/pkg/fabricitem/resource_schema.go
index 8d867243..9b8a953e 100644
--- a/internal/pkg/fabricitem/resource_schema.go
+++ b/internal/pkg/fabricitem/resource_schema.go
@@ -9,6 +9,7 @@ import (
"regexp"
supertypes "github.com/FrangipaneTeam/terraform-plugin-framework-supertypes"
+ superstringvalidator "github.com/FrangipaneTeam/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework-timeouts/resource/timeouts"
"github.com/hashicorp/terraform-plugin-framework-validators/mapvalidator"
"github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
@@ -37,7 +38,7 @@ func GetResourceFabricItemSchema(ctx context.Context, r ResourceFabricItem) sche
func GetResourceFabricItemDefinitionSchema(ctx context.Context, r ResourceFabricItemDefinition) schema.Schema {
attributes := getResourceFabricItemBaseAttributes(ctx, r.Name, r.DisplayNameMaxLength, r.DescriptionMaxLength, r.NameRenameAllowed)
- for key, value := range getResourceFabricItemDefinitionAttributes(ctx, r.Name, r.FormatTypeDefault, r.FormatTypes, r.DefinitionPathDocsURL, r.DefinitionPathKeys, r.DefinitionPathKeysValidator, r.DefinitionRequired) {
+ for key, value := range getResourceFabricItemDefinitionAttributes(ctx, r.Name, r.DefinitionPathDocsURL, r.DefinitionFormats, r.DefinitionPathKeysValidator, r.DefinitionRequired) {
attributes[key] = value
}
@@ -61,7 +62,7 @@ func GetResourceFabricItemDefinitionPropertiesSchema(ctx context.Context, r Reso
attributes := getResourceFabricItemBaseAttributes(ctx, r.Name, r.DisplayNameMaxLength, r.DescriptionMaxLength, r.NameRenameAllowed)
attributes["properties"] = properties
- for key, value := range getResourceFabricItemDefinitionAttributes(ctx, r.Name, r.FormatTypeDefault, r.FormatTypes, r.DefinitionPathDocsURL, r.DefinitionPathKeys, r.DefinitionPathKeysValidator, r.DefinitionRequired) {
+ for key, value := range getResourceFabricItemDefinitionAttributes(ctx, r.Name, r.DefinitionPathDocsURL, r.DefinitionFormats, r.DefinitionPathKeysValidator, r.DefinitionRequired) {
attributes[key] = value
}
@@ -75,7 +76,7 @@ func GetResourceFabricItemDefinitionPropertiesSchema1[Ttfprop, Titemprop any](ct
attributes := getResourceFabricItemBaseAttributes(ctx, r.Name, r.DisplayNameMaxLength, r.DescriptionMaxLength, r.NameRenameAllowed)
attributes["properties"] = r.PropertiesSchema
- for key, value := range getResourceFabricItemDefinitionAttributes(ctx, r.Name, r.FormatTypeDefault, r.FormatTypes, r.DefinitionPathDocsURL, r.DefinitionPathKeys, r.DefinitionPathKeysValidator, r.DefinitionRequired) {
+ for key, value := range getResourceFabricItemDefinitionAttributes(ctx, r.Name, r.DefinitionPathDocsURL, r.DefinitionFormats, r.DefinitionPathKeysValidator, r.DefinitionRequired) {
attributes[key] = value
}
@@ -147,7 +148,7 @@ func getResourceFabricItemBaseAttributes(ctx context.Context, name string, displ
}
// Helper function to get Fabric Item definition attributes.
-func getResourceFabricItemDefinitionAttributes(ctx context.Context, name, formatTypeDefault string, formatTypes []string, definitionPathDocsURL string, definitionPathKeys []string, definitionPathKeysValidator []validator.Map, definitionRequired bool) map[string]schema.Attribute { //revive:disable-line:flag-parameter
+func getResourceFabricItemDefinitionAttributes(ctx context.Context, name, definitionPathDocsURL string, definitionFormatTypes []DefinitionFormat, definitionPathKeysValidator []validator.Map, definitionRequired bool) map[string]schema.Attribute { //revive:disable-line:flag-parameter
attributes := make(map[string]schema.Attribute)
attributes["definition_update_enabled"] = schema.BoolAttribute{
@@ -157,38 +158,48 @@ func getResourceFabricItemDefinitionAttributes(ctx context.Context, name, format
Default: booldefault.StaticBool(true),
}
- if len(formatTypes) > 0 {
- attributes["format"] = schema.StringAttribute{
- MarkdownDescription: fmt.Sprintf("The %s format. Possible values: %s.", name, utils.ConvertStringSlicesToString(formatTypes, true, false)),
- Computed: true,
- Default: stringdefault.StaticString(formatTypeDefault),
+ formatTypes := GetDefinitionFormats(definitionFormatTypes)
+ definitionPathKeys := GetDefinitionFormatsPaths(definitionFormatTypes)
+
+ // format attribute
+ attrFormat := schema.StringAttribute{}
+
+ if len(formatTypes) > 1 || (len(formatTypes) == 1 && formatTypes[0] != "") {
+ attrFormat.MarkdownDescription = fmt.Sprintf("The %s format. Possible values: %s", name, utils.ConvertStringSlicesToString(formatTypes, true, false))
+ attrFormat.Validators = []validator.String{
+ stringvalidator.OneOf(utils.ConvertEnumsToStringSlices(formatTypes, true)...),
+ superstringvalidator.RequireIfAttributeIsSet(path.MatchRoot("definition")),
}
- } else {
- attributes["format"] = schema.StringAttribute{
- MarkdownDescription: fmt.Sprintf("The %s format. Possible values: `%s`", name, DefinitionFormatNotApplicable),
- Computed: true,
- Default: stringdefault.StaticString(DefinitionFormatNotApplicable),
+
+ if definitionRequired {
+ attrFormat.Required = true
+ } else {
+ attrFormat.Optional = true
}
+ } else {
+ attrFormat.MarkdownDescription = fmt.Sprintf("The %s format. Possible values: `%s`", name, DefinitionFormatNotApplicable)
+ attrFormat.Computed = true
+ attrFormat.Default = stringdefault.StaticString(DefinitionFormatNotApplicable)
}
+ attributes["format"] = attrFormat
+
+ // definition attribute
+ attrDefinition := schema.MapNestedAttribute{}
+
+ attrDefinition.MarkdownDescription = fmt.Sprintf("Definition parts. Accepted path keys: %s. Read more about [%s definition part paths](%s).", utils.ConvertStringSlicesToString(definitionPathKeys, true, false), name, definitionPathDocsURL)
+ attrDefinition.CustomType = supertypes.NewMapNestedObjectTypeOf[ResourceFabricItemDefinitionPartModel](ctx)
+ attrDefinition.Validators = definitionPathKeysValidator
+ attrDefinition.NestedObject = getResourceFabricItemDefinitionPartSchema(ctx)
+
if definitionRequired {
- attributes["definition"] = schema.MapNestedAttribute{
- MarkdownDescription: fmt.Sprintf("Definition parts. Accepted path keys: %s. Read more about [%s definition part paths](%s).", utils.ConvertStringSlicesToString(definitionPathKeys, true, false), name, definitionPathDocsURL),
- Required: true,
- CustomType: supertypes.NewMapNestedObjectTypeOf[ResourceFabricItemDefinitionPartModel](ctx),
- Validators: definitionPathKeysValidator,
- NestedObject: getResourceFabricItemDefinitionPartSchema(ctx),
- }
+ attrDefinition.Required = true
} else {
- attributes["definition"] = schema.MapNestedAttribute{
- MarkdownDescription: fmt.Sprintf("Definition parts. Accepted path keys: %s. Read more about [%s definition part paths](%s).", utils.ConvertStringSlicesToString(definitionPathKeys, true, false), name, definitionPathDocsURL),
- Optional: true,
- CustomType: supertypes.NewMapNestedObjectTypeOf[ResourceFabricItemDefinitionPartModel](ctx),
- Validators: definitionPathKeysValidator,
- NestedObject: getResourceFabricItemDefinitionPartSchema(ctx),
- }
+ attrDefinition.Optional = true
}
+ attributes["definition"] = attrDefinition
+
return attributes
}
diff --git a/internal/provider/utils/values.go b/internal/provider/utils/values.go
index cf6877df..5fe05cad 100644
--- a/internal/provider/utils/values.go
+++ b/internal/provider/utils/values.go
@@ -16,10 +16,10 @@ import (
func GetValueOrFileValue(attValue, attFile string, value, file types.String) (string, error) {
valueResult := value.ValueString()
- if path := file.ValueString(); path != "" {
- fileRaw, err := os.ReadFile(path)
+ if p := file.ValueString(); p != "" {
+ fileRaw, err := os.ReadFile(p)
if err != nil {
- return "", fmt.Errorf("reading '%s' from file %q: %w", attFile, path, err)
+ return "", fmt.Errorf("reading '%s' from file %q: %w", attFile, p, err)
}
fileResult := strings.TrimSpace(string(fileRaw))
@@ -36,10 +36,10 @@ func GetValueOrFileValue(attValue, attFile string, value, file types.String) (st
func GetCertOrFileCert(attValue, attFile string, value, file types.String) (string, error) {
valueResult := strings.TrimSpace(value.ValueString())
- if path := file.ValueString(); path != "" {
- b64, err := auth.ConvertFileToBase64(path)
+ if p := file.ValueString(); p != "" {
+ b64, err := auth.ConvertFileToBase64(p)
if err != nil {
- return "", fmt.Errorf("reading '%s' from file %q: %w", attFile, path, err)
+ return "", fmt.Errorf("reading '%s' from file %q: %w", attFile, p, err)
}
fileResult := strings.TrimSpace(b64)
diff --git a/internal/services/datapipeline/base.go b/internal/services/datapipeline/base.go
index 89ec5804..a3726454 100644
--- a/internal/services/datapipeline/base.go
+++ b/internal/services/datapipeline/base.go
@@ -7,6 +7,7 @@ import (
fabcore "github.com/microsoft/fabric-sdk-go/fabric/core"
"github.com/microsoft/terraform-provider-fabric/internal/common"
+ "github.com/microsoft/terraform-provider-fabric/internal/pkg/fabricitem"
)
const (
@@ -21,4 +22,10 @@ const (
ItemDefinitionPathDocsURL = "https://learn.microsoft.com/fabric/data-factory/pipeline-rest-api"
)
-var ItemDefinitionPaths = []string{"pipeline-content.json"} //nolint:gochecknoglobals
+var itemDefinitionFormats = []fabricitem.DefinitionFormat{
+ {
+ Type: "",
+ API: "",
+ Paths: []string{"pipeline-content.json"},
+ },
+}
diff --git a/internal/services/datapipeline/resource_data_pipeline.go b/internal/services/datapipeline/resource_data_pipeline.go
index 2730140b..ff3d6b7e 100644
--- a/internal/services/datapipeline/resource_data_pipeline.go
+++ b/internal/services/datapipeline/resource_data_pipeline.go
@@ -24,15 +24,14 @@ func NewResourceDataPipeline() resource.Resource {
DisplayNameMaxLength: 123,
DescriptionMaxLength: 256,
FormatTypeDefault: "",
- FormatTypes: []string{},
DefinitionPathDocsURL: ItemDefinitionPathDocsURL,
- DefinitionPathKeys: ItemDefinitionPaths,
DefinitionPathKeysValidator: []validator.Map{
mapvalidator.SizeAtMost(1),
- mapvalidator.KeysAre(stringvalidator.OneOf(ItemDefinitionPaths...)),
+ mapvalidator.KeysAre(stringvalidator.OneOf(fabricitem.GetDefinitionFormatsPaths(itemDefinitionFormats)...)),
},
DefinitionRequired: false,
DefinitionEmpty: ItemDefinitionEmpty,
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewResourceFabricItemDefinition(config)
diff --git a/internal/services/notebook/base.go b/internal/services/notebook/base.go
index 881e382d..7ad0e386 100644
--- a/internal/services/notebook/base.go
+++ b/internal/services/notebook/base.go
@@ -7,22 +7,31 @@ import (
fabcore "github.com/microsoft/fabric-sdk-go/fabric/core"
"github.com/microsoft/terraform-provider-fabric/internal/common"
+ "github.com/microsoft/terraform-provider-fabric/internal/pkg/fabricitem"
)
const (
- ItemName = "Notebook"
- ItemTFName = "notebook"
- ItemsName = "Notebooks"
- ItemsTFName = "notebooks"
- ItemType = fabcore.ItemTypeNotebook
- ItemDocsSPNSupport = common.DocsSPNSupported
- ItemDocsURL = "https://learn.microsoft.com/fabric/data-engineering/how-to-use-notebook"
- ItemFormatTypeDefault = "ipynb"
- ItemDefinitionEmptyIPYNB = `{"cells":[{"cell_type":"code","metadata":{},"source":["# Welcome to your notebook"]}],"metadata":{"language_info":{"name":"python"}},"nbformat":4,"nbformat_minor":5}`
- ItemDefinitionPathDocsURL = "https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/notebook-definition"
+ ItemName = "Notebook"
+ ItemTFName = "notebook"
+ ItemsName = "Notebooks"
+ ItemsTFName = "notebooks"
+ ItemType = fabcore.ItemTypeNotebook
+ ItemDocsSPNSupport = common.DocsSPNSupported
+ ItemDocsURL = "https://learn.microsoft.com/fabric/data-engineering/how-to-use-notebook"
+ ItemDefinitionFormatTypeDefault = "ipynb"
+ ItemDefinitionEmptyIPYNB = `{"cells":[{"cell_type":"code","metadata":{},"source":["# Welcome to your notebook"]}],"metadata":{"language_info":{"name":"python"}},"nbformat":4,"nbformat_minor":5}`
+ ItemDefinitionPathDocsURL = "https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/notebook-definition"
)
-var (
- ItemFormatTypes = []string{"ipynb"} //nolint:gochecknoglobals
- ItemDefinitionPathsIPYNB = []string{"notebook-content.ipynb"} //nolint:gochecknoglobals
-)
+var itemDefinitionFormats = []fabricitem.DefinitionFormat{
+ {
+ Type: "ipynb",
+ API: "ipynb",
+ Paths: []string{"notebook-content.ipynb"},
+ },
+ {
+ Type: "py",
+ API: "",
+ Paths: []string{"notebook-content.py"},
+ },
+}
diff --git a/internal/services/notebook/data_notebook.go b/internal/services/notebook/data_notebook.go
index d3c9e299..50907fb7 100644
--- a/internal/services/notebook/data_notebook.go
+++ b/internal/services/notebook/data_notebook.go
@@ -18,9 +18,8 @@ func NewDataSourceNotebook() datasource.DataSource {
"Use this data source to fetch a [" + ItemName + "](" + ItemDocsURL + ").\n\n" +
ItemDocsSPNSupport,
IsDisplayNameUnique: true,
- FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
- DefinitionPathKeys: ItemDefinitionPathsIPYNB,
+ FormatTypeDefault: ItemDefinitionFormatTypeDefault,
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewDataSourceFabricItemDefinition(config)
diff --git a/internal/services/notebook/data_notebook_test.go b/internal/services/notebook/data_notebook_test.go
index a8df755a..fbb326e1 100644
--- a/internal/services/notebook/data_notebook_test.go
+++ b/internal/services/notebook/data_notebook_test.go
@@ -173,6 +173,7 @@ func TestAcc_NotebookDataSource(t *testing.T) {
resource.TestCheckResourceAttr(testDataSourceItemFQN, "id", entityID),
resource.TestCheckResourceAttr(testDataSourceItemFQN, "display_name", entityDisplayName),
resource.TestCheckResourceAttr(testDataSourceItemFQN, "description", entityDescription),
+ resource.TestCheckNoResourceAttr(testDataSourceItemFQN, "definition"),
),
},
// read by id - not found
@@ -200,6 +201,7 @@ func TestAcc_NotebookDataSource(t *testing.T) {
resource.TestCheckResourceAttr(testDataSourceItemFQN, "id", entityID),
resource.TestCheckResourceAttr(testDataSourceItemFQN, "display_name", entityDisplayName),
resource.TestCheckResourceAttr(testDataSourceItemFQN, "description", entityDescription),
+ resource.TestCheckNoResourceAttr(testDataSourceItemFQN, "definition"),
),
},
// read by name - not found
@@ -213,5 +215,55 @@ func TestAcc_NotebookDataSource(t *testing.T) {
),
ExpectError: regexp.MustCompile(common.ErrorReadHeader),
},
+ // read by id with definition - default
+ {
+ Config: at.CompileConfig(
+ testDataSourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "id": entityID,
+ "output_definition": true,
+ },
+ ),
+ ExpectError: regexp.MustCompile("Invalid configuration for attribute format"),
+ },
+ // read by id with definition - py
+ {
+ Config: at.CompileConfig(
+ testDataSourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "id": entityID,
+ "output_definition": true,
+ "format": "py",
+ },
+ ),
+ Check: resource.ComposeAggregateTestCheckFunc(
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "workspace_id", workspaceID),
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "id", entityID),
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "display_name", entityDisplayName),
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "description", entityDescription),
+ resource.TestCheckResourceAttrSet(testDataSourceItemFQN, "definition.notebook-content.py.content"),
+ ),
+ },
+ // read by id with definition - ipynb
+ {
+ Config: at.CompileConfig(
+ testDataSourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "id": entityID,
+ "output_definition": true,
+ "format": "ipynb",
+ },
+ ),
+ Check: resource.ComposeAggregateTestCheckFunc(
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "workspace_id", workspaceID),
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "id", entityID),
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "display_name", entityDisplayName),
+ resource.TestCheckResourceAttr(testDataSourceItemFQN, "description", entityDescription),
+ resource.TestCheckResourceAttrSet(testDataSourceItemFQN, "definition.notebook-content.ipynb.content"),
+ ),
+ },
}))
}
diff --git a/internal/services/notebook/resource_notebook.go b/internal/services/notebook/resource_notebook.go
index 48e72528..98fbc5b8 100644
--- a/internal/services/notebook/resource_notebook.go
+++ b/internal/services/notebook/resource_notebook.go
@@ -5,7 +5,6 @@ package notebook
import (
"github.com/hashicorp/terraform-plugin-framework-validators/mapvalidator"
- "github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/schema/validator"
@@ -23,16 +22,15 @@ func NewResourceNotebook() resource.Resource {
ItemDocsSPNSupport,
DisplayNameMaxLength: 123,
DescriptionMaxLength: 256,
- FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
+ FormatTypeDefault: ItemDefinitionFormatTypeDefault,
DefinitionPathDocsURL: ItemDefinitionPathDocsURL,
- DefinitionPathKeys: ItemDefinitionPathsIPYNB,
DefinitionPathKeysValidator: []validator.Map{
mapvalidator.SizeAtMost(1),
- mapvalidator.KeysAre(stringvalidator.OneOf(ItemDefinitionPathsIPYNB...)),
+ mapvalidator.KeysAre(fabricitem.DefinitionPathKeysValidator(itemDefinitionFormats)...),
},
DefinitionRequired: false,
DefinitionEmpty: ItemDefinitionEmptyIPYNB,
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewResourceFabricItemDefinition(config)
diff --git a/internal/services/notebook/resource_notebook_test.go b/internal/services/notebook/resource_notebook_test.go
index 4964133a..88aaa3f7 100644
--- a/internal/services/notebook/resource_notebook_test.go
+++ b/internal/services/notebook/resource_notebook_test.go
@@ -28,12 +28,18 @@ var testHelperLocals = at.CompileLocalsConfig(map[string]any{
"path": testhelp.GetFixturesDirPath("notebook"),
})
-var testHelperDefinition = map[string]any{
+var testHelperDefinitionIPYNB = map[string]any{
`"notebook-content.ipynb"`: map[string]any{
"source": "${local.path}/notebook.ipynb.tmpl",
},
}
+var testHelperDefinitionPY = map[string]any{
+ `"notebook-content.py"`: map[string]any{
+ "source": "${local.path}/notebook.py.tmpl",
+ },
+}
+
func TestUnit_NotebookResource_Attributes(t *testing.T) {
resource.ParallelTest(t, testhelp.NewTestUnitCase(t, &testResourceItemFQN, fakes.FakeServer.ServerFactory, nil, []resource.TestStep{
// error - no attributes
@@ -58,7 +64,8 @@ func TestUnit_NotebookResource_Attributes(t *testing.T) {
map[string]any{
"workspace_id": "invalid uuid",
"display_name": "test",
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
ExpectError: regexp.MustCompile(customtypes.UUIDTypeErrorInvalidStringHeader),
@@ -74,7 +81,8 @@ func TestUnit_NotebookResource_Attributes(t *testing.T) {
"workspace_id": "00000000-0000-0000-0000-000000000000",
"display_name": "test",
"unexpected_attr": "test",
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
ExpectError: regexp.MustCompile(`An argument named "unexpected_attr" is not expected here`),
@@ -88,7 +96,8 @@ func TestUnit_NotebookResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"display_name": "test",
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
ExpectError: regexp.MustCompile(`The argument "workspace_id" is required, but no definition was found.`),
@@ -102,7 +111,8 @@ func TestUnit_NotebookResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"workspace_id": "00000000-0000-0000-0000-000000000000",
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
ExpectError: regexp.MustCompile(`The argument "display_name" is required, but no definition was found.`),
@@ -125,7 +135,8 @@ func TestUnit_NotebookResource_ImportState(t *testing.T) {
map[string]any{
"workspace_id": *entity.WorkspaceID,
"display_name": *entity.DisplayName,
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
))
@@ -202,7 +213,8 @@ func TestUnit_NotebookResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityExist.WorkspaceID,
"display_name": *entityExist.DisplayName,
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
ExpectError: regexp.MustCompile(common.ErrorCreateHeader),
@@ -217,7 +229,8 @@ func TestUnit_NotebookResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityBefore.DisplayName,
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
Check: resource.ComposeAggregateTestCheckFunc(
@@ -236,7 +249,8 @@ func TestUnit_NotebookResource_CRUD(t *testing.T) {
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityAfter.DisplayName,
"description": *entityAfter.Description,
- "definition": testHelperDefinition,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
},
)),
Check: resource.ComposeAggregateTestCheckFunc(
@@ -268,7 +282,110 @@ func TestAcc_NotebookResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityCreateDisplayName,
- "definition": testHelperDefinition,
+ },
+ )),
+ Check: resource.ComposeAggregateTestCheckFunc(
+ resource.TestCheckResourceAttr(testResourceItemFQN, "display_name", entityCreateDisplayName),
+ resource.TestCheckResourceAttr(testResourceItemFQN, "description", ""),
+ ),
+ },
+ // Update and Read
+ {
+ ResourceName: testResourceItemFQN,
+ Config: at.JoinConfigs(
+ testHelperLocals,
+ at.CompileConfig(
+ testResourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "display_name": entityUpdateDisplayName,
+ "description": entityUpdateDescription,
+ },
+ )),
+ Check: resource.ComposeAggregateTestCheckFunc(
+ resource.TestCheckResourceAttr(testResourceItemFQN, "display_name", entityUpdateDisplayName),
+ resource.TestCheckResourceAttr(testResourceItemFQN, "description", entityUpdateDescription),
+ ),
+ },
+ },
+ ))
+}
+
+func TestAcc_NotebookDefinitionIPYNBResource_CRUD(t *testing.T) {
+ workspace := testhelp.WellKnown()["Workspace"].(map[string]any)
+ workspaceID := workspace["id"].(string)
+
+ entityCreateDisplayName := testhelp.RandomName()
+ entityUpdateDisplayName := testhelp.RandomName()
+ entityUpdateDescription := testhelp.RandomName()
+
+ resource.Test(t, testhelp.NewTestAccCase(t, &testResourceItemFQN, nil, []resource.TestStep{
+ // Create and Read
+ {
+ ResourceName: testResourceItemFQN,
+ Config: at.JoinConfigs(
+ testHelperLocals,
+ at.CompileConfig(
+ testResourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "display_name": entityCreateDisplayName,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
+ },
+ )),
+ Check: resource.ComposeAggregateTestCheckFunc(
+ resource.TestCheckResourceAttr(testResourceItemFQN, "display_name", entityCreateDisplayName),
+ resource.TestCheckResourceAttr(testResourceItemFQN, "description", ""),
+ resource.TestCheckResourceAttr(testResourceItemFQN, "definition_update_enabled", "true"),
+ ),
+ },
+ // Update and Read
+ {
+ ResourceName: testResourceItemFQN,
+ Config: at.JoinConfigs(
+ testHelperLocals,
+ at.CompileConfig(
+ testResourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "display_name": entityUpdateDisplayName,
+ "description": entityUpdateDescription,
+ "format": "ipynb",
+ "definition": testHelperDefinitionIPYNB,
+ },
+ )),
+ Check: resource.ComposeAggregateTestCheckFunc(
+ resource.TestCheckResourceAttr(testResourceItemFQN, "display_name", entityUpdateDisplayName),
+ resource.TestCheckResourceAttr(testResourceItemFQN, "description", entityUpdateDescription),
+ resource.TestCheckResourceAttr(testResourceItemFQN, "definition_update_enabled", "true"),
+ ),
+ },
+ },
+ ))
+}
+
+func TestAcc_NotebookDefinitionPYResource_CRUD(t *testing.T) {
+ workspace := testhelp.WellKnown()["Workspace"].(map[string]any)
+ workspaceID := workspace["id"].(string)
+
+ entityCreateDisplayName := testhelp.RandomName()
+ entityUpdateDisplayName := testhelp.RandomName()
+ entityUpdateDescription := testhelp.RandomName()
+
+ resource.Test(t, testhelp.NewTestAccCase(t, &testResourceItemFQN, nil, []resource.TestStep{
+ // Create and Read
+ {
+ ResourceName: testResourceItemFQN,
+ Config: at.JoinConfigs(
+ testHelperLocals,
+ at.CompileConfig(
+ testResourceItemHeader,
+ map[string]any{
+ "workspace_id": workspaceID,
+ "display_name": entityCreateDisplayName,
+ "format": "py",
+ "definition": testHelperDefinitionPY,
},
)),
Check: resource.ComposeAggregateTestCheckFunc(
@@ -288,7 +405,8 @@ func TestAcc_NotebookResource_CRUD(t *testing.T) {
"workspace_id": workspaceID,
"display_name": entityUpdateDisplayName,
"description": entityUpdateDescription,
- "definition": testHelperDefinition,
+ "format": "py",
+ "definition": testHelperDefinitionPY,
},
)),
Check: resource.ComposeAggregateTestCheckFunc(
diff --git a/internal/services/report/base.go b/internal/services/report/base.go
index 84e3c701..8adb9c1f 100644
--- a/internal/services/report/base.go
+++ b/internal/services/report/base.go
@@ -7,6 +7,7 @@ import (
fabcore "github.com/microsoft/fabric-sdk-go/fabric/core"
"github.com/microsoft/terraform-provider-fabric/internal/common"
+ "github.com/microsoft/terraform-provider-fabric/internal/pkg/fabricitem"
)
const (
@@ -21,7 +22,10 @@ const (
ItemDefinitionPathDocsURL = "https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/report-definition"
)
-var (
- ItemFormatTypes = []string{"PBIR-Legacy"} //nolint:gochecknoglobals
- ItemDefinitionPathsPBIRLegacy = []string{"report.json", "definition.pbir", "StaticResources/RegisteredResources/*", "StaticResources/SharedResources/*"} //nolint:gochecknoglobals
-)
+var itemDefinitionFormats = []fabricitem.DefinitionFormat{
+ {
+ Type: "PBIR-Legacy",
+ API: "PBIR-Legacy",
+ Paths: []string{"report.json", "definition.pbir", "StaticResources/RegisteredResources/*", "StaticResources/SharedResources/*"},
+ },
+}
diff --git a/internal/services/report/data_report.go b/internal/services/report/data_report.go
index fd6e4002..cf359c73 100644
--- a/internal/services/report/data_report.go
+++ b/internal/services/report/data_report.go
@@ -19,8 +19,7 @@ func NewDataSourceReport() datasource.DataSource {
ItemDocsSPNSupport,
IsDisplayNameUnique: false,
FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
- DefinitionPathKeys: ItemDefinitionPathsPBIRLegacy,
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewDataSourceFabricItemDefinition(config)
diff --git a/internal/services/report/data_report_test.go b/internal/services/report/data_report_test.go
index ae21f46d..a2aa0455 100644
--- a/internal/services/report/data_report_test.go
+++ b/internal/services/report/data_report_test.go
@@ -156,6 +156,7 @@ func TestAcc_ReportDataSource(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"id": entityID,
+ "format": "PBIR-Legacy",
"output_definition": true,
},
),
diff --git a/internal/services/report/resource_report.go b/internal/services/report/resource_report.go
index 37935f31..e2bda416 100644
--- a/internal/services/report/resource_report.go
+++ b/internal/services/report/resource_report.go
@@ -27,20 +27,19 @@ func NewResourceReport() resource.Resource {
DisplayNameMaxLength: 123,
DescriptionMaxLength: 256,
FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
DefinitionPathDocsURL: ItemDefinitionPathDocsURL,
- DefinitionPathKeys: ItemDefinitionPathsPBIRLegacy,
DefinitionPathKeysValidator: []validator.Map{
mapvalidator.SizeAtLeast(3),
mapvalidator.KeysAre(
stringvalidator.RegexMatches(
regexp.MustCompile(`^(report\.json|definition\.pbir|StaticResources/RegisteredResources/.*|StaticResources/SharedResources/.*)$`),
- "Definition path must match one of the following: "+utils.ConvertStringSlicesToString(ItemDefinitionPathsPBIRLegacy, true, false),
+ "Definition path must match one of the following: "+utils.ConvertStringSlicesToString(fabricitem.GetDefinitionFormatPaths(itemDefinitionFormats, "PBIR-Legacy"), true, false),
),
),
},
DefinitionRequired: true,
DefinitionEmpty: "",
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewResourceFabricItemDefinition(config)
diff --git a/internal/services/report/resource_report_test.go b/internal/services/report/resource_report_test.go
index 57964a52..853efacf 100644
--- a/internal/services/report/resource_report_test.go
+++ b/internal/services/report/resource_report_test.go
@@ -68,6 +68,7 @@ func TestUnit_ReportResource_Attributes(t *testing.T) {
map[string]any{
"workspace_id": "invalid uuid",
"display_name": "test",
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -84,6 +85,7 @@ func TestUnit_ReportResource_Attributes(t *testing.T) {
"workspace_id": "00000000-0000-0000-0000-000000000000",
"display_name": "test",
"unexpected_attr": "test",
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -98,6 +100,7 @@ func TestUnit_ReportResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"display_name": "test",
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -112,6 +115,7 @@ func TestUnit_ReportResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"workspace_id": "00000000-0000-0000-0000-000000000000",
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -149,6 +153,7 @@ func TestUnit_ReportResource_ImportState(t *testing.T) {
map[string]any{
"workspace_id": *entity.WorkspaceID,
"display_name": *entity.DisplayName,
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
))
@@ -231,6 +236,7 @@ func TestUnit_ReportResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityExist.WorkspaceID,
"display_name": *entityExist.DisplayName,
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -246,6 +252,7 @@ func TestUnit_ReportResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityBefore.DisplayName,
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -265,6 +272,7 @@ func TestUnit_ReportResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityAfter.DisplayName,
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -300,6 +308,7 @@ func TestAcc_ReportResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityCreateDisplayName,
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
@@ -319,6 +328,7 @@ func TestAcc_ReportResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityUpdateDisplayName,
+ "format": "PBIR-Legacy",
"definition": testHelperDefinition,
},
)),
diff --git a/internal/services/semanticmodel/base.go b/internal/services/semanticmodel/base.go
index b28d72dd..ed355ee8 100644
--- a/internal/services/semanticmodel/base.go
+++ b/internal/services/semanticmodel/base.go
@@ -7,6 +7,7 @@ import (
fabcore "github.com/microsoft/fabric-sdk-go/fabric/core"
"github.com/microsoft/terraform-provider-fabric/internal/common"
+ "github.com/microsoft/terraform-provider-fabric/internal/pkg/fabricitem"
)
const (
@@ -21,7 +22,10 @@ const (
ItemDefinitionPathDocsURL = "https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/semantic-model-definition"
)
-var (
- ItemFormatTypes = []string{"TMSL"} //nolint:gochecknoglobals
- ItemDefinitionPathsTMSL = []string{"model.bim", "definition.pbism", "diagramLayout.json"} //nolint:gochecknoglobals
-)
+var itemDefinitionFormats = []fabricitem.DefinitionFormat{
+ {
+ Type: "TMSL",
+ API: "TMSL",
+ Paths: []string{"model.bim", "definition.pbism", "diagramLayout.json"},
+ },
+}
diff --git a/internal/services/semanticmodel/data_semantic_model.go b/internal/services/semanticmodel/data_semantic_model.go
index b0ed23bb..fcd7bbdd 100644
--- a/internal/services/semanticmodel/data_semantic_model.go
+++ b/internal/services/semanticmodel/data_semantic_model.go
@@ -19,8 +19,7 @@ func NewDataSourceSemanticModel() datasource.DataSource {
ItemDocsSPNSupport,
IsDisplayNameUnique: false,
FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
- DefinitionPathKeys: ItemDefinitionPathsTMSL,
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewDataSourceFabricItemDefinition(config)
diff --git a/internal/services/semanticmodel/data_semantic_model_test.go b/internal/services/semanticmodel/data_semantic_model_test.go
index 2a27b96a..adb73f57 100644
--- a/internal/services/semanticmodel/data_semantic_model_test.go
+++ b/internal/services/semanticmodel/data_semantic_model_test.go
@@ -156,6 +156,7 @@ func TestAcc_SemanticModelDataSource(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"id": entityID,
+ "format": "TMSL",
"output_definition": true,
},
),
diff --git a/internal/services/semanticmodel/resource_semantic_model.go b/internal/services/semanticmodel/resource_semantic_model.go
index 8cfcfafa..27203b36 100644
--- a/internal/services/semanticmodel/resource_semantic_model.go
+++ b/internal/services/semanticmodel/resource_semantic_model.go
@@ -24,15 +24,14 @@ func NewResourceSemanticModel() resource.Resource {
DisplayNameMaxLength: 123,
DescriptionMaxLength: 256,
FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
DefinitionPathDocsURL: ItemDefinitionPathDocsURL,
- DefinitionPathKeys: ItemDefinitionPathsTMSL,
DefinitionPathKeysValidator: []validator.Map{
mapvalidator.SizeAtLeast(2),
- mapvalidator.KeysAre(stringvalidator.OneOf(ItemDefinitionPathsTMSL...)),
+ mapvalidator.KeysAre(stringvalidator.OneOf(fabricitem.GetDefinitionFormatPaths(itemDefinitionFormats, "TMSL")...)),
},
DefinitionRequired: true,
DefinitionEmpty: "",
+ DefinitionFormats: itemDefinitionFormats,
}
return fabricitem.NewResourceFabricItemDefinition(config)
diff --git a/internal/services/semanticmodel/resource_semantic_model_test.go b/internal/services/semanticmodel/resource_semantic_model_test.go
index b430f6e1..cd7be7ac 100644
--- a/internal/services/semanticmodel/resource_semantic_model_test.go
+++ b/internal/services/semanticmodel/resource_semantic_model_test.go
@@ -64,6 +64,7 @@ func TestUnit_SemanticModelResource_Attributes(t *testing.T) {
map[string]any{
"workspace_id": "invalid uuid",
"display_name": "test",
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -80,6 +81,7 @@ func TestUnit_SemanticModelResource_Attributes(t *testing.T) {
"workspace_id": "00000000-0000-0000-0000-000000000000",
"display_name": "test",
"unexpected_attr": "test",
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -94,6 +96,7 @@ func TestUnit_SemanticModelResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"display_name": "test",
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -108,6 +111,7 @@ func TestUnit_SemanticModelResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"workspace_id": "00000000-0000-0000-0000-000000000000",
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -145,6 +149,7 @@ func TestUnit_SemanticModelResource_ImportState(t *testing.T) {
map[string]any{
"workspace_id": *entity.WorkspaceID,
"display_name": *entity.DisplayName,
+ "format": "TMSL",
"definition": testHelperDefinition,
},
))
@@ -222,6 +227,7 @@ func TestUnit_SemanticModelResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityExist.WorkspaceID,
"display_name": *entityExist.DisplayName,
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -237,6 +243,7 @@ func TestUnit_SemanticModelResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityBefore.DisplayName,
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -256,6 +263,7 @@ func TestUnit_SemanticModelResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityAfter.DisplayName,
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -286,6 +294,7 @@ func TestAcc_SemanticModelResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityCreateDisplayName,
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
@@ -305,6 +314,7 @@ func TestAcc_SemanticModelResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityUpdateDisplayName,
+ "format": "TMSL",
"definition": testHelperDefinition,
},
)),
diff --git a/internal/services/sparkjobdefinition/base.go b/internal/services/sparkjobdefinition/base.go
index 5a9ed42a..f2c7fc2c 100644
--- a/internal/services/sparkjobdefinition/base.go
+++ b/internal/services/sparkjobdefinition/base.go
@@ -7,6 +7,7 @@ import (
fabcore "github.com/microsoft/fabric-sdk-go/fabric/core"
"github.com/microsoft/terraform-provider-fabric/internal/common"
+ "github.com/microsoft/terraform-provider-fabric/internal/pkg/fabricitem"
)
const (
@@ -22,7 +23,10 @@ const (
ItemDefinitionPathDocsURL = "https://learn.microsoft.com/rest/api/fabric/articles/item-management/definitions/spark-job-definition"
)
-var (
- ItemFormatTypes = []string{"SparkJobDefinitionV1"} //nolint:gochecknoglobals
- ItemDefinitionPaths = []string{"SparkJobDefinitionV1.json"} //nolint:gochecknoglobals
-)
+var itemDefinitionFormats = []fabricitem.DefinitionFormat{
+ {
+ Type: "SparkJobDefinitionV1",
+ API: "SparkJobDefinitionV1",
+ Paths: []string{"SparkJobDefinitionV1.json"},
+ },
+}
diff --git a/internal/services/sparkjobdefinition/data_spark_job_definition.go b/internal/services/sparkjobdefinition/data_spark_job_definition.go
index 7fd54d14..c9f7e8cf 100644
--- a/internal/services/sparkjobdefinition/data_spark_job_definition.go
+++ b/internal/services/sparkjobdefinition/data_spark_job_definition.go
@@ -93,8 +93,7 @@ func NewDataSourceSparkJobDefinition(ctx context.Context) datasource.DataSource
ItemDocsSPNSupport,
IsDisplayNameUnique: true,
FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
- DefinitionPathKeys: ItemDefinitionPaths,
+ DefinitionFormats: itemDefinitionFormats,
},
PropertiesSchema: propertiesSchema,
PropertiesSetter: propertiesSetter,
diff --git a/internal/services/sparkjobdefinition/data_spark_job_definition_test.go b/internal/services/sparkjobdefinition/data_spark_job_definition_test.go
index b1b01f8b..0ed0d0bf 100644
--- a/internal/services/sparkjobdefinition/data_spark_job_definition_test.go
+++ b/internal/services/sparkjobdefinition/data_spark_job_definition_test.go
@@ -222,6 +222,7 @@ func TestAcc_SparkJobDefinitionDataSource(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"id": entityID,
+ "format": "SparkJobDefinitionV1",
"output_definition": true,
},
),
diff --git a/internal/services/sparkjobdefinition/resource_spark_job_definition.go b/internal/services/sparkjobdefinition/resource_spark_job_definition.go
index e09f01b4..490b2e52 100644
--- a/internal/services/sparkjobdefinition/resource_spark_job_definition.go
+++ b/internal/services/sparkjobdefinition/resource_spark_job_definition.go
@@ -8,7 +8,6 @@ import (
supertypes "github.com/FrangipaneTeam/terraform-plugin-framework-supertypes"
"github.com/hashicorp/terraform-plugin-framework-validators/mapvalidator"
- "github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework/diag"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
@@ -75,15 +74,14 @@ func NewResourceSparkJobDefinition(ctx context.Context) resource.Resource {
DisplayNameMaxLength: 123,
DescriptionMaxLength: 256,
FormatTypeDefault: ItemFormatTypeDefault,
- FormatTypes: ItemFormatTypes,
DefinitionPathDocsURL: ItemDefinitionPathDocsURL,
- DefinitionPathKeys: ItemDefinitionPaths,
DefinitionPathKeysValidator: []validator.Map{
mapvalidator.SizeAtMost(1),
- mapvalidator.KeysAre(stringvalidator.OneOf(ItemDefinitionPaths...)),
+ mapvalidator.KeysAre(fabricitem.DefinitionPathKeysValidator(itemDefinitionFormats)...),
},
DefinitionRequired: false,
DefinitionEmpty: ItemDefinitionEmpty,
+ DefinitionFormats: itemDefinitionFormats,
},
PropertiesSchema: propertiesSchema,
PropertiesSetter: propertiesSetter,
diff --git a/internal/services/sparkjobdefinition/resource_spark_job_definition_test.go b/internal/services/sparkjobdefinition/resource_spark_job_definition_test.go
index eba7fd27..b8ae38cf 100644
--- a/internal/services/sparkjobdefinition/resource_spark_job_definition_test.go
+++ b/internal/services/sparkjobdefinition/resource_spark_job_definition_test.go
@@ -58,6 +58,7 @@ func TestUnit_SparkJobDefinitionResource_Attributes(t *testing.T) {
map[string]any{
"workspace_id": "invalid uuid",
"display_name": "test",
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -74,6 +75,7 @@ func TestUnit_SparkJobDefinitionResource_Attributes(t *testing.T) {
"workspace_id": "00000000-0000-0000-0000-000000000000",
"display_name": "test",
"unexpected_attr": "test",
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -88,6 +90,7 @@ func TestUnit_SparkJobDefinitionResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"display_name": "test",
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -102,6 +105,7 @@ func TestUnit_SparkJobDefinitionResource_Attributes(t *testing.T) {
testResourceItemHeader,
map[string]any{
"workspace_id": "00000000-0000-0000-0000-000000000000",
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -125,6 +129,7 @@ func TestUnit_SparkJobDefinitionResource_ImportState(t *testing.T) {
map[string]any{
"workspace_id": *entity.WorkspaceID,
"display_name": *entity.DisplayName,
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
))
@@ -202,6 +207,7 @@ func TestUnit_SparkJobDefinitionResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityExist.WorkspaceID,
"display_name": *entityExist.DisplayName,
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -217,6 +223,7 @@ func TestUnit_SparkJobDefinitionResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityBefore.DisplayName,
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -236,6 +243,7 @@ func TestUnit_SparkJobDefinitionResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": *entityBefore.WorkspaceID,
"display_name": *entityAfter.DisplayName,
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -266,6 +274,7 @@ func TestAcc_SparkJobDefinitionResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityCreateDisplayName,
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
@@ -285,6 +294,7 @@ func TestAcc_SparkJobDefinitionResource_CRUD(t *testing.T) {
map[string]any{
"workspace_id": workspaceID,
"display_name": entityUpdateDisplayName,
+ "format": "SparkJobDefinitionV1",
"definition": testHelperDefinition,
},
)),
diff --git a/internal/services/workspace/resource_workspace_git.go b/internal/services/workspace/resource_workspace_git.go
index 07348d7f..adb69cf3 100644
--- a/internal/services/workspace/resource_workspace_git.go
+++ b/internal/services/workspace/resource_workspace_git.go
@@ -315,7 +315,7 @@ func (r *resourceWorkspaceGit) Read(ctx context.Context, req resource.ReadReques
return
}
- resp.Diagnostics.Append(resp.State.Set(ctx, &state)...)
+ resp.Diagnostics.Append(resp.State.Set(ctx, state)...)
tflog.Debug(ctx, "READ", map[string]any{
"action": "end",
diff --git a/internal/testhelp/fixtures/notebook/notebook.py.tmpl b/internal/testhelp/fixtures/notebook/notebook.py.tmpl
new file mode 100644
index 00000000..6e270f50
--- /dev/null
+++ b/internal/testhelp/fixtures/notebook/notebook.py.tmpl
@@ -0,0 +1,22 @@
+# Fabric notebook source
+
+# METADATA ********************
+
+# META {
+# META "kernel_info": {
+# META "name": "synapse_pyspark"
+# META },
+# META "dependencies": {}
+# META }
+
+# CELL ********************
+
+# Welcome to your new notebook
+# Type here in the cell editor to add code!
+
+# METADATA ********************
+
+# META {
+# META "language": "python",
+# META "language_group": "synapse_pyspark"
+# META }