From b5e554762eca453e46807b782c2e2ce10053b719 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 25 Jul 2024 14:41:11 +0100 Subject: [PATCH 001/117] importer-msgraph-metadata: correct casing for constant names --- .../importer-msgraph-metadata/components/parser/resourceids.go | 2 +- .../internal/pipeline/task_translate_service.go | 3 ++- .../internal/pipeline/translate_models.go | 3 ++- tools/importer-msgraph-metadata/main.go | 2 +- 4 files changed, 6 insertions(+), 4 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 6410d8644e4..63197bfce27 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -82,7 +82,7 @@ func (r ResourceId) DataApiSdkResourceId() (*sdkModels.ResourceID, error) { case SegmentAction, SegmentCast, SegmentFunction, SegmentLabel, SegmentODataReference: sdkSegments = append(sdkSegments, sdkModels.NewStaticValueResourceIDSegment(segment.Value, segment.Value)) case SegmentUserValue: - sdkSegments = append(sdkSegments, sdkModels.NewUserSpecifiedResourceIDSegment(*segment.Field, segment.Value)) + sdkSegments = append(sdkSegments, sdkModels.NewUserSpecifiedResourceIDSegment(normalize.CleanNameCamel(*segment.Field), segment.Value)) default: return nil, fmt.Errorf("unknown segment type %q at index %d for resource ID: %q", segment.Type, i, r.Name) } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index 30e57bb9687..acf1c65286c 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -7,6 +7,7 @@ import ( "fmt" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" ) @@ -159,7 +160,7 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta constantValues := make(map[string]string) if constant, ok := constants[*field.ConstantName]; ok { for _, value := range constant.Enum { - constantValues[value] = value + constantValues[normalize.CleanName(value)] = value } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go index 00666340fff..2bdb5f614a9 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go @@ -5,6 +5,7 @@ package pipeline import ( sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" ) @@ -24,7 +25,7 @@ func translateModelsToDataApiSdkTypes(models parser.Models, constants parser.Con for constantName, constant := range constants { constantValues := make(map[string]string) for _, value := range constant.Enum { - constantValues[value] = value + constantValues[normalize.CleanName(value)] = value } // TODO support additional types, if there are any diff --git a/tools/importer-msgraph-metadata/main.go b/tools/importer-msgraph-metadata/main.go index 15287f26c40..1c919bbf236 100644 --- a/tools/importer-msgraph-metadata/main.go +++ b/tools/importer-msgraph-metadata/main.go @@ -27,7 +27,7 @@ func main() { } logging.Log = hclog.New(loggingOpts) - c := cli.NewCLI("importer-msgraph-metadata", "0.2.0") + c := cli.NewCLI("importer-msgraph-metadata", "0.2.1") c.Args = os.Args[1:] c.Commands = map[string]cli.CommandFactory{ "import": cmd.NewImportCommand(metadataDirectory, microsoftGraphConfig, openApiFilePattern, outputDirectory), From 19178fc6f42a94b80c6ee9464dacb0331f9ca760 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 25 Jul 2024 14:45:03 +0100 Subject: [PATCH 002/117] =?UTF-8?q?generator-go-sdk:=20support=20generatin?= =?UTF-8?q?g=20Common=20Types=20and=20a=20Microsoft=20Graph=20SDK=20?= =?UTF-8?q?=F0=9F=8D=BB?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../internal/differ/differ_helpers.go | 2 +- .../golang_type_for_sdk_object_definition.go | 20 ++- .../v1/models/source_data_types.go | 4 + .../generator-go-sdk/internal/cmd/generate.go | 32 +++-- .../internal/generator/data.go | 128 +++++++++++++++--- .../internal/generator/meta_client.go | 37 ----- .../internal/generator/service.go | 47 ++++--- .../internal/generator/settings.go | 1 + .../internal/generator/stage_clients.go | 2 +- .../internal/generator/stage_common_types.go | 48 +++++++ .../internal/generator/stage_constants.go | 2 +- .../internal/generator/stage_ids.go | 2 +- .../internal/generator/stage_meta_client.go | 39 ++++++ .../internal/generator/stage_methods.go | 2 +- .../internal/generator/stage_models.go | 2 +- .../internal/generator/stage_predicates.go | 2 +- .../internal/generator/stage_readme.go | 2 +- .../internal/generator/stage_version.go | 2 +- .../internal/generator/templater.go | 4 +- .../internal/generator/templater_clients.go | 2 +- .../generator/templater_clients_autorest.go | 2 +- .../templater_clients_autorest_test.go | 2 +- .../generator/templater_clients_test.go | 2 +- .../internal/generator/templater_constants.go | 2 +- .../generator/templater_constants_test.go | 4 +- .../internal/generator/templater_id_parser.go | 11 +- .../generator/templater_id_parser_test.go | 4 +- .../generator/templater_id_parser_tests.go | 2 +- .../generator/templater_meta_client.go | 3 +- .../templater_meta_client_autorest.go | 3 +- .../internal/generator/templater_methods.go | 36 +++-- .../generator/templater_methods_autorest.go | 32 ++--- .../templater_methods_autorest_test.go | 10 +- .../templater_methods_discriminators_test.go | 8 +- ...er_methods_long_running_operations_test.go | 6 +- .../generator/templater_methods_test.go | 10 +- .../internal/generator/templater_models.go | 24 ++-- .../templater_models_constant_test.go | 36 ++--- .../templater_models_discriminators_test.go | 8 +- .../generator/templater_models_test.go | 6 +- .../generator/templater_predicates.go | 4 +- .../internal/generator/templater_readme.go | 32 ++--- .../generator/templater_readme_test.go | 42 +++--- .../internal/generator/templater_version.go | 2 +- .../generator/templater_version_test.go | 2 +- .../generator/mappings/assignment_direct.go | 6 +- .../mappings/assignment_model_to_model.go | 4 +- .../resource/component_create_func.go | 2 +- .../resource/component_update_func.go | 2 +- 49 files changed, 439 insertions(+), 248 deletions(-) delete mode 100644 tools/generator-go-sdk/internal/generator/meta_client.go create mode 100644 tools/generator-go-sdk/internal/generator/stage_common_types.go create mode 100644 tools/generator-go-sdk/internal/generator/stage_meta_client.go diff --git a/tools/data-api-differ/internal/differ/differ_helpers.go b/tools/data-api-differ/internal/differ/differ_helpers.go index 6766612e1e6..ff7e36b646c 100644 --- a/tools/data-api-differ/internal/differ/differ_helpers.go +++ b/tools/data-api-differ/internal/differ/differ_helpers.go @@ -12,7 +12,7 @@ import ( // stringifySDKObjectDefinition returns a human readable, string version of this SDKObjectDefinition. func (d differ) stringifySDKObjectDefinition(input models.SDKObjectDefinition) (*string, error) { - return helpers.GolangTypeForSDKObjectDefinition(input, nil) + return helpers.GolangTypeForSDKObjectDefinition(input, nil, nil, nil) } // stringifySDKOperationOptionObjectDefinition returns a human readable, string version of this Object Definition. diff --git a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go index 856da78baf4..4a000b4a0f3 100644 --- a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go +++ b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go @@ -11,7 +11,7 @@ import ( ) // GolangTypeForSDKObjectDefinition returns the Golang type name for the SDKObjectDefinition provided in `input`. -func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPackageName *string) (*string, error) { +func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPackageName *string, serviceTypeNames *[]string, commonTypesPackageName *string) (*string, error) { // TODO: we should look to add unit tests for this method // NOTE: all of this validation should be done in the Importer and the API - this is purely sanity checking @@ -31,7 +31,7 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa return nil, fmt.Errorf("a dictionary type must have a nested item but didn't get one") } - innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName) + innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName, serviceTypeNames, commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining inner type for dictionary: %+v", err) } @@ -44,7 +44,7 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa return nil, fmt.Errorf("a list type must have a nested item but didn't get one") } - innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName) + innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName, serviceTypeNames, commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining inner type for list: %+v", err) } @@ -60,7 +60,21 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa out := *input.ReferenceName if golangPackageName != nil { out = fmt.Sprintf("%s.%s", *golangPackageName, out) + } else if serviceTypeNames != nil && commonTypesPackageName != nil { + // when serviceTypeNames is specified, look to see if the referenced type is present in the service, and if not, assume it's a common type + found := false + for _, typeName := range *serviceTypeNames { + if typeName == *input.ReferenceName { + found = true + break + } + } + + if !found { + out = fmt.Sprintf("%s.%s", *commonTypesPackageName, out) + } } + return pointer.To(out), nil } diff --git a/tools/data-api-sdk/v1/models/source_data_types.go b/tools/data-api-sdk/v1/models/source_data_types.go index 552884900fa..fc2e71a2b8d 100644 --- a/tools/data-api-sdk/v1/models/source_data_types.go +++ b/tools/data-api-sdk/v1/models/source_data_types.go @@ -13,3 +13,7 @@ const ( // ResourceManagerSourceDataType defines that this Data is related to Azure Resource Manager. ResourceManagerSourceDataType SourceDataType = "resource-manager" ) + +func SourceDataTypeIsDataPlane(sourceDataType SourceDataType) bool { + return sourceDataType == ResourceManagerSourceDataType +} diff --git a/tools/generator-go-sdk/internal/cmd/generate.go b/tools/generator-go-sdk/internal/cmd/generate.go index 419a61b975b..c0ea36444d0 100644 --- a/tools/generator-go-sdk/internal/cmd/generate.go +++ b/tools/generator-go-sdk/internal/cmd/generate.go @@ -34,6 +34,8 @@ type GeneratorInput struct { settings generator.Settings } +const commonTypesPackageName = "common-types" + func NewGenerateCommand(sourceDataType models.SourceDataType) func() (cli.Command, error) { return func() (cli.Command, error) { return GenerateCommand{ @@ -51,7 +53,9 @@ func (g GenerateCommand) Run(args []string) int { ctx := context.Background() input := GeneratorInput{ - settings: generator.Settings{}, + settings: generator.Settings{ + CommonTypesPackageName: commonTypesPackageName, + }, } input.settings.UseOldBaseLayerFor( @@ -138,17 +142,25 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { logging.Debugf("Service %q", serviceName) for versionNumber, versionDetails := range service.APIVersions { logging.Debugf(" Version %q", versionNumber) + + var commonTypes models.CommonTypes + if v, ok := data.CommonTypes[versionNumber]; ok { + commonTypes = v + } + for resourceName, resourceDetails := range versionDetails.Resources { logging.Debugf(" Resource %q", resourceName) generatorData := generator.ServiceGeneratorInput{ - ServiceName: serviceName, - ServiceDetails: service, - VersionName: versionNumber, - VersionDetails: versionDetails, - ResourceName: resourceName, - ResourceDetails: resourceDetails, + CommonTypes: commonTypes, OutputDirectory: input.outputDirectory, + ResourceDetails: resourceDetails, + ResourceName: resourceName, + ServiceDetails: service, + ServiceName: serviceName, Source: versionDetails.Source, + Type: g.sourceDataType, + VersionDetails: versionDetails, + VersionName: versionNumber, } logging.Debugf("Generating Service %q / Version %q / Resource %q", serviceName, versionNumber, resourceName) if err := generatorService.Generate(generatorData); err != nil { @@ -158,13 +170,15 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { logging.Debugf("Generated Service %q / Version %q / Resource %q", serviceName, versionNumber, resourceName) } - // then output the Meta Client - generatorData := generator.VersionInput{ + // then output Common Types and Meta Client + generatorData := generator.VersionGeneratorInput{ OutputDirectory: input.outputDirectory, + CommonTypes: commonTypes, ServiceName: serviceName, VersionName: versionNumber, Resources: versionDetails.Resources, Source: versionDetails.Source, + Type: g.sourceDataType, } generatorData.UseNewBaseLayer = false if input.settings.ShouldUseNewBaseLayer(serviceName, versionNumber) { diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 67fc7913399..508fe4e0dc5 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -8,13 +8,23 @@ import ( "path/filepath" "strings" + "github.com/hashicorp/go-azure-helpers/lang/pointer" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) -type ServiceGeneratorData struct { - // the name of the package which should be used e.g. Service1 becomes service1 +type GeneratorData struct { + // the name of the package which should be used packageName string + // commonTypes contains any common models and constants for this API version + commonTypes models.CommonTypes + + // the name of the package containing common types (unique to the API version) + commonTypesPackageName *string + + // relative include path for common types package + commonTypesIncludePath *string + // the name of the Client e.g. MyThingClient serviceClientName string @@ -23,7 +33,7 @@ type ServiceGeneratorData struct { idsOutputPath string // This is the working directory where files should be output for this specific service - // for example {workingDir}/service/version/resource + // for example {workingDir}/{service}/{version}/{resource} resourceOutputPath string // constants is a map of Constant Name (key) to SDKConstant (value) describing @@ -51,6 +61,16 @@ type ServiceGeneratorData struct { // the name of the service as a package (e.g. resources or eventhub) servicePackageName string + // slice of constant/model type names present for the service, used to determine whether a referenced type + // is local to the service, or is a common type + serviceTypeNames []string + + // sourceType is the source data type and is the SDK package name + sourceType models.SourceDataType + + // whether this is a data plane SDK (omits certain Resource Manager specific features) + isDataPlane bool + // development feature flag - this requires work in the Resource ID parser to handle name conflicts // @tombuildsstuff: fix this useIdAliases bool @@ -60,30 +80,96 @@ type ServiceGeneratorData struct { useNewBaseLayer bool } -func (i ServiceGeneratorInput) generatorData(settings Settings) ServiceGeneratorData { +func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { + resourcePackageName := strings.ToLower(i.ResourceName) servicePackageName := strings.ToLower(i.ServiceName) versionPackageName := strings.ToLower(i.VersionName) - // TODO: it'd be nice to make these snake_case but that's a problem for another day - resourcePackageName := strings.ToLower(i.ResourceName) + versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionPackageName) - resourceOutputPath := filepath.Join(versionOutputPath, resourcePackageName) idsPath := filepath.Join(versionOutputPath, "ids") + resourceOutputPath := filepath.Join(versionOutputPath, resourcePackageName) + + useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) + + serviceTypeNames := make([]string, 0) + for constantName := range i.ResourceDetails.Constants { + serviceTypeNames = append(serviceTypeNames, constantName) + } + for modelName := range i.ResourceDetails.Models { + serviceTypeNames = append(serviceTypeNames, modelName) + } + + var commonTypesPackageName, commonTypesIncludePath *string + if len(i.CommonTypes.Constants) > 0 && len(i.CommonTypes.Models) > 0 { + commonTypesPackageName = pointer.To(strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_"))) + commonTypesIncludePath = pointer.To(fmt.Sprintf("%s/%s", settings.CommonTypesPackageName, *commonTypesPackageName)) + } + + return GeneratorData{ + apiVersion: i.VersionName, + commonTypes: i.CommonTypes, + commonTypesIncludePath: commonTypesIncludePath, + commonTypesPackageName: commonTypesPackageName, + constants: i.ResourceDetails.Constants, + idsOutputPath: idsPath, + isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), + models: i.ResourceDetails.Models, + operations: i.ResourceDetails.Operations, + packageName: resourcePackageName, + resourceIds: i.ResourceDetails.ResourceIDs, + resourceOutputPath: resourceOutputPath, + serviceClientName: fmt.Sprintf("%sClient", strings.Title(i.ResourceName)), + servicePackageName: strings.ToLower(i.ServiceName), + serviceTypeNames: serviceTypeNames, + source: i.Source, + sourceType: i.Type, + useIdAliases: false, + useNewBaseLayer: useNewBaseLayer, + } +} + +type VersionGeneratorData struct { + GeneratorData + + // This is the directory where versioned common types should be output + // for example {workingDir}/common-types/{version} + commonTypesOutputPath string + + // resources specifies a map of API Resource Names (key) to APIResource (value). + resources map[string]models.APIResource + + // This is the directory for the API version where the meta client should be output + // for example {workingDir}/{service}/{version} + versionOutputPath string + + // the name of the version as a package + versionPackageName string +} + +func (i VersionGeneratorInput) generatorData(settings Settings) VersionGeneratorData { + servicePackageName := strings.ToLower(i.ServiceName) + versionPackageName := strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_")) + + commonTypesOutputPath := filepath.Join(i.OutputDirectory, settings.CommonTypesPackageName, versionPackageName) + versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionPackageName) useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) - return ServiceGeneratorData{ - apiVersion: i.VersionName, - constants: i.ResourceDetails.Constants, - idsOutputPath: idsPath, - models: i.ResourceDetails.Models, - operations: i.ResourceDetails.Operations, - resourceOutputPath: resourceOutputPath, - packageName: resourcePackageName, - resourceIds: i.ResourceDetails.ResourceIDs, - serviceClientName: fmt.Sprintf("%sClient", strings.Title(i.ResourceName)), - servicePackageName: strings.ToLower(i.ServiceName), - source: i.Source, - useIdAliases: false, - useNewBaseLayer: useNewBaseLayer, + return VersionGeneratorData{ + GeneratorData: GeneratorData{ + apiVersion: i.VersionName, + commonTypes: i.CommonTypes, + constants: i.CommonTypes.Constants, + isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), + models: i.CommonTypes.Models, + servicePackageName: strings.ToLower(i.ServiceName), + source: i.Source, + sourceType: i.Type, + useNewBaseLayer: useNewBaseLayer, + }, + commonTypesOutputPath: commonTypesOutputPath, + resources: i.Resources, + versionOutputPath: versionOutputPath, + versionPackageName: versionPackageName, } } diff --git a/tools/generator-go-sdk/internal/generator/meta_client.go b/tools/generator-go-sdk/internal/generator/meta_client.go deleted file mode 100644 index 917f9c00197..00000000000 --- a/tools/generator-go-sdk/internal/generator/meta_client.go +++ /dev/null @@ -1,37 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package generator - -import ( - "fmt" -) - -func (s *ServiceGenerator) metaClient(data VersionInput, versionDirectory string) error { - if len(data.Resources) == 0 { - return nil - } - - var templater templaterForVersion - if data.UseNewBaseLayer { - templater = metaClientTemplater{ - serviceName: data.ServiceName, - apiVersion: data.VersionName, - resources: data.Resources, - source: data.Source, - } - } else { - templater = metaClientAutorestTemplater{ - serviceName: data.ServiceName, - apiVersion: data.VersionName, - resources: data.Resources, - source: data.Source, - } - } - - if err := s.writeToPathForVersion(versionDirectory, "client.go", templater); err != nil { - return fmt.Errorf("templating meta client for API Version %q / Service %q: %+v", data.VersionName, data.ServiceName, err) - } - - return nil -} diff --git a/tools/generator-go-sdk/internal/generator/service.go b/tools/generator-go-sdk/internal/generator/service.go index 4f51f3cfaa2..4cac1c2052e 100644 --- a/tools/generator-go-sdk/internal/generator/service.go +++ b/tools/generator-go-sdk/internal/generator/service.go @@ -7,8 +7,6 @@ import ( "fmt" "os" "os/exec" - "path/filepath" - "strings" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/generator-go-sdk/internal/logging" @@ -25,14 +23,16 @@ func NewServiceGenerator(settings Settings) ServiceGenerator { } type ServiceGeneratorInput struct { - ServiceName string - ServiceDetails models.Service - VersionName string - VersionDetails models.APIVersion - ResourceName string - ResourceDetails models.APIResource + CommonTypes models.CommonTypes OutputDirectory string + ResourceDetails models.APIResource + ResourceName string + ServiceDetails models.Service + ServiceName string Source models.SourceDataOrigin + Type models.SourceDataType + VersionDetails models.APIVersion + VersionName string } func (s *ServiceGenerator) Generate(input ServiceGeneratorInput) error { @@ -47,7 +47,7 @@ func (s *ServiceGenerator) Generate(input ServiceGeneratorInput) error { } } - stages := map[string]func(data ServiceGeneratorData) error{ + stages := map[string]func(data GeneratorData) error{ "clients": s.clients, "constants": s.constants, "ids": s.ids, @@ -70,32 +70,41 @@ func (s *ServiceGenerator) Generate(input ServiceGeneratorInput) error { return nil } -type VersionInput struct { +type VersionGeneratorInput struct { + CommonTypes models.CommonTypes OutputDirectory string Resources map[string]models.APIResource ServiceName string Source models.SourceDataOrigin + Type models.SourceDataType UseNewBaseLayer bool VersionName string } -func (s *ServiceGenerator) GenerateForVersion(input VersionInput) error { - input.ServiceName = strings.ToLower(input.ServiceName) - input.VersionName = strings.ToLower(input.VersionName) - versionDirectory := filepath.Join(input.OutputDirectory, input.ServiceName, input.VersionName) +func (s *ServiceGenerator) GenerateForVersion(input VersionGeneratorInput) error { + data := input.generatorData(s.settings) - stages := map[string]func(data VersionInput, versionDirectory string) error{ - "metaClient": s.metaClient, + if err := cleanAndRecreateWorkingDirectory(data.commonTypesOutputPath); err != nil { + return fmt.Errorf("cleaning/recreating working directory %q: %+v", data.commonTypesOutputPath, err) + } + + stages := map[string]func(data VersionGeneratorData) error{ + "commonTypes": s.commonTypes, + "metaClient": s.metaClient, } for name, stage := range stages { logging.Debugf("Running Stage %q..", name) - if err := stage(input, versionDirectory); err != nil { + if err := stage(data); err != nil { return fmt.Errorf("generating %s: %+v", name, err) } } - runGoFmt(versionDirectory) - runGoImports(versionDirectory) + runGoFmt(data.commonTypesOutputPath) + runGoImports(data.commonTypesOutputPath) + + runGoFmt(data.versionOutputPath) + runGoImports(data.versionOutputPath) + return nil } diff --git a/tools/generator-go-sdk/internal/generator/settings.go b/tools/generator-go-sdk/internal/generator/settings.go index 8e4fffdf704..663937a06cc 100644 --- a/tools/generator-go-sdk/internal/generator/settings.go +++ b/tools/generator-go-sdk/internal/generator/settings.go @@ -8,6 +8,7 @@ import ( ) type Settings struct { + CommonTypesPackageName string servicesUsingOldBaseLayer map[string]struct{} } diff --git a/tools/generator-go-sdk/internal/generator/stage_clients.go b/tools/generator-go-sdk/internal/generator/stage_clients.go index c293949bd7a..a58a376d935 100644 --- a/tools/generator-go-sdk/internal/generator/stage_clients.go +++ b/tools/generator-go-sdk/internal/generator/stage_clients.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) clients(data ServiceGeneratorData) error { +func (s *ServiceGenerator) clients(data GeneratorData) error { if data.useNewBaseLayer { if err := s.writeToPathForResource(data.resourceOutputPath, "client.go", clientsTemplater{}, data); err != nil { return fmt.Errorf("templating client (using hashicorp/go-azure-sdk): %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/stage_common_types.go b/tools/generator-go-sdk/internal/generator/stage_common_types.go new file mode 100644 index 00000000000..b4c4ea7c5c9 --- /dev/null +++ b/tools/generator-go-sdk/internal/generator/stage_common_types.go @@ -0,0 +1,48 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package generator + +import ( + "fmt" + "strings" + + "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" +) + +func (s *ServiceGenerator) commonTypes(data VersionGeneratorData) error { + if len(data.commonTypes.Constants) == 0 && len(data.commonTypes.Models) == 0 { + return nil + } + + data.packageName = data.versionPackageName + + // output constants into individual files + for constantName, constant := range data.commonTypes.Constants { + t := constantsTemplater{ + constantTemplateFunc: templateForConstant, + } + constantData := data.GeneratorData + constantData.constants = map[string]models.SDKConstant{ + constantName: constant, + } + fileName := fmt.Sprintf("constant_%s.go", strings.ToLower(constantName)) + if err := s.writeToPathForResource(data.commonTypesOutputPath, fileName, t, constantData); err != nil { + return fmt.Errorf("templating constants: %+v", err) + } + } + + for modelName, model := range data.commonTypes.Models { + fileName := fmt.Sprintf("model_%s.go", strings.ToLower(modelName)) + gen := modelsTemplater{ + name: modelName, + model: model, + } + + if err := s.writeToPathForResource(data.commonTypesOutputPath, fileName, gen, data.GeneratorData); err != nil { + return fmt.Errorf("templating model for %q: %+v", modelName, err) + } + } + + return nil +} diff --git a/tools/generator-go-sdk/internal/generator/stage_constants.go b/tools/generator-go-sdk/internal/generator/stage_constants.go index 9215af06fa8..c863b33790c 100644 --- a/tools/generator-go-sdk/internal/generator/stage_constants.go +++ b/tools/generator-go-sdk/internal/generator/stage_constants.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) constants(data ServiceGeneratorData) error { +func (s *ServiceGenerator) constants(data GeneratorData) error { if len(data.constants) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_ids.go b/tools/generator-go-sdk/internal/generator/stage_ids.go index b7997b2b06f..8381e1511b1 100644 --- a/tools/generator-go-sdk/internal/generator/stage_ids.go +++ b/tools/generator-go-sdk/internal/generator/stage_ids.go @@ -8,7 +8,7 @@ import ( "strings" ) -func (s *ServiceGenerator) ids(data ServiceGeneratorData) error { +func (s *ServiceGenerator) ids(data GeneratorData) error { outputDirectory := data.resourceOutputPath for idName, resourceData := range data.resourceIds { diff --git a/tools/generator-go-sdk/internal/generator/stage_meta_client.go b/tools/generator-go-sdk/internal/generator/stage_meta_client.go new file mode 100644 index 00000000000..68b34079ac4 --- /dev/null +++ b/tools/generator-go-sdk/internal/generator/stage_meta_client.go @@ -0,0 +1,39 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package generator + +import ( + "fmt" +) + +func (s *ServiceGenerator) metaClient(data VersionGeneratorData) error { + if len(data.resources) == 0 { + return nil + } + + var templater templaterForVersion + if data.useNewBaseLayer { + templater = metaClientTemplater{ + serviceName: data.servicePackageName, + apiVersion: data.versionPackageName, + resources: data.resources, + source: data.source, + sourceType: data.sourceType, + } + } else { + templater = metaClientAutorestTemplater{ + serviceName: data.servicePackageName, + apiVersion: data.versionPackageName, + resources: data.resources, + source: data.source, + sourceType: data.sourceType, + } + } + + if err := s.writeToPathForVersion(data.versionOutputPath, "client.go", templater); err != nil { + return fmt.Errorf("templating meta client for API Version %q / Service %q: %+v", data.versionPackageName, data.servicePackageName, err) + } + + return nil +} diff --git a/tools/generator-go-sdk/internal/generator/stage_methods.go b/tools/generator-go-sdk/internal/generator/stage_methods.go index 41e6722e4c7..852e37d577f 100644 --- a/tools/generator-go-sdk/internal/generator/stage_methods.go +++ b/tools/generator-go-sdk/internal/generator/stage_methods.go @@ -8,7 +8,7 @@ import ( "strings" ) -func (s *ServiceGenerator) methods(data ServiceGeneratorData) error { +func (s *ServiceGenerator) methods(data GeneratorData) error { for operationName, operation := range data.operations { if data.useNewBaseLayer { diff --git a/tools/generator-go-sdk/internal/generator/stage_models.go b/tools/generator-go-sdk/internal/generator/stage_models.go index 97c618f713c..5e43a0ce5d4 100644 --- a/tools/generator-go-sdk/internal/generator/stage_models.go +++ b/tools/generator-go-sdk/internal/generator/stage_models.go @@ -8,7 +8,7 @@ import ( "strings" ) -func (s *ServiceGenerator) models(data ServiceGeneratorData) error { +func (s *ServiceGenerator) models(data GeneratorData) error { if len(data.models) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_predicates.go b/tools/generator-go-sdk/internal/generator/stage_predicates.go index 3e4d527f01a..d2d6bf03f12 100644 --- a/tools/generator-go-sdk/internal/generator/stage_predicates.go +++ b/tools/generator-go-sdk/internal/generator/stage_predicates.go @@ -8,7 +8,7 @@ import ( "sort" ) -func (s *ServiceGenerator) predicates(data ServiceGeneratorData) error { +func (s *ServiceGenerator) predicates(data GeneratorData) error { modelNames := make(map[string]struct{}, 0) for _, operation := range data.operations { if operation.FieldContainingPaginationDetails == nil { diff --git a/tools/generator-go-sdk/internal/generator/stage_readme.go b/tools/generator-go-sdk/internal/generator/stage_readme.go index 52028ca4b4c..4c32b9c02dc 100644 --- a/tools/generator-go-sdk/internal/generator/stage_readme.go +++ b/tools/generator-go-sdk/internal/generator/stage_readme.go @@ -8,7 +8,7 @@ import ( "sort" ) -func (s *ServiceGenerator) readmeFile(data ServiceGeneratorData) error { +func (s *ServiceGenerator) readmeFile(data GeneratorData) error { if len(data.models) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_version.go b/tools/generator-go-sdk/internal/generator/stage_version.go index 3f829d6df57..e95817a3068 100644 --- a/tools/generator-go-sdk/internal/generator/stage_version.go +++ b/tools/generator-go-sdk/internal/generator/stage_version.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) version(data ServiceGeneratorData) error { +func (s *ServiceGenerator) version(data GeneratorData) error { if err := s.writeToPathForResource(data.resourceOutputPath, "version.go", versionTemplater{}, data); err != nil { return fmt.Errorf("templating version: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater.go b/tools/generator-go-sdk/internal/generator/templater.go index b1fb2b5f4f8..fc926f29969 100644 --- a/tools/generator-go-sdk/internal/generator/templater.go +++ b/tools/generator-go-sdk/internal/generator/templater.go @@ -12,14 +12,14 @@ import ( ) type templaterForResource interface { - template(data ServiceGeneratorData) (*string, error) + template(GeneratorData) (*string, error) } type templaterForVersion interface { template() (*string, error) } -func (s *ServiceGenerator) writeToPathForResource(directory, filePath string, templater templaterForResource, data ServiceGeneratorData) error { +func (s *ServiceGenerator) writeToPathForResource(directory, filePath string, templater templaterForResource, data GeneratorData) error { fileContents, err := templater.template(data) if err != nil { return fmt.Errorf("templating: %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/templater_clients.go b/tools/generator-go-sdk/internal/generator/templater_clients.go index 7dc0d51fd93..5e01b90287d 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients.go @@ -7,7 +7,7 @@ var _ templaterForResource = clientsTemplater{} type clientsTemplater struct { } -func (c clientsTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c clientsTemplater) template(data GeneratorData) (*string, error) { copyrightLines, err := copyrightLinesForSource(data.source) if err != nil { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/templater_clients_autorest.go b/tools/generator-go-sdk/internal/generator/templater_clients_autorest.go index 58d29fd1c4f..ac6ab0234ae 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients_autorest.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients_autorest.go @@ -7,7 +7,7 @@ var _ templaterForResource = clientsAutoRestTemplater{} type clientsAutoRestTemplater struct { } -func (c clientsAutoRestTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c clientsAutoRestTemplater) template(data GeneratorData) (*string, error) { copyrightLines, err := copyrightLinesForSource(data.source) if err != nil { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/templater_clients_autorest_test.go b/tools/generator-go-sdk/internal/generator/templater_clients_autorest_test.go index ec0f2bd8e2d..ce0cbecba51 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients_autorest_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients_autorest_test.go @@ -8,7 +8,7 @@ import ( ) func TestTemplateAutoRestClient(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "somepackage", serviceClientName: "ExampleClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_clients_test.go b/tools/generator-go-sdk/internal/generator/templater_clients_test.go index ac327a0f46a..7567c6ac9f6 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients_test.go @@ -8,7 +8,7 @@ import ( ) func TestTemplateClient(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "somepackage", serviceClientName: "ExampleClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_constants.go b/tools/generator-go-sdk/internal/generator/templater_constants.go index eb06f8988f2..a1917c1e93e 100644 --- a/tools/generator-go-sdk/internal/generator/templater_constants.go +++ b/tools/generator-go-sdk/internal/generator/templater_constants.go @@ -14,7 +14,7 @@ type constantsTemplater struct { constantTemplateFunc func(name string, details models.SDKConstant, generateNormalizationFunction, usedInAResourceId bool) (*string, error) } -func (c constantsTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c constantsTemplater) template(data GeneratorData) (*string, error) { copyrightLines, err := copyrightLinesForSource(data.source) if err != nil { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/templater_constants_test.go b/tools/generator-go-sdk/internal/generator/templater_constants_test.go index 5e330577ba9..bd97d7773f4 100644 --- a/tools/generator-go-sdk/internal/generator/templater_constants_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_constants_test.go @@ -17,7 +17,7 @@ func TestTemplateConstantsSingle(t *testing.T) { out := fmt.Sprintf("// template for %s", name) return &out, nil }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ constants: map[string]models.SDKConstant{ "first": { Type: models.StringSDKConstantType, @@ -50,7 +50,7 @@ func TestTemplateConstantsMultiple(t *testing.T) { out := fmt.Sprintf("// template for %s", name) return &out, nil }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ constants: map[string]models.SDKConstant{ "third": {}, "first": {}, diff --git a/tools/generator-go-sdk/internal/generator/templater_id_parser.go b/tools/generator-go-sdk/internal/generator/templater_id_parser.go index fd6acfec53a..4622d6da18b 100644 --- a/tools/generator-go-sdk/internal/generator/templater_id_parser.go +++ b/tools/generator-go-sdk/internal/generator/templater_id_parser.go @@ -18,7 +18,7 @@ type resourceIdTemplater struct { constantDetails map[string]models.SDKConstant } -func (r resourceIdTemplater) template(data ServiceGeneratorData) (*string, error) { +func (r resourceIdTemplater) template(data GeneratorData) (*string, error) { copyrightLines, err := copyrightLinesForSource(data.source) if err != nil { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) @@ -33,6 +33,11 @@ func (r resourceIdTemplater) template(data ServiceGeneratorData) (*string, error return nil, fmt.Errorf("generating methods: %+v", err) } + registerId := "" + if !data.isDataPlane { + registerId = r.registerId() + } + out := fmt.Sprintf(`package %[1]s import ( "fmt" @@ -48,7 +53,7 @@ import ( %[3]s %[4]s %[5]s -`, data.packageName, *copyrightLines, r.registerId(), *structBody, *methods) +`, data.packageName, *copyrightLines, registerId, *structBody, *methods) return &out, nil } @@ -297,7 +302,7 @@ func %[1]s(input string) (*%[2]s, error) { } id := %[2]s{} - if err := id.FromParseResult(*parsed); err != nil { + if err = id.FromParseResult(*parsed); err != nil { return nil, err } diff --git a/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go b/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go index b82a367983f..c08cbbb08a4 100644 --- a/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go @@ -19,7 +19,7 @@ func TestTemplateIdParserBasic(t *testing.T) { models.NewSubscriptionIDResourceIDSegment("subscriptionId"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ resourceIds: map[string]models.ResourceID{ "empty": { CommonIDAlias: stringPointer("basic"), @@ -167,7 +167,7 @@ func TestTemplateIdParserConstantsOnly(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ resourceIds: map[string]models.ResourceID{ "empty": { CommonIDAlias: stringPointer("thing"), diff --git a/tools/generator-go-sdk/internal/generator/templater_id_parser_tests.go b/tools/generator-go-sdk/internal/generator/templater_id_parser_tests.go index c58556e8782..4d1b1f01f57 100644 --- a/tools/generator-go-sdk/internal/generator/templater_id_parser_tests.go +++ b/tools/generator-go-sdk/internal/generator/templater_id_parser_tests.go @@ -18,7 +18,7 @@ type resourceIdTestsTemplater struct { constantDetails map[string]models.SDKConstant } -func (i resourceIdTestsTemplater) template(data ServiceGeneratorData) (*string, error) { +func (i resourceIdTestsTemplater) template(data GeneratorData) (*string, error) { res, err := i.generateTests(data.packageName, data.source) if err != nil { return nil, fmt.Errorf("while generating parser tests: %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client.go b/tools/generator-go-sdk/internal/generator/templater_meta_client.go index ed0454ab6a8..1fa156c3b33 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client.go @@ -13,6 +13,7 @@ type metaClientTemplater struct { apiVersion string resources map[string]models.APIResource source models.SourceDataOrigin + sourceType models.SourceDataType } func (m metaClientTemplater) template() (*string, error) { @@ -34,7 +35,7 @@ func (m metaClientTemplater) template() (*string, error) { for _, resourceName := range resourceNames { variableName := fmt.Sprintf("%s%sClient", strings.ToLower(string(resourceName[0])), resourceName[1:]) - imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/resource-manager/%s/%s/%s"`, strings.ToLower(m.serviceName), m.apiVersion, strings.ToLower(resourceName))) + imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s/%s/%s"`, m.sourceType, strings.ToLower(m.serviceName), m.apiVersion, strings.ToLower(resourceName))) fields = append(fields, fmt.Sprintf("%[1]s *%[2]s.%[1]sClient", resourceName, strings.ToLower(resourceName))) clientInitializationTemplate := fmt.Sprintf(`%[1]s, err := %[2]s.New%[3]sClientWithBaseURI(sdkApi) if err != nil { diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go b/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go index 8e0938dfdac..d5d07aa3553 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go @@ -13,6 +13,7 @@ type metaClientAutorestTemplater struct { apiVersion string resources map[string]models.APIResource source models.SourceDataOrigin + sourceType models.SourceDataType } func (m metaClientAutorestTemplater) template() (*string, error) { @@ -34,7 +35,7 @@ func (m metaClientAutorestTemplater) template() (*string, error) { for _, resourceName := range resourceNames { variableName := fmt.Sprintf("%s%sClient", strings.ToLower(string(resourceName[0])), resourceName[1:]) - imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/resource-manager/%s/%s/%s"`, strings.ToLower(m.serviceName), m.apiVersion, strings.ToLower(resourceName))) + imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s/%s/%s"`, m.sourceType, strings.ToLower(m.serviceName), m.apiVersion, strings.ToLower(resourceName))) fields = append(fields, fmt.Sprintf("%[1]s *%[2]s.%[1]sClient", resourceName, strings.ToLower(resourceName))) clientInitializationTemplate := fmt.Sprintf(` %[1]s := %[2]s.New%[3]sClientWithBaseURI(endpoint) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index a65405f51ca..97b463781e3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -18,7 +18,7 @@ type methodsPandoraTemplater struct { constants map[string]models.SDKConstant } -func (c methodsPandoraTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) template(data GeneratorData) (*string, error) { methods, err := c.methods(data) if err != nil { return nil, fmt.Errorf("building methods: %+v", err) @@ -29,6 +29,11 @@ func (c methodsPandoraTemplater) template(data ServiceGeneratorData) (*string, e return nil, fmt.Errorf("retrieving copyright lines: %+v", err) } + commonTypesInclude := "" + if data.commonTypesIncludePath != nil { + commonTypesInclude = fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s"`, data.sourceType, *data.commonTypesIncludePath) + } + template := fmt.Sprintf(`package %[1]s import ( @@ -42,16 +47,17 @@ import ( "github.com/hashicorp/go-azure-sdk/sdk/client/pollers" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" "github.com/hashicorp/go-azure-sdk/sdk/odata" + %[4]s ) %[2]s %[3]s -`, data.packageName, *copyrightLines, *methods) +`, data.packageName, *copyrightLines, *methods, commonTypesInclude) return &template, nil } -func (c methodsPandoraTemplater) methods(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) methods(data GeneratorData) (*string, error) { switch strings.ToUpper(c.operation.Method) { case "DELETE": if c.operation.LongRunning { @@ -129,7 +135,7 @@ func (c methodsPandoraTemplater) methods(data ServiceGeneratorData) (*string, er return nil, fmt.Errorf("unsupported HTTP Method %q", c.operation.Method) } -func (c methodsPandoraTemplater) immediateOperationTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) immediateOperationTemplate(data GeneratorData) (*string, error) { methodArguments, err := c.argumentsTemplateForMethod(data) if err != nil { return nil, fmt.Errorf("building arguments for immediate operation: %+v", err) @@ -191,7 +197,7 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, return &templated, nil } -func (c methodsPandoraTemplater) longRunningOperationTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) longRunningOperationTemplate(data GeneratorData) (*string, error) { methodArguments, err := c.argumentsTemplateForMethod(data) if err != nil { return nil, fmt.Errorf("building arguments for long running template: %+v", err) @@ -272,7 +278,7 @@ func (c %[1]s) %[2]sThenPoll(ctx context.Context %[3]s) error { return &templated, nil } -func (c methodsPandoraTemplater) listOperationTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) listOperationTemplate(data GeneratorData) (*string, error) { methodArguments, err := c.argumentsTemplateForMethod(data) if err != nil { return nil, fmt.Errorf("building arguments for list operation: %+v", err) @@ -295,7 +301,7 @@ func (c methodsPandoraTemplater) listOperationTemplate(data ServiceGeneratorData if err != nil { return nil, fmt.Errorf("building options struct: %+v", err) } - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -450,7 +456,7 @@ func (c methodsPandoraTemplater) requestArgumentsTemplate() string { return fmt.Sprintf(", %s", strings.Join(args, ", ")) } -func (c methodsPandoraTemplater) argumentsTemplateForMethod(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) argumentsTemplateForMethod(data GeneratorData) (*string, error) { arguments := make([]string, 0) if c.operation.ResourceIDName != nil { idName := *c.operation.ResourceIDName @@ -465,7 +471,7 @@ func (c methodsPandoraTemplater) argumentsTemplateForMethod(data ServiceGenerato arguments = append(arguments, fmt.Sprintf("id %s", idName)) } if c.operation.RequestObject != nil { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining type name for request object: %+v", err) } @@ -566,7 +572,7 @@ func (c methodsPandoraTemplater) marshalerTemplate() (*string, error) { return &output, nil } -func (c methodsPandoraTemplater) unmarshalerTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*string, error) { var output string if c.operation.LongRunning { @@ -577,7 +583,7 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data ServiceGeneratorData) } if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } @@ -655,7 +661,7 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data ServiceGeneratorData) result.Model = &model `, discriminatedTypeParentName) } else { - responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } @@ -682,11 +688,11 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data ServiceGeneratorData) return &output, nil } -func (c methodsPandoraTemplater) responseStructTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) responseStructTemplate(data GeneratorData) (*string, error) { model := "" typeName := "" if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } @@ -733,7 +739,7 @@ type %[1]s struct { return &output, nil } -func (c methodsPandoraTemplater) optionsStruct(data ServiceGeneratorData) (*string, error) { +func (c methodsPandoraTemplater) optionsStruct(data GeneratorData) (*string, error) { if len(c.operation.Options) == 0 { out := "" return &out, nil diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go b/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go index 7b78941dc01..8b2702d183d 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go @@ -21,7 +21,7 @@ type methodsAutoRestTemplater struct { constants map[string]models.SDKConstant } -func (c methodsAutoRestTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) template(data GeneratorData) (*string, error) { methods, err := c.methods(data) if err != nil { return nil, fmt.Errorf("building methods: %+v", err) @@ -56,7 +56,7 @@ import ( return &template, nil } -func (c methodsAutoRestTemplater) methods(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) methods(data GeneratorData) (*string, error) { // NOTE: most of this logic is sanity checking, but should be within the API and it's validators too // that could be a separate validation tool, but this would be most useful as unit tests @@ -137,7 +137,7 @@ func (c methodsAutoRestTemplater) methods(data ServiceGeneratorData) (*string, e return nil, fmt.Errorf("unsupported HTTP Method %q", c.operation.Method) } -func (c methodsAutoRestTemplater) immediateOperationTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) immediateOperationTemplate(data GeneratorData) (*string, error) { responseStructName, err := c.responseStructName(data) if err != nil { return nil, err @@ -198,7 +198,7 @@ func (c %[1]s) %[2]s(ctx context.Context %[4]s) (result %[10]s, err error) { return &templated, nil } -func (c methodsAutoRestTemplater) listOperationTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) listOperationTemplate(data GeneratorData) (*string, error) { responseStructName, err := c.responseStructName(data) if err != nil { return nil, err @@ -225,7 +225,7 @@ func (c methodsAutoRestTemplater) listOperationTemplate(data ServiceGeneratorDat if err != nil { return nil, fmt.Errorf("building responder template: %+v", err) } - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -351,7 +351,7 @@ func (c %[1]s) %[2]sComplete(ctx context.Context%[3]s) (result %[2]sCompleteResu return &templated, nil } -func (c methodsAutoRestTemplater) longRunningOperationTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) longRunningOperationTemplate(data GeneratorData) (*string, error) { responseStructName, err := c.responseStructName(data) if err != nil { return nil, err @@ -435,7 +435,7 @@ func (c methodsAutoRestTemplater) argumentsTemplate() string { return fmt.Sprintf(", %s", strings.Join(args, ", ")) } -func (c methodsAutoRestTemplater) argumentsTemplateForMethod(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) argumentsTemplateForMethod(data GeneratorData) (*string, error) { arguments := make([]string, 0) if c.operation.ResourceIDName != nil { idName := *c.operation.ResourceIDName @@ -450,7 +450,7 @@ func (c methodsAutoRestTemplater) argumentsTemplateForMethod(data ServiceGenerat arguments = append(arguments, fmt.Sprintf("id %s", idName)) } if c.operation.RequestObject != nil { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining type name for request object: %+v", err) } @@ -467,7 +467,7 @@ func (c methodsAutoRestTemplater) argumentsTemplateForMethod(data ServiceGenerat return &out, nil } -func (c methodsAutoRestTemplater) preparerTemplate(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) preparerTemplate(data GeneratorData) (*string, error) { arguments, err := c.argumentsTemplateForMethod(data) if err != nil { return nil, fmt.Errorf("building arguments for preparer template: %+v", err) @@ -574,7 +574,7 @@ func (c %[1]s) preparerFor%[2]sWithNextLink(ctx context.Context, nextLink string return &output, nil } -func (c methodsAutoRestTemplater) responderTemplate(responseStructName string, data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) responderTemplate(responseStructName string, data GeneratorData) (*string, error) { expectedStatusCodes := make([]string, 0) for _, statusCodeInt := range c.operation.ExpectedStatusCodes { statusCode := golangConstantForStatusCode(statusCodeInt) @@ -600,7 +600,7 @@ func (c methodsAutoRestTemplater) responderTemplate(responseStructName string, d steps = append(steps, "autorest.ByClosing()") if c.operation.FieldContainingPaginationDetails != nil && discriminatedType == "" { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -652,7 +652,7 @@ func (c %[1]s) responderFor%[2]s(resp *http.Response) (result %[6]s, err error) } if discriminatedType != "" && c.operation.FieldContainingPaginationDetails != nil { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -751,7 +751,7 @@ func (c %[1]s) responderFor%[2]s(resp *http.Response) (result %[5]s, err error) return &output, nil } -func (c methodsAutoRestTemplater) responseStructName(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) responseStructName(data GeneratorData) (*string, error) { responseStructName := fmt.Sprintf("%[1]sOperationResponse", c.operationName) if _, hasExistingModel := data.models[responseStructName]; hasExistingModel { responseStructName = fmt.Sprintf("%[1]sOperationApiResponse", c.operationName) @@ -767,7 +767,7 @@ func (c methodsAutoRestTemplater) responseStructTemplate(responseStructName stri model := "" typeName := "" if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil) + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } @@ -835,7 +835,7 @@ type %[1]s struct { return &output, nil } -func (c methodsAutoRestTemplater) senderLongRunningOperationTemplate(data ServiceGeneratorData) string { +func (c methodsAutoRestTemplater) senderLongRunningOperationTemplate(data GeneratorData) string { return fmt.Sprintf(` // senderFor%[2]s sends the %[2]s request. The method will close the // http.Response Body if it receives an error. @@ -852,7 +852,7 @@ func (c %[1]s) senderFor%[2]s(ctx context.Context, req *http.Request) (future %[ `, data.serviceClientName, c.operationName) } -func (c methodsAutoRestTemplater) optionsStruct(data ServiceGeneratorData) (*string, error) { +func (c methodsAutoRestTemplater) optionsStruct(data GeneratorData) (*string, error) { if len(c.operation.Options) == 0 { out := "" return &out, nil diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_autorest_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_autorest_test.go index ae8a9ea2e47..0b8c176e30d 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_autorest_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_autorest_test.go @@ -10,7 +10,7 @@ import ( ) func TestTemplateMethodsAutoRestLRODelete(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -41,7 +41,7 @@ func (c pandaClient) senderForDelete(ctx context.Context, req *http.Request) (fu } func TestTemplateMethodsAutoRestLROCreate(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -73,7 +73,7 @@ func (c pandaClient) senderForCreate(ctx context.Context, req *http.Request) (fu } func TestTemplateMethodAutoRestDiscriminatedTypeResponder(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -124,7 +124,7 @@ func TestTemplateMethodAutoRestDiscriminatedTypeResponder(t *testing.T) { } func TestTemplateMethodsAutoRestBaseTypePredicates(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -244,7 +244,7 @@ func (c pandaClient) ListComplete(ctx context.Context, id PandaPop) (result List } func TestTemplateMethodsAutoRestNonBaseTypePredicates(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go index f8c5b1c5516..bf514e4c2ca 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go @@ -19,7 +19,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsImplementation_Get(t *tes // In this instance the Discriminated Implementation Type should be used in the Response and the // Discriminated Parent's `unmarshal` function shouldn't be called. - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -121,7 +121,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsParent_Get(t *testing.T) // This test covers the Response Object being a Discriminated Parent Type for a GET operation // In this instance the `unmarshal` function for the Parent Type should be called as a part // of the Response. - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -228,7 +228,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsImplementation_List(t *te // for a GET Operation. In this instance a slice of the Discriminated Implementation Type // should be output - and the Discriminated Parent's `unmarshal` function shouldn't be called. - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -384,7 +384,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsParent_List(t *testing.T) // for a GET Operation. In this instance a slice of the Discriminated Parent Type // should be output - and the (Discriminated Parent's) `unmarshal` function should be called. - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go index 7a11a064bf5..0a2a0249a70 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go @@ -10,7 +10,7 @@ import ( ) func TestTemplateMethodsLROCreate(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -103,7 +103,7 @@ func (c pandaClient) CreateThenPoll(ctx context.Context , id PandaPop, input str } func TestTemplateMethodsLROReboot(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -194,7 +194,7 @@ func TestTemplateMethodsLRODoesNotCallUnmarshal(t *testing.T) { // As such this test asserts that we don't call `Unmarshal` prior to creating the LRO // Poller - since (for the moment) consumers will need to decide when `Unmarshal` should // be called on the LRO Result, if needed at all. - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_test.go index be691cba4dd..9773931ccfc 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_test.go @@ -14,7 +14,7 @@ import ( // TODO: split the tests out into sub-groupings & add more coverage func TestTemplateMethodsGet(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -87,7 +87,7 @@ func (c pandaClient) Get(ctx context.Context , id PandaPop) (result GetOperation } func TestTemplateMethodsGetAsTextPowerShell(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -165,7 +165,7 @@ func (c pandaClient) Get(ctx context.Context , id PandaPop) (result GetOperation // As such these tests (whilst similar) cover the two different code paths. func TestTemplateMethodsListWithDiscriminatedType(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -333,7 +333,7 @@ func (c pandaClient) ListCompleteMatchingPredicate(ctx context.Context, id Panda } func TestTemplateMethodsListWithSimpleType(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -454,7 +454,7 @@ func (c pandaClient) ListComplete(ctx context.Context, id PandaPop) (result List } func TestTemplateMethodsListWithObject(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 5a37aebee76..2147884e529 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -20,7 +20,7 @@ type modelsTemplater struct { model models.SDKModel } -func (c modelsTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) template(data GeneratorData) (*string, error) { copyrightLines, err := copyrightLinesForSource(data.source) if err != nil { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) @@ -58,7 +58,7 @@ import ( return &template, nil } -func (c modelsTemplater) structCode(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) structCode(data GeneratorData) (*string, error) { // if this is an Abstract/Type Hint, we output an Interface with a manual unmarshal func that gets called wherever it's used if c.model.FieldNameContainingDiscriminatedValue != nil && c.model.ParentTypeName == nil { out := fmt.Sprintf(` @@ -87,7 +87,7 @@ type Raw%[1]sImpl struct { for _, fieldName := range fields { fieldDetails := c.model.Fields[fieldName] fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil) + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) } @@ -138,7 +138,7 @@ type Raw%[1]sImpl struct { for _, fieldName := range parentFields { fieldDetails := parent.Fields[fieldName] fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil) + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) } @@ -169,7 +169,7 @@ type %[1]s struct { return &out, nil } -func (c modelsTemplater) methods(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) methods(data GeneratorData) (*string, error) { code := make([]string, 0) dateFunctions, err := c.codeForDateFunctions(data) @@ -196,7 +196,7 @@ func (c modelsTemplater) methods(data ServiceGeneratorData) (*string, error) { return &output, nil } -func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDetails models.SDKField, data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDetails models.SDKField, data GeneratorData) (*string, error) { jsonDetails := fieldDetails.JsonName isOptional := false @@ -239,7 +239,7 @@ func (c modelsTemplater) dateFormatString(input models.SDKDateFormat) string { } } -func (c modelsTemplater) codeForDateFunctions(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) codeForDateFunctions(data GeneratorData) (*string, error) { fieldsRequiringDateFunctions := make([]string, 0) // parent models are output as interfaces with no fields - so we can skip these // since the inherited models output the fields from their parents, the methods are output there @@ -332,7 +332,7 @@ func (c modelsTemplater) dateFunctionForField(fieldName string, fieldDetails mod return &out, nil } -func (c modelsTemplater) codeForMarshalFunctions(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, error) { output := "" if c.model.DiscriminatedValue != nil { @@ -392,7 +392,7 @@ func (s %[1]s) MarshalJSON() ([]byte, error) { return &output, nil } -func (c modelsTemplater) codeForUnmarshalFunctions(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) codeForUnmarshalFunctions(data GeneratorData) (*string, error) { unmarshalFunction, err := c.codeForUnmarshalStructFunction(data) if err != nil { return nil, fmt.Errorf("generating code for unmarshal struct function: %+v", err) @@ -410,7 +410,7 @@ func (c modelsTemplater) codeForUnmarshalFunctions(data ServiceGeneratorData) (* return &output, nil } -func (c modelsTemplater) codeForUnmarshalParentFunction(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) codeForUnmarshalParentFunction(data GeneratorData) (*string, error) { // if this is a Discriminated Type (e.g. Parent) then we need to generate a unmarshal{Name}Implementations // function which can be used in any usages lines := make([]string, 0) @@ -489,7 +489,7 @@ func unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { return &output, nil } -func (c modelsTemplater) codeForUnmarshalStructFunction(data ServiceGeneratorData) (*string, error) { +func (c modelsTemplater) codeForUnmarshalStructFunction(data GeneratorData) (*string, error) { // this is a parent, therefore there'll be no struct fields to check here if c.model.IsDiscriminatedParentType() { out := "" @@ -689,7 +689,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) // recurseParentModels walks the models hierarchy to find the parentName and field details of the model for disciminated types // This is a temporary measure until we update the swagger importer to connect the model fields inheritance for multiple parents. // Tracked at: https://github.com/hashicorp/pandora/issues/1235 -func (c modelsTemplater) recurseParentModels(data ServiceGeneratorData, model string, typeHint string) (*models.SDKField, *string, error) { +func (c modelsTemplater) recurseParentModels(data GeneratorData, model string, typeHint string) (*models.SDKField, *string, error) { parentModel, ok := data.models[model] if !ok { return nil, nil, fmt.Errorf("the parent model %q for model %q was not found", model, c.name) diff --git a/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go b/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go index ca8f6df4101..3ba3d203d08 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go @@ -26,7 +26,7 @@ func TestModelTemplaterWithOptionalFloatConstant(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -93,7 +93,7 @@ func TestModelTemplaterWithOptionalIntegerConstant(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -161,7 +161,7 @@ func TestModelTemplaterWithOptionalStringConstant(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -229,7 +229,7 @@ func TestModelTemplaterWithRequiredFloatConstant(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -296,7 +296,7 @@ func TestModelTemplaterWithRequiredIntegerConstant(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -363,7 +363,7 @@ func TestModelTemplaterWithRequiredStringConstant(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -437,7 +437,7 @@ func TestModelTemplaterWithOptionalListOfFloatConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -513,7 +513,7 @@ func TestModelTemplaterWithOptionalListOfIntegerConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -586,7 +586,7 @@ func TestModelTemplaterWithOptionalListOfStringConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -660,7 +660,7 @@ func TestModelTemplaterWithRequiredListOfFloatConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -733,7 +733,7 @@ func TestModelTemplaterWithRequiredListOfIntegerConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -809,7 +809,7 @@ func TestModelTemplaterWithRequiredListOfStringConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -883,7 +883,7 @@ func TestModelTemplaterWithOptionalMapOfFloatConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -959,7 +959,7 @@ func TestModelTemplaterWithOptionalMapOfIntegerConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -1032,7 +1032,7 @@ func TestModelTemplaterWithOptionalMapOfStringConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -1106,7 +1106,7 @@ func TestModelTemplaterWithRequiredMapOfFloatConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -1182,7 +1182,7 @@ func TestModelTemplaterWithRequiredMapOfIntegerConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -1255,7 +1255,7 @@ func TestModelTemplaterWithRequiredMapOfStringConstants(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index dcd7d480502..ebc9c0a5692 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -26,7 +26,7 @@ func TestTemplaterModelsParent(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "ModeOfTransit": { @@ -151,7 +151,7 @@ func TestTemplaterModelsImplementation(t *testing.T) { ParentTypeName: stringPointer("ModeOfTransit"), DiscriminatedValue: stringPointer("train"), }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "ModeOfTransit": { @@ -261,7 +261,7 @@ func TestTemplaterModelsFieldImplementation(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "ModeOfTransit": { @@ -356,7 +356,7 @@ func TestTemplaterModelsImplementationInheritedFromParentType(t *testing.T) { FieldNameContainingDiscriminatedValue: stringPointer("Type"), ParentTypeName: stringPointer("First"), }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "First": { diff --git a/tools/generator-go-sdk/internal/generator/templater_models_test.go b/tools/generator-go-sdk/internal/generator/templater_models_test.go index 05df5867a35..562d64905e0 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_test.go @@ -32,7 +32,7 @@ func TestModelTemplaterSimple(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -105,7 +105,7 @@ func TestModelTemplaterWithDate(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { @@ -199,7 +199,7 @@ func TestModelTemplaterWithOptionalObject(t *testing.T) { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "somepackage", models: map[string]models.SDKModel{ "Basic": { diff --git a/tools/generator-go-sdk/internal/generator/templater_predicates.go b/tools/generator-go-sdk/internal/generator/templater_predicates.go index 13f3251c25b..c2c242f05c0 100644 --- a/tools/generator-go-sdk/internal/generator/templater_predicates.go +++ b/tools/generator-go-sdk/internal/generator/templater_predicates.go @@ -16,7 +16,7 @@ type predicateTemplater struct { models map[string]models.SDKModel } -func (p predicateTemplater) template(data ServiceGeneratorData) (*string, error) { +func (p predicateTemplater) template(data GeneratorData) (*string, error) { output := make([]string, 0) for _, modelName := range p.sortedModelNames { model := data.models[modelName] @@ -95,7 +95,7 @@ func (p predicateTemplater) templateForModel(predicateStructName string, name st for _, fieldName := range fieldNames { fieldVal := model.Fields[fieldName] - typeInfo, err := helpers.GolangTypeForSDKObjectDefinition(fieldVal.ObjectDefinition, nil) + typeInfo, err := helpers.GolangTypeForSDKObjectDefinition(fieldVal.ObjectDefinition, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining type information for field %q in model %q with info %q: %+v", fieldName, name, string(fieldVal.ObjectDefinition.Type), err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_readme.go b/tools/generator-go-sdk/internal/generator/templater_readme.go index 92aec08ec9a..ca3831a735e 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme.go @@ -17,7 +17,7 @@ type readmeTemplater struct { operations map[string]models.SDKOperation } -func (r readmeTemplater) template(data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) template(data GeneratorData) (*string, error) { summary := r.packageSummary(data) clientInit := r.clientInitialization(data.packageName, data.serviceClientName) examples, err := r.exampleUsages(data) @@ -33,9 +33,9 @@ func (r readmeTemplater) template(data ServiceGeneratorData) (*string, error) { return &out, nil } -func (r readmeTemplater) packageSummary(data ServiceGeneratorData) string { +func (r readmeTemplater) packageSummary(data GeneratorData) string { importLines := []string{ - fmt.Sprintf(`import "github.com/hashicorp/go-azure-sdk/resource-manager/%[1]s/%[2]s/%[3]s"`, data.servicePackageName, data.apiVersion, data.packageName), + fmt.Sprintf(`import "github.com/hashicorp/go-azure-sdk/%[1]s/%[2]s/%[3]s/%[4]s"`, data.sourceType, data.servicePackageName, data.apiVersion, data.packageName), } containsCommonId := false for _, resourceId := range data.resourceIds { @@ -50,18 +50,18 @@ func (r readmeTemplater) packageSummary(data ServiceGeneratorData) string { sort.Strings(importLines) return fmt.Sprintf(` -## 'github.com/hashicorp/go-azure-sdk/resource-manager/%[1]s/%[2]s/%[3]s' Documentation +## 'github.com/hashicorp/go-azure-sdk/%[1]s/%[2]s/%[3]s/%[4]s' Documentation -The '%[3]s' SDK allows for interaction with the Azure Resource Manager Service '%[1]s' (API Version '%[2]s'). +The '%[4]s' SDK allows for interaction with the Azure Resource Manager Service '%[2]s' (API Version '%[3]s'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). ### Import Path '''go -%[4]s +%[5]s ''' -`, data.servicePackageName, data.apiVersion, data.packageName, strings.Join(importLines, "\n")) +`, data.sourceType, data.servicePackageName, data.apiVersion, data.packageName, strings.Join(importLines, "\n")) } func (r readmeTemplater) clientInitialization(packageName, clientName string) string { @@ -75,7 +75,7 @@ client.Client.Authorizer = authorizer `, packageName, clientName) } -func (r readmeTemplater) exampleUsages(data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) exampleUsages(data GeneratorData) (*string, error) { examples := make([]string, 0) for _, operationName := range r.sortedOperationNames { @@ -96,7 +96,7 @@ func (r readmeTemplater) exampleUsages(data ServiceGeneratorData) (*string, erro return &out, nil } -func (r readmeTemplater) exampleUsageForOperation(packageName, clientName, operationName string, operation models.SDKOperation, data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) exampleUsageForOperation(packageName, clientName, operationName string, operation models.SDKOperation, data GeneratorData) (*string, error) { if operation.FieldContainingPaginationDetails != nil { return r.exampleUsageForListOperation(packageName, clientName, operationName, operation, data) } @@ -108,7 +108,7 @@ func (r readmeTemplater) exampleUsageForOperation(packageName, clientName, opera return r.exampleUsageForRegularOperation(packageName, clientName, operationName, operation, data) } -func (r readmeTemplater) resourceIdInitialization(operation models.SDKOperation, data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) resourceIdInitialization(operation models.SDKOperation, data GeneratorData) (*string, error) { resourceId, ok := data.resourceIds[*operation.ResourceIDName] if !ok { return nil, fmt.Errorf("resource id %q was not found", *operation.ResourceIDName) @@ -132,7 +132,7 @@ func (r readmeTemplater) resourceIdInitialization(operation models.SDKOperation, return &out, nil } -func (r readmeTemplater) exampleUsageForRegularOperation(packageName, clientName, operationName string, operation models.SDKOperation, data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) exampleUsageForRegularOperation(packageName, clientName, operationName string, operation models.SDKOperation, data GeneratorData) (*string, error) { lines := make([]string, 0) methodArgs := []string{ @@ -157,7 +157,7 @@ payload := %[1]s.%[2]s{ `, packageName, *operation.RequestObject.ReferenceName)) } else { // for simplicities sake - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for request object: %+v", err) } @@ -188,7 +188,7 @@ if model := read.Model; model != nil { return &out, nil } -func (r readmeTemplater) exampleUsageForListOperation(packageName, clientName, operationName string, operation models.SDKOperation, data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) exampleUsageForListOperation(packageName, clientName, operationName string, operation models.SDKOperation, data GeneratorData) (*string, error) { lines := make([]string, 0) methodArgs := []string{ @@ -213,7 +213,7 @@ payload := %[1]s.%[2]s{ `, packageName, *operation.RequestObject.ReferenceName)) } else { // for simplicities sake - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for request object: %+v", err) } @@ -245,7 +245,7 @@ for _, item := range items { return &out, nil } -func (r readmeTemplater) exampleUsageForLongRunningOperation(packageName, clientName, operationName string, operation models.SDKOperation, data ServiceGeneratorData) (*string, error) { +func (r readmeTemplater) exampleUsageForLongRunningOperation(packageName, clientName, operationName string, operation models.SDKOperation, data GeneratorData) (*string, error) { lines := make([]string, 0) methodArgs := []string{ @@ -270,7 +270,7 @@ payload := %[1]s.%[2]s{ `, packageName, *operation.RequestObject.ReferenceName)) } else { // for simplicities sake - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for request object: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_readme_test.go b/tools/generator-go-sdk/internal/generator/templater_readme_test.go index 0e5950483b4..242ba8888af 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme_test.go @@ -36,7 +36,7 @@ client.Client.Authorizer = authorizer actual, err := readmeTemplater{ sortedOperationNames: []string{}, operations: map[string]models.SDKOperation{}, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -93,7 +93,7 @@ if model := read.Model; model != nil { ResourceIDName: stringPointer("Disk"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -159,7 +159,7 @@ if model := read.Model; model != nil { ResourceIDName: stringPointer("Disk"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -232,7 +232,7 @@ if model := read.Model; model != nil { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -319,7 +319,7 @@ if model := read.Model; model != nil { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -399,7 +399,7 @@ if model := read.Model; model != nil { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -463,7 +463,7 @@ if model := read.Model; model != nil { ResourceIDName: nil, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -528,7 +528,7 @@ if model := read.Model; model != nil { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -588,7 +588,7 @@ for _, item := range items { ResourceIDName: stringPointer("Disk"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -656,7 +656,7 @@ for _, item := range items { ResourceIDName: stringPointer("Disk"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -731,7 +731,7 @@ for _, item := range items { FieldContainingPaginationDetails: stringPointer("SomeField"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -820,7 +820,7 @@ for _, item := range items { FieldContainingPaginationDetails: stringPointer("SomeField"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -902,7 +902,7 @@ for _, item := range items { FieldContainingPaginationDetails: stringPointer("SomeField"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -975,7 +975,7 @@ for _, item := range items { FieldContainingPaginationDetails: stringPointer("SomeField"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1030,7 +1030,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id); err != nil { ResourceIDName: stringPointer("Disk"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1093,7 +1093,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id); err != nil { ResourceIDName: stringPointer("Disk"), }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1163,7 +1163,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id, payload); err != nil { }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1247,7 +1247,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id, payload, disks.DefaultSomeLong }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1323,7 +1323,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id, disks.DefaultSomeLongRunningOp }, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1385,7 +1385,7 @@ if err := client.SomeLongRunningThenPoll(ctx); err != nil { ResourceIDName: nil, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", @@ -1461,7 +1461,7 @@ if model := read.Model; model != nil { ResourceIDName: nil, }, }, - }.template(ServiceGeneratorData{ + }.template(GeneratorData{ packageName: "disks", apiVersion: "2022-02-01", servicePackageName: "compute", diff --git a/tools/generator-go-sdk/internal/generator/templater_version.go b/tools/generator-go-sdk/internal/generator/templater_version.go index d4e1d190f44..ce35d8e2145 100644 --- a/tools/generator-go-sdk/internal/generator/templater_version.go +++ b/tools/generator-go-sdk/internal/generator/templater_version.go @@ -7,7 +7,7 @@ var _ templaterForResource = versionTemplater{} type versionTemplater struct { } -func (c versionTemplater) template(data ServiceGeneratorData) (*string, error) { +func (c versionTemplater) template(data GeneratorData) (*string, error) { copyrightLines, err := copyrightLinesForSource(data.source) if err != nil { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/templater_version_test.go b/tools/generator-go-sdk/internal/generator/templater_version_test.go index 66ab1dae9ae..666a9b04f20 100644 --- a/tools/generator-go-sdk/internal/generator/templater_version_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_version_test.go @@ -8,7 +8,7 @@ import ( ) func TestTemplateVersion(t *testing.T) { - input := ServiceGeneratorData{ + input := GeneratorData{ packageName: "somepackage", apiVersion: "2022-02-01", source: AccTestLicenceType, diff --git a/tools/generator-terraform/internal/generator/mappings/assignment_direct.go b/tools/generator-terraform/internal/generator/mappings/assignment_direct.go index 99cf004cdf4..4e7179967f1 100644 --- a/tools/generator-terraform/internal/generator/mappings/assignment_direct.go +++ b/tools/generator-terraform/internal/generator/mappings/assignment_direct.go @@ -248,7 +248,7 @@ output.%[1]s = &%[4]s return nil, fmt.Errorf("expected a DirectAssignment between %q and %q but got %q", string(schemaField.ObjectDefinition.NestedObject.Type), string(v), string(sdkField.ObjectDefinition.NestedItem.Type)) } - variableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName) + variableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type for Sdk Model %q / Field %q Object Definition: %+v", mapping.DirectAssignment.SDKModelName, mapping.DirectAssignment.SDKFieldName, err) } @@ -293,12 +293,12 @@ output.%[1]s = &%[3]s } func (d directAssignmentLine) schemaToSdkMappingBetweenListOfReferenceFields(mapping models.TerraformDirectAssignmentFieldMappingDefinition, schemaField models.TerraformSchemaField, sdkField models.SDKField, apiResourcePackageName string) (*string, error) { - listVariableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName) + listVariableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type for Sdk Model %q / Field %q Object Definition: %+v", mapping.DirectAssignment.SDKModelName, mapping.DirectAssignment.SDKFieldName, err) } - listItemVariableReferenceType, err := helpers.GolangTypeForSDKObjectDefinition(*sdkField.ObjectDefinition.NestedItem, &apiResourcePackageName) + listItemVariableReferenceType, err := helpers.GolangTypeForSDKObjectDefinition(*sdkField.ObjectDefinition.NestedItem, &apiResourcePackageName, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type for Sdk Model %q / Field %q Nested Object Definition: %+v", mapping.DirectAssignment.SDKModelName, mapping.DirectAssignment.SDKFieldName, err) } diff --git a/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go b/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go index 699a5c2ddac..ddbd7096e21 100644 --- a/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go +++ b/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go @@ -29,7 +29,7 @@ func (m modelToModelAssignmentLine) assignmentForCreateUpdateMapping(mapping mod return nil, fmt.Errorf("a ModelToModel mapping must be a Reference but got %q", string(sdkField.ObjectDefinition.Type)) } outputModelName := *sdkField.ObjectDefinition.ReferenceName - sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName) + sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type Name for SDK Field: %+v", err) } @@ -68,7 +68,7 @@ func (m modelToModelAssignmentLine) assignmentForReadMapping(mapping models.Terr } outputModelName := *sdkField.ObjectDefinition.ReferenceName - sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName) + sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type Name for SDK Field: %+v", err) } diff --git a/tools/generator-terraform/internal/generator/resource/component_create_func.go b/tools/generator-terraform/internal/generator/resource/component_create_func.go index 5eb4d0b5314..20f2ecdec58 100644 --- a/tools/generator-terraform/internal/generator/resource/component_create_func.go +++ b/tools/generator-terraform/internal/generator/resource/component_create_func.go @@ -199,7 +199,7 @@ id := %[3]s(%[4]s) func (h createFunctionComponents) payloadDefinition() (*string, error) { // NOTE: whilst Payload is _technically_ optional in the API endpoint it's not, else it // wouldn't be a Create method - createObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.createMethod.RequestObject, &h.sdkResourceNameLowered) + createObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.createMethod.RequestObject, &h.sdkResourceNameLowered, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type name for Create Request Object: %+v", err) } diff --git a/tools/generator-terraform/internal/generator/resource/component_update_func.go b/tools/generator-terraform/internal/generator/resource/component_update_func.go index 373b6557411..6c27c497d55 100644 --- a/tools/generator-terraform/internal/generator/resource/component_update_func.go +++ b/tools/generator-terraform/internal/generator/resource/component_update_func.go @@ -130,7 +130,7 @@ func (h updateFuncHelpers) modelDecode() (*string, error) { } func (h updateFuncHelpers) payloadDefinition() (*string, error) { - updateObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.updateMethod.RequestObject, &h.sdkResourceNameLowered) + updateObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.updateMethod.RequestObject, &h.sdkResourceNameLowered, nil, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type name for Update Request Object: %+v", err) } From 013a475f3ec42642c0dde19ee0c441f1cb92e1da Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 29 Jul 2024 22:13:34 +0100 Subject: [PATCH 003/117] importer-msgraph-metadata: set correct response model for list operations --- .../pipeline/task_translate_service.go | 36 +++++++++++++++---- 1 file changed, 29 insertions(+), 7 deletions(-) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index acf1c65286c..785fc24c2be 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -5,7 +5,6 @@ package pipeline import ( "fmt" - sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" @@ -97,22 +96,45 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta for _, response := range operation.Responses { if response.Type != nil && *response.Type == parser.DataTypeModel && response.ModelName != nil { - if !models.Found(*response.ModelName) { - return nil, fmt.Errorf("response model %q was not found for operation: %s", *response.ModelName, operation.Name) + modelName := *response.ModelName + + if !models.Found(modelName) { + return nil, fmt.Errorf("response model %q was not found for operation: %s", modelName, operation.Name) } - if model := models[*response.ModelName]; !model.IsValid() { - return nil, fmt.Errorf("response model %q was invalid for operation: %s", *response.ModelName, operation.Name) + model := models[modelName] + + if !model.IsValid() { + return nil, fmt.Errorf("response model %q was invalid for operation: %s", modelName, operation.Name) } else if !model.Common { responseObjectIsCommonType = false - if err := serviceModels.MergeDependants(models, *response.ModelName); err != nil { + if err := serviceModels.MergeDependants(models, modelName); err != nil { return nil, err } } + // List operations return a "CollectionResponse" object, which we are not interested in + // We want the actual underlying model, expected to be in the `value` field + if operation.Type == parser.OperationTypeList { + if value, ok := model.Fields["Value"]; ok && value != nil && *value.Type == parser.DataTypeArray && value.ModelName != nil { + responseObjectIsCommonType = true + modelName = *value.ModelName + + if !models.Found(modelName) { + return nil, fmt.Errorf("nested response model %q was not found for operation: %s", modelName, operation.Name) + } else if !model.Common { + responseObjectIsCommonType = false + + if err := serviceModels.MergeDependants(models, modelName); err != nil { + return nil, err + } + } + } + } + responseObject = &sdkModels.SDKObjectDefinition{ - ReferenceName: response.ModelName, + ReferenceName: &modelName, ReferenceNameIsCommonType: &responseObjectIsCommonType, Type: sdkModels.ReferenceSDKObjectDefinitionType, } From cf468917e33c4957276ae6c49578c9d06d2a494a Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 29 Jul 2024 22:16:52 +0100 Subject: [PATCH 004/117] data-api-repository: support `ReferenceNameIsCommonType` for object definitions --- .../repository/internal/models/object_definition.go | 4 +++- .../internal/transforms/sdk_field_object_definition.go | 9 +++++++-- 2 files changed, 10 insertions(+), 3 deletions(-) diff --git a/tools/data-api-repository/repository/internal/models/object_definition.go b/tools/data-api-repository/repository/internal/models/object_definition.go index 9f1ba91501b..d25ce9fe449 100644 --- a/tools/data-api-repository/repository/internal/models/object_definition.go +++ b/tools/data-api-repository/repository/internal/models/object_definition.go @@ -8,10 +8,12 @@ type ObjectDefinition struct { // ObjectDefinitionType defines what kind of ObjectDefinition this is, such as a Reference, String or List Type ObjectDefinitionType `json:"type"` - // NOTE: we could split this into ConstantName and ModelName in time, but not today. // ReferenceName is the name of the Constant or Model that this is a reference to ReferenceName *string `json:"referenceName"` + // ReferenceNameIsCommonType indicates whether ReferenceName refers to a common type (true) or a type that is local to the service (false) + ReferenceNameIsCommonType *bool `json:"referenceNameIsCommonType"` + // NestedItem is a nested ObjectDefinition when Type is a Dictionary or List // NOTE: that it's possible to have deeply-nested ObjectDefinitions, e.g. a List of a List of a Dictionary[String (key, fixed as a string) : Integer (value) NestedItem *ObjectDefinition `json:"nestedItem,omitempty"` diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go b/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go index 3aabb1e05cd..5aa61ffbc55 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go @@ -17,8 +17,9 @@ func mapSDKFieldObjectDefinitionFromRepository(input repositoryModels.ObjectDefi return nil, fmt.Errorf("internal-error: missing a mapping for the ObjectDefinitionType %q", string(input.Type)) } output := sdkModels.SDKObjectDefinition{ - ReferenceName: input.ReferenceName, - Type: typeVal, + ReferenceName: input.ReferenceName, + ReferenceNameIsCommonType: input.ReferenceNameIsCommonType, + Type: typeVal, } if input.NestedItem != nil { @@ -47,6 +48,10 @@ func mapSDKFieldObjectDefinitionToRepository(input sdkModels.SDKObjectDefinition output.ReferenceName = input.ReferenceName } + if input.ReferenceNameIsCommonType != nil { + output.ReferenceNameIsCommonType = input.ReferenceNameIsCommonType + } + if input.NestedItem != nil { nestedItem, err := mapSDKFieldObjectDefinitionToRepository(*input.NestedItem, knownData) if err != nil { From eb707ce63db316933e2accfe2f7d0499df36edbb Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 29 Jul 2024 22:18:06 +0100 Subject: [PATCH 005/117] generator-go-sdk: use the value of `ReferenceNameIsCommonType` instead of trying to work it out for each model --- .../internal/differ/differ_helpers.go | 2 +- .../golang_type_for_sdk_object_definition.go | 22 ++++++------------- .../internal/generator/data.go | 13 ----------- .../internal/generator/service.go | 4 ++-- .../internal/generator/templater_methods.go | 22 ++++++++++++------- .../generator/templater_methods_autorest.go | 10 ++++----- .../internal/generator/templater_models.go | 4 ++-- .../generator/templater_predicates.go | 2 +- .../internal/generator/templater_readme.go | 6 ++--- .../generator/mappings/assignment_direct.go | 6 ++--- .../mappings/assignment_model_to_model.go | 4 ++-- .../resource/component_create_func.go | 2 +- .../resource/component_update_func.go | 2 +- 13 files changed, 42 insertions(+), 57 deletions(-) diff --git a/tools/data-api-differ/internal/differ/differ_helpers.go b/tools/data-api-differ/internal/differ/differ_helpers.go index ff7e36b646c..bfb860a79c5 100644 --- a/tools/data-api-differ/internal/differ/differ_helpers.go +++ b/tools/data-api-differ/internal/differ/differ_helpers.go @@ -12,7 +12,7 @@ import ( // stringifySDKObjectDefinition returns a human readable, string version of this SDKObjectDefinition. func (d differ) stringifySDKObjectDefinition(input models.SDKObjectDefinition) (*string, error) { - return helpers.GolangTypeForSDKObjectDefinition(input, nil, nil, nil) + return helpers.GolangTypeForSDKObjectDefinition(input, nil, nil) } // stringifySDKOperationOptionObjectDefinition returns a human readable, string version of this Object Definition. diff --git a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go index 4a000b4a0f3..0ea495b912e 100644 --- a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go +++ b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go @@ -11,7 +11,7 @@ import ( ) // GolangTypeForSDKObjectDefinition returns the Golang type name for the SDKObjectDefinition provided in `input`. -func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPackageName *string, serviceTypeNames *[]string, commonTypesPackageName *string) (*string, error) { +func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPackageName *string, commonTypesPackageName *string) (*string, error) { // TODO: we should look to add unit tests for this method // NOTE: all of this validation should be done in the Importer and the API - this is purely sanity checking @@ -31,7 +31,7 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa return nil, fmt.Errorf("a dictionary type must have a nested item but didn't get one") } - innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName, serviceTypeNames, commonTypesPackageName) + innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName, commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining inner type for dictionary: %+v", err) } @@ -44,7 +44,7 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa return nil, fmt.Errorf("a list type must have a nested item but didn't get one") } - innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName, serviceTypeNames, commonTypesPackageName) + innerType, err := GolangTypeForSDKObjectDefinition(*input.NestedItem, golangPackageName, commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining inner type for list: %+v", err) } @@ -60,19 +60,11 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa out := *input.ReferenceName if golangPackageName != nil { out = fmt.Sprintf("%s.%s", *golangPackageName, out) - } else if serviceTypeNames != nil && commonTypesPackageName != nil { - // when serviceTypeNames is specified, look to see if the referenced type is present in the service, and if not, assume it's a common type - found := false - for _, typeName := range *serviceTypeNames { - if typeName == *input.ReferenceName { - found = true - break - } - } - - if !found { - out = fmt.Sprintf("%s.%s", *commonTypesPackageName, out) + } else if input.ReferenceNameIsCommonType != nil && *input.ReferenceNameIsCommonType { + if commonTypesPackageName == nil { + return nil, fmt.Errorf("ReferenceName %q is indicated to be a common type, but common types package name was not specified", *input.ReferenceName) } + out = fmt.Sprintf("%s.%s", *commonTypesPackageName, out) } return pointer.To(out), nil diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 508fe4e0dc5..5c0c9dd1fce 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -61,10 +61,6 @@ type GeneratorData struct { // the name of the service as a package (e.g. resources or eventhub) servicePackageName string - // slice of constant/model type names present for the service, used to determine whether a referenced type - // is local to the service, or is a common type - serviceTypeNames []string - // sourceType is the source data type and is the SDK package name sourceType models.SourceDataType @@ -91,14 +87,6 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) - serviceTypeNames := make([]string, 0) - for constantName := range i.ResourceDetails.Constants { - serviceTypeNames = append(serviceTypeNames, constantName) - } - for modelName := range i.ResourceDetails.Models { - serviceTypeNames = append(serviceTypeNames, modelName) - } - var commonTypesPackageName, commonTypesIncludePath *string if len(i.CommonTypes.Constants) > 0 && len(i.CommonTypes.Models) > 0 { commonTypesPackageName = pointer.To(strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_"))) @@ -120,7 +108,6 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { resourceOutputPath: resourceOutputPath, serviceClientName: fmt.Sprintf("%sClient", strings.Title(i.ResourceName)), servicePackageName: strings.ToLower(i.ServiceName), - serviceTypeNames: serviceTypeNames, source: i.Source, sourceType: i.Type, useIdAliases: false, diff --git a/tools/generator-go-sdk/internal/generator/service.go b/tools/generator-go-sdk/internal/generator/service.go index 4cac1c2052e..feb3c5766d5 100644 --- a/tools/generator-go-sdk/internal/generator/service.go +++ b/tools/generator-go-sdk/internal/generator/service.go @@ -100,9 +100,9 @@ func (s *ServiceGenerator) GenerateForVersion(input VersionGeneratorInput) error } runGoFmt(data.commonTypesOutputPath) - runGoImports(data.commonTypesOutputPath) - runGoFmt(data.versionOutputPath) + + runGoImports(data.commonTypesOutputPath) runGoImports(data.versionOutputPath) return nil diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 97b463781e3..c6dce5a0b8f 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -301,10 +301,16 @@ func (c methodsPandoraTemplater) listOperationTemplate(data GeneratorData) (*str if err != nil { return nil, fmt.Errorf("building options struct: %+v", err) } - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } + predicateName := "OperationPredicate" + if parts := strings.SplitN(*typeName, ".", 2); len(parts) == 2 { + predicateName = fmt.Sprintf("%s%s", parts[1], predicateName) + } else { + predicateName = fmt.Sprintf("%s%s", *typeName, predicateName) + } templated := fmt.Sprintf(` %[6]s @@ -342,11 +348,11 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, // %[2]sComplete retrieves all the results into a single object func (c %[1]s) %[2]sComplete(ctx context.Context%[4]s) (%[2]sCompleteResult, error) { - return c.%[2]sCompleteMatchingPredicate(ctx%[6]s, %[7]sOperationPredicate{}) + return c.%[2]sCompleteMatchingPredicate(ctx%[6]s, %[8]s{}) } // %[2]sCompleteMatchingPredicate retrieves all the results and then applies the predicate -func (c %[1]s) %[2]sCompleteMatchingPredicate(ctx context.Context%[4]s, predicate %[7]sOperationPredicate) (result %[2]sCompleteResult, err error) { +func (c %[1]s) %[2]sCompleteMatchingPredicate(ctx context.Context%[4]s, predicate %[8]s) (result %[2]sCompleteResult, err error) { items := make([]%[7]s, 0) resp, err := c.%[2]s(ctx%[6]s) @@ -369,7 +375,7 @@ func (c %[1]s) %[2]sCompleteMatchingPredicate(ctx context.Context%[4]s, predicat } return } -`, data.serviceClientName, c.operationName, data.packageName, *methodArguments, *responseStruct, argumentsCode, *typeName) +`, data.serviceClientName, c.operationName, data.packageName, *methodArguments, *responseStruct, argumentsCode, *typeName, predicateName) } else { templated += fmt.Sprintf(` // %[2]sComplete retrieves all the results into a single object @@ -471,7 +477,7 @@ func (c methodsPandoraTemplater) argumentsTemplateForMethod(data GeneratorData) arguments = append(arguments, fmt.Sprintf("id %s", idName)) } if c.operation.RequestObject != nil { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining type name for request object: %+v", err) } @@ -583,7 +589,7 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin } if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } @@ -661,7 +667,7 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin result.Model = &model `, discriminatedTypeParentName) } else { - responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) + responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } @@ -692,7 +698,7 @@ func (c methodsPandoraTemplater) responseStructTemplate(data GeneratorData) (*st model := "" typeName := "" if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, &data.serviceTypeNames, data.commonTypesPackageName) + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go b/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go index 8b2702d183d..4d5f40ff501 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_autorest.go @@ -225,7 +225,7 @@ func (c methodsAutoRestTemplater) listOperationTemplate(data GeneratorData) (*st if err != nil { return nil, fmt.Errorf("building responder template: %+v", err) } - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -450,7 +450,7 @@ func (c methodsAutoRestTemplater) argumentsTemplateForMethod(data GeneratorData) arguments = append(arguments, fmt.Sprintf("id %s", idName)) } if c.operation.RequestObject != nil { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.RequestObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining type name for request object: %+v", err) } @@ -600,7 +600,7 @@ func (c methodsAutoRestTemplater) responderTemplate(responseStructName string, d steps = append(steps, "autorest.ByClosing()") if c.operation.FieldContainingPaginationDetails != nil && discriminatedType == "" { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -652,7 +652,7 @@ func (c %[1]s) responderFor%[2]s(resp *http.Response) (result %[6]s, err error) } if discriminatedType != "" && c.operation.FieldContainingPaginationDetails != nil { - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for response object: %+v", err) } @@ -767,7 +767,7 @@ func (c methodsAutoRestTemplater) responseStructTemplate(responseStructName stri model := "" typeName := "" if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil, nil) + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, nil) if err != nil { return nil, fmt.Errorf("determing golang type name for response object: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 2147884e529..c37fc2e5d67 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -87,7 +87,7 @@ type Raw%[1]sImpl struct { for _, fieldName := range fields { fieldDetails := c.model.Fields[fieldName] fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil, nil) + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil) if err != nil { return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) } @@ -138,7 +138,7 @@ type Raw%[1]sImpl struct { for _, fieldName := range parentFields { fieldDetails := parent.Fields[fieldName] fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil, nil) + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil) if err != nil { return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_predicates.go b/tools/generator-go-sdk/internal/generator/templater_predicates.go index c2c242f05c0..8e2a71e2494 100644 --- a/tools/generator-go-sdk/internal/generator/templater_predicates.go +++ b/tools/generator-go-sdk/internal/generator/templater_predicates.go @@ -95,7 +95,7 @@ func (p predicateTemplater) templateForModel(predicateStructName string, name st for _, fieldName := range fieldNames { fieldVal := model.Fields[fieldName] - typeInfo, err := helpers.GolangTypeForSDKObjectDefinition(fieldVal.ObjectDefinition, nil, nil, nil) + typeInfo, err := helpers.GolangTypeForSDKObjectDefinition(fieldVal.ObjectDefinition, nil, nil) if err != nil { return nil, fmt.Errorf("determining type information for field %q in model %q with info %q: %+v", fieldName, name, string(fieldVal.ObjectDefinition.Type), err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_readme.go b/tools/generator-go-sdk/internal/generator/templater_readme.go index ca3831a735e..b4cfa44960d 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme.go @@ -157,7 +157,7 @@ payload := %[1]s.%[2]s{ `, packageName, *operation.RequestObject.ReferenceName)) } else { // for simplicities sake - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for request object: %+v", err) } @@ -213,7 +213,7 @@ payload := %[1]s.%[2]s{ `, packageName, *operation.RequestObject.ReferenceName)) } else { // for simplicities sake - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for request object: %+v", err) } @@ -270,7 +270,7 @@ payload := %[1]s.%[2]s{ `, packageName, *operation.RequestObject.ReferenceName)) } else { // for simplicities sake - typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil, nil) + typeName, err := helpers.GolangTypeForSDKObjectDefinition(*operation.RequestObject, nil, nil) if err != nil { return nil, fmt.Errorf("determining golang type name for request object: %+v", err) } diff --git a/tools/generator-terraform/internal/generator/mappings/assignment_direct.go b/tools/generator-terraform/internal/generator/mappings/assignment_direct.go index 4e7179967f1..9c95e18dcc0 100644 --- a/tools/generator-terraform/internal/generator/mappings/assignment_direct.go +++ b/tools/generator-terraform/internal/generator/mappings/assignment_direct.go @@ -248,7 +248,7 @@ output.%[1]s = &%[4]s return nil, fmt.Errorf("expected a DirectAssignment between %q and %q but got %q", string(schemaField.ObjectDefinition.NestedObject.Type), string(v), string(sdkField.ObjectDefinition.NestedItem.Type)) } - variableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) + variableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type for Sdk Model %q / Field %q Object Definition: %+v", mapping.DirectAssignment.SDKModelName, mapping.DirectAssignment.SDKFieldName, err) } @@ -293,12 +293,12 @@ output.%[1]s = &%[3]s } func (d directAssignmentLine) schemaToSdkMappingBetweenListOfReferenceFields(mapping models.TerraformDirectAssignmentFieldMappingDefinition, schemaField models.TerraformSchemaField, sdkField models.SDKField, apiResourcePackageName string) (*string, error) { - listVariableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) + listVariableType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type for Sdk Model %q / Field %q Object Definition: %+v", mapping.DirectAssignment.SDKModelName, mapping.DirectAssignment.SDKFieldName, err) } - listItemVariableReferenceType, err := helpers.GolangTypeForSDKObjectDefinition(*sdkField.ObjectDefinition.NestedItem, &apiResourcePackageName, nil, nil) + listItemVariableReferenceType, err := helpers.GolangTypeForSDKObjectDefinition(*sdkField.ObjectDefinition.NestedItem, &apiResourcePackageName, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type for Sdk Model %q / Field %q Nested Object Definition: %+v", mapping.DirectAssignment.SDKModelName, mapping.DirectAssignment.SDKFieldName, err) } diff --git a/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go b/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go index ddbd7096e21..46afe4d6d09 100644 --- a/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go +++ b/tools/generator-terraform/internal/generator/mappings/assignment_model_to_model.go @@ -29,7 +29,7 @@ func (m modelToModelAssignmentLine) assignmentForCreateUpdateMapping(mapping mod return nil, fmt.Errorf("a ModelToModel mapping must be a Reference but got %q", string(sdkField.ObjectDefinition.Type)) } outputModelName := *sdkField.ObjectDefinition.ReferenceName - sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) + sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type Name for SDK Field: %+v", err) } @@ -68,7 +68,7 @@ func (m modelToModelAssignmentLine) assignmentForReadMapping(mapping models.Terr } outputModelName := *sdkField.ObjectDefinition.ReferenceName - sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil, nil) + sdkFieldType, err := helpers.GolangTypeForSDKObjectDefinition(sdkField.ObjectDefinition, &apiResourcePackageName, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type Name for SDK Field: %+v", err) } diff --git a/tools/generator-terraform/internal/generator/resource/component_create_func.go b/tools/generator-terraform/internal/generator/resource/component_create_func.go index 20f2ecdec58..d3746eaf368 100644 --- a/tools/generator-terraform/internal/generator/resource/component_create_func.go +++ b/tools/generator-terraform/internal/generator/resource/component_create_func.go @@ -199,7 +199,7 @@ id := %[3]s(%[4]s) func (h createFunctionComponents) payloadDefinition() (*string, error) { // NOTE: whilst Payload is _technically_ optional in the API endpoint it's not, else it // wouldn't be a Create method - createObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.createMethod.RequestObject, &h.sdkResourceNameLowered, nil, nil) + createObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.createMethod.RequestObject, &h.sdkResourceNameLowered, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type name for Create Request Object: %+v", err) } diff --git a/tools/generator-terraform/internal/generator/resource/component_update_func.go b/tools/generator-terraform/internal/generator/resource/component_update_func.go index 6c27c497d55..e4ae804da84 100644 --- a/tools/generator-terraform/internal/generator/resource/component_update_func.go +++ b/tools/generator-terraform/internal/generator/resource/component_update_func.go @@ -130,7 +130,7 @@ func (h updateFuncHelpers) modelDecode() (*string, error) { } func (h updateFuncHelpers) payloadDefinition() (*string, error) { - updateObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.updateMethod.RequestObject, &h.sdkResourceNameLowered, nil, nil) + updateObjectName, err := helpers.GolangTypeForSDKObjectDefinition(*h.updateMethod.RequestObject, &h.sdkResourceNameLowered, nil) if err != nil { return nil, fmt.Errorf("determining Golang Type name for Update Request Object: %+v", err) } From 4d0069932471dddbf7248d29c64842ce1e03544f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 02:07:35 +0100 Subject: [PATCH 006/117] data-api-sdk: oops, resource manager is not data plan (and vice versa) --- tools/data-api-sdk/v1/models/source_data_types.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/data-api-sdk/v1/models/source_data_types.go b/tools/data-api-sdk/v1/models/source_data_types.go index fc2e71a2b8d..41aa3061e5b 100644 --- a/tools/data-api-sdk/v1/models/source_data_types.go +++ b/tools/data-api-sdk/v1/models/source_data_types.go @@ -15,5 +15,5 @@ const ( ) func SourceDataTypeIsDataPlane(sourceDataType SourceDataType) bool { - return sourceDataType == ResourceManagerSourceDataType + return sourceDataType != ResourceManagerSourceDataType } From d49c26f7881902416ca1b5cab562bc59db7ebc24 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 02:16:40 +0100 Subject: [PATCH 007/117] importer-msgraph-metadata: prefix constant key names with underscore, to prevent naming conflicts with annoyingly named models --- .../internal/pipeline/translate_models.go | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go index 2bdb5f614a9..696baa53c25 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go @@ -4,6 +4,8 @@ package pipeline import ( + "fmt" + sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" @@ -25,7 +27,8 @@ func translateModelsToDataApiSdkTypes(models parser.Models, constants parser.Con for constantName, constant := range constants { constantValues := make(map[string]string) for _, value := range constant.Enum { - constantValues[normalize.CleanName(value)] = value + // prefix constant value names with underscore to prevent naming conflicts with similarly named models in the generated SDK + constantValues[fmt.Sprintf("_%s", normalize.CleanName(value))] = value } // TODO support additional types, if there are any From 48e45999374ee97fbf24431254c60364bd5f6b26 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 02:56:40 +0100 Subject: [PATCH 008/117] importer-msgraph-metadata: blacklist some resources --- .../components/blacklisted/blacklist.go | 24 +++++++++++++++++++ .../pipeline/task_translate_service.go | 6 +++++ 2 files changed, 30 insertions(+) create mode 100644 tools/importer-msgraph-metadata/components/blacklisted/blacklist.go diff --git a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go new file mode 100644 index 00000000000..6269a58abf4 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go @@ -0,0 +1,24 @@ +package blacklisted + +import ( + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "strings" +) + +func Resource(resource *parser.Resource) bool { + if resource.Service == "Groups" { + + // GroupSiteTermStore resources have repeating ID segments which are not supported at this time + if strings.Contains(resource.Name, "TermStore") { + return true + } + + // Onenote resources have repeating ID segments which are not supported at this time + if strings.Contains(resource.Name, "Onenote") { + return true + } + + } + + return false +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index 785fc24c2be..a7b8c623ea2 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -5,7 +5,9 @@ package pipeline import ( "fmt" + sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/blacklisted" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" @@ -15,6 +17,10 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta sdkServices := make(map[string]sdkModels.Service) for _, resource := range resources { + if blacklisted.Resource(resource) { + continue + } + // First scaffold all discovered services, version(s) and resources (categories) if _, ok := sdkServices[resource.Service]; !ok { sdkServices[resource.Service] = sdkModels.Service{ From f1dfce0ab1d943d98036b2a480c0ccc6dab61e83 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 02:57:12 +0100 Subject: [PATCH 009/117] importer-msgraph-metadata: fix openapi file pattern for list-tags command --- .../internal/supported-services/supported_services.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/internal/supported-services/supported_services.go b/tools/importer-msgraph-metadata/internal/supported-services/supported_services.go index ab65978bb0a..b80b513c2ee 100644 --- a/tools/importer-msgraph-metadata/internal/supported-services/supported_services.go +++ b/tools/importer-msgraph-metadata/internal/supported-services/supported_services.go @@ -26,7 +26,7 @@ type SupportedServicesInput struct { func OutputSupportedServices(input SupportedServicesInput) error { for _, apiVersion := range versions.Supported { - openApiFile := fmt.Sprintf(input.OpenApiFilePattern, apiVersion) + openApiFile := fmt.Sprintf(input.OpenApiFilePattern, versions.Upstream(apiVersion)) spec, err := openapi3.NewLoader().LoadFromFile(filepath.Join(input.MetadataDirectory, openApiFile)) if err != nil { From 589ea903b28cbfb89745047dd6b90281025da5f0 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:03:24 +0100 Subject: [PATCH 010/117] generator-go-sdk: try to appease apparent golang race condition with creating lots of files in a directory --- .../internal/generator/service.go | 17 +++++++++++++++-- .../internal/generator/templater.go | 4 +++- 2 files changed, 18 insertions(+), 3 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/service.go b/tools/generator-go-sdk/internal/generator/service.go index feb3c5766d5..59aab046bdf 100644 --- a/tools/generator-go-sdk/internal/generator/service.go +++ b/tools/generator-go-sdk/internal/generator/service.go @@ -7,6 +7,7 @@ import ( "fmt" "os" "os/exec" + "path/filepath" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/generator-go-sdk/internal/logging" @@ -121,12 +122,24 @@ func runGoImports(path string) { } func cleanAndRecreateWorkingDirectory(path string) error { - os.RemoveAll(path) - // TODO: make these less exciting + // first, ensure the directory exists if err := os.MkdirAll(path, 0777); err != nil { return fmt.Errorf("creating %q: %+v", path, err) } + // determine contents of output directory + pathsToDelete, err := filepath.Glob(filepath.Join(path, "*")) + if err != nil { + return fmt.Errorf("globbing files to delete: %+v", err) + } + + // delete any contained files and directories + for _, pathToDelete := range pathsToDelete { + if err = os.RemoveAll(pathToDelete); err != nil { + return fmt.Errorf("deleting %q: %w", pathToDelete, err) + } + } + return nil } diff --git a/tools/generator-go-sdk/internal/generator/templater.go b/tools/generator-go-sdk/internal/generator/templater.go index fc926f29969..3f1459afbd9 100644 --- a/tools/generator-go-sdk/internal/generator/templater.go +++ b/tools/generator-go-sdk/internal/generator/templater.go @@ -29,7 +29,9 @@ func (s *ServiceGenerator) writeToPathForResource(directory, filePath string, te // remove any existing file if it exists _ = os.Remove(fullFilePath) - file, err := os.Create(fullFilePath) + + // call os.OpenFile instead of os.Create to reduce spurious "file not found" errors when O_TRUNC is used + file, err := os.OpenFile(fullFilePath, os.O_WRONLY|os.O_CREATE, 0666) defer file.Close() if err != nil { return fmt.Errorf("opening %q: %+v", fullFilePath, err) From 98d0e96bbd12c05a3c8a47209ad1527c0f616428 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:04:45 +0100 Subject: [PATCH 011/117] generator-go-sdk: fix up remaining package references for common models in operations and predicates --- .../generator-go-sdk/internal/generator/data.go | 2 +- .../internal/generator/stage_predicates.go | 13 +++++++++++-- .../internal/generator/templater_constants.go | 6 +++++- .../internal/generator/templater_predicates.go | 16 ++++++++++++---- 4 files changed, 29 insertions(+), 8 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 5c0c9dd1fce..e937f6bfc0a 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -64,7 +64,7 @@ type GeneratorData struct { // sourceType is the source data type and is the SDK package name sourceType models.SourceDataType - // whether this is a data plane SDK (omits certain Resource Manager specific features) + // whether this is a data plane SDK (omits certain Resource Manager specific features, currently used in ID parsers) isDataPlane bool // development feature flag - this requires work in the Resource ID parser to handle name conflicts diff --git a/tools/generator-go-sdk/internal/generator/stage_predicates.go b/tools/generator-go-sdk/internal/generator/stage_predicates.go index d2d6bf03f12..5db4c06d28a 100644 --- a/tools/generator-go-sdk/internal/generator/stage_predicates.go +++ b/tools/generator-go-sdk/internal/generator/stage_predicates.go @@ -9,7 +9,7 @@ import ( ) func (s *ServiceGenerator) predicates(data GeneratorData) error { - modelNames := make(map[string]struct{}, 0) + modelNames := make(map[string]string) for _, operation := range data.operations { if operation.FieldContainingPaginationDetails == nil { continue @@ -23,7 +23,15 @@ func (s *ServiceGenerator) predicates(data GeneratorData) error { continue } - modelNames[*operation.ResponseObject.ReferenceName] = struct{}{} + modelNameWithPackage := *operation.ResponseObject.ReferenceName + if operation.ResponseObject.ReferenceNameIsCommonType != nil && *operation.ResponseObject.ReferenceNameIsCommonType { + if data.commonTypesPackageName == nil { + return fmt.Errorf("building predicate models: encountered a common model %q but `commonTypesPackageName` was nil", *operation.ResponseObject.ReferenceName) + } + modelNameWithPackage = fmt.Sprintf("%s.%s", *data.commonTypesPackageName, modelNameWithPackage) + } + + modelNames[*operation.ResponseObject.ReferenceName] = modelNameWithPackage } sortedModelNames := make([]string, 0) @@ -37,6 +45,7 @@ func (s *ServiceGenerator) predicates(data GeneratorData) error { } templater := predicateTemplater{ + modelNames: modelNames, sortedModelNames: sortedModelNames, models: data.models, } diff --git a/tools/generator-go-sdk/internal/generator/templater_constants.go b/tools/generator-go-sdk/internal/generator/templater_constants.go index a1917c1e93e..bb212b8a582 100644 --- a/tools/generator-go-sdk/internal/generator/templater_constants.go +++ b/tools/generator-go-sdk/internal/generator/templater_constants.go @@ -54,7 +54,11 @@ func (c constantsTemplater) template(data GeneratorData) (*string, error) { template := fmt.Sprintf(`package %[1]s -import "strings" +import ( + "encoding/json" + "fmt" + "strings" +) %[3]s diff --git a/tools/generator-go-sdk/internal/generator/templater_predicates.go b/tools/generator-go-sdk/internal/generator/templater_predicates.go index 8e2a71e2494..4e6a161cc36 100644 --- a/tools/generator-go-sdk/internal/generator/templater_predicates.go +++ b/tools/generator-go-sdk/internal/generator/templater_predicates.go @@ -12,6 +12,7 @@ import ( // TODO: add unit tests covering this type predicateTemplater struct { + modelNames map[string]string sortedModelNames []string models map[string]models.SDKModel } @@ -19,13 +20,14 @@ type predicateTemplater struct { func (p predicateTemplater) template(data GeneratorData) (*string, error) { output := make([]string, 0) for _, modelName := range p.sortedModelNames { + modelNameWithPackage := p.modelNames[modelName] model := data.models[modelName] predicateStructName := fmt.Sprintf("%sOperationPredicate", modelName) if _, hasExisting := data.models[predicateStructName]; hasExisting { return nil, fmt.Errorf("existing model %q conflicts with predicate model for %q", predicateStructName, modelName) } - templated, err := p.templateForModel(predicateStructName, modelName, model) + templated, err := p.templateForModel(predicateStructName, modelName, modelNameWithPackage, model) if err != nil { return nil, err } @@ -37,15 +39,21 @@ func (p predicateTemplater) template(data GeneratorData) (*string, error) { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) } + commonTypesInclude := "" + if data.commonTypesIncludePath != nil { + commonTypesInclude = fmt.Sprintf(`import "github.com/hashicorp/go-azure-sdk/%s/%s"`, data.sourceType, *data.commonTypesIncludePath) + } + template := fmt.Sprintf(`package %[1]s %[2]s %[3]s -`, data.packageName, *copyrightLines, strings.Join(output, "\n")) +%[4]s +`, data.packageName, *copyrightLines, commonTypesInclude, strings.Join(output, "\n")) return &template, nil } -func (p predicateTemplater) templateForModel(predicateStructName string, name string, model models.SDKModel) (*string, error) { +func (p predicateTemplater) templateForModel(predicateStructName string, name, nameWithPackage string, model models.SDKModel) (*string, error) { fieldNames := make([]string, 0) // unsupported at this time - see https://github.com/hashicorp/pandora/issues/164 @@ -128,6 +136,6 @@ func (p %[1]s) Matches(input %[2]s) bool { return true } -`, predicateStructName, name, strings.Join(structLines, "\n"), strings.Join(matchLines, "\n")) +`, predicateStructName, nameWithPackage, strings.Join(structLines, "\n"), strings.Join(matchLines, "\n")) return &template, nil } From cfaff1421c571efdbcc26758b5aa20b99e5fafe7 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:05:52 +0100 Subject: [PATCH 012/117] msgraph: import more services, attain feature parity (and much more) with hamilton --- config/microsoft-graph.hcl | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/config/microsoft-graph.hcl b/config/microsoft-graph.hcl index d38cd735dcb..29ddecbc534 100644 --- a/config/microsoft-graph.hcl +++ b/config/microsoft-graph.hcl @@ -21,6 +21,16 @@ service "applicationTemplates" { available = ["stable", "beta"] } +service "auditLogs" { + name = "AuditLogs" + available = ["stable", "beta"] +} + +service "deviceManagement" { + name = "DeviceManagement" + available = ["stable", "beta"] +} + service "directory" { name = "Directory" available = ["stable", "beta"] @@ -46,6 +56,11 @@ service "identity" { available = ["stable"] } +service "identityGovernance" { + name = "IdentityGovernance" + available = ["stable"] +} + service "invitations" { name = "Invitations" available = ["stable", "beta"] @@ -66,11 +81,21 @@ service "policies" { available = ["stable", "beta"] } +service "reports" { + name = "Reports" + available = ["stable", "beta"] +} + service "roleManagement" { name = "RoleManagement" available = ["stable", "beta"] } +service "schemaExtensions" { + name = "SchemaExtensions" + available = ["stable", "beta"] +} + service "servicePrincipals" { name = "ServicePrincipals" available = ["stable", "beta"] From 755f766e6606040939539388ab8c3a4c10c885b5 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:35:40 +0100 Subject: [PATCH 013/117] importer-msgraph-metadata: trim leading word from resource names when it matches the service name --- .../internal/pipeline/task_parse_resources.go | 53 +++++++++++++------ 1 file changed, 38 insertions(+), 15 deletions(-) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index cbff6c62184..90268ddf3d7 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -300,27 +300,50 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + // Look for resources without a category, then iterate the known paths of it and all potential parent resources + // to find a match by truncating its path to the preceding label segment. Once a match is found, adopt the + // resource category of the matched parent to ensure they are grouped together. for _, resource := range resources { - // Look for resources without a category, then iterate the known paths of it and all potential parent resources - // to find a match by truncating its path to the preceding label segment. Once a match is found, adopt the - // resource category of the matched parent to ensure they are grouped together. - if pathsLen := len(resource.Paths); resource.Category == "" && pathsLen > 0 { - for _, path := range resource.Paths { - if trimmedPath := path.TruncateToLastSegmentOfTypeBeforeSegment([]parser.ResourceIdSegmentType{parser.SegmentLabel}, -1); trimmedPath != nil { - for _, parentResource := range resources { - if parentResource.Category != "" { - for _, parentPath := range parentResource.Paths { - if parentPath.ID() == trimmedPath.ID() { - resource.Category = parentResource.Category - break - } - } - } + pathsLen := len(resource.Paths) + if resource.Category != "" || pathsLen == 0 { + continue + } + + for _, path := range resource.Paths { + trimmedPath := path.TruncateToLastSegmentOfTypeBeforeSegment([]parser.ResourceIdSegmentType{parser.SegmentLabel}, -1) + if trimmedPath == nil { + continue + } + + for _, parentResource := range resources { + if parentResource.Category == "" { + continue + } + + for _, parentPath := range parentResource.Paths { + if parentPath.ID() == trimmedPath.ID() { + resource.Category = parentResource.Category + break } } } } } + // Loop through resources and trim the leading word if it matches the category _and_ there are more words after it + for _, resource := range resources { + if resource.Service == "" || resource.Category == "" { + continue + } + + serviceSingularized := normalize.Singularize(resource.Service) + if strings.HasPrefix(resource.Category, serviceSingularized) { + trimmedCategory := strings.TrimPrefix(resource.Category, serviceSingularized) + if len(trimmedCategory) > 1 { + resource.Category = trimmedCategory + } + } + } + return } From 43b5f4afe2a69442a850564eaeef7f679231460f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:36:08 +0100 Subject: [PATCH 014/117] importer-msgraph-metadata: blacklist OneNote resources in the Users service --- .../components/blacklisted/blacklist.go | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go index 6269a58abf4..aa0d8943c11 100644 --- a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go +++ b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go @@ -20,5 +20,14 @@ func Resource(resource *parser.Resource) bool { } + if resource.Service == "Users" { + + // Onenote resources have repeating ID segments which are not supported at this time + if strings.Contains(resource.Name, "Onenote") { + return true + } + + } + return false } From 3e96ea0b4c1a8761cbf1f97cc7d508545d8f60dc Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:36:58 +0100 Subject: [PATCH 015/117] importer-msgraph-metadata: remove the "ById" resource/operation suffix --- .../importer-msgraph-metadata/components/parser/resourceids.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 63197bfce27..7fc69527f20 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -14,7 +14,7 @@ import ( ) const ( - ResourceSuffix = "ById" + ResourceSuffix = "" ResourceIdSuffix = "Id" ) From 8f528c432a8dab1653301e5c9d95e94394503472 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 03:37:19 +0100 Subject: [PATCH 016/117] importer-msgraph-metadata: bump tool version --- tools/importer-msgraph-metadata/main.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/main.go b/tools/importer-msgraph-metadata/main.go index 1c919bbf236..ecdc3b884c8 100644 --- a/tools/importer-msgraph-metadata/main.go +++ b/tools/importer-msgraph-metadata/main.go @@ -27,7 +27,7 @@ func main() { } logging.Log = hclog.New(loggingOpts) - c := cli.NewCLI("importer-msgraph-metadata", "0.2.1") + c := cli.NewCLI("importer-msgraph-metadata", "0.2.2") c.Args = os.Args[1:] c.Commands = map[string]cli.CommandFactory{ "import": cmd.NewImportCommand(metadataDirectory, microsoftGraphConfig, openApiFilePattern, outputDirectory), From 85c3badb23e09a0350fa9cdb11e3dc19cfb75480 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 04:22:56 +0100 Subject: [PATCH 017/117] importer-msgraph-metadata: trim leading word from operation names when it matches the service name --- .../internal/pipeline/task_parse_resources.go | 27 +++++++++++-------- 1 file changed, 16 insertions(+), 11 deletions(-) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 90268ddf3d7..3844f9bbbbd 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -207,36 +207,41 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model prefixToTrim := normalize.Singularize(normalize.CleanName(p.service)) if resourceId != nil && uriSuffix == nil { - prefixToTrim = fmt.Sprintf("%sById", prefixToTrim) + prefixToTrim = fmt.Sprintf("%s%s", prefixToTrim, parser.ResourceSuffix) } shortResourceName := strings.TrimPrefix(resourceName, prefixToTrim) + operationName = shortResourceName + if len(operationName) == 0 { + operationName = resourceName + } + switch operationType { case parser.OperationTypeList: if _, ok = normalize.Verbs.Match(shortResourceName); ok { - operationName = normalize.Pluralize(normalize.Singularize(resourceName)) + operationName = normalize.Pluralize(normalize.Singularize(operationName)) } else { - operationName = fmt.Sprintf("List%s", normalize.Pluralize(normalize.Singularize(resourceName))) + operationName = fmt.Sprintf("List%s", normalize.Pluralize(normalize.Singularize(operationName))) } case parser.OperationTypeRead: - operationName = fmt.Sprintf("Get%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("Get%s", normalize.Singularize(operationName)) case parser.OperationTypeCreate: if _, ok = normalize.Verbs.Match(shortResourceName); ok { - operationName = normalize.Singularize(resourceName) + operationName = normalize.Singularize(operationName) } else if lastSegment.Type == parser.SegmentODataReference { - operationName = fmt.Sprintf("Add%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("Add%s", normalize.Singularize(operationName)) } else { - operationName = fmt.Sprintf("Create%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("Create%s", normalize.Singularize(operationName)) } case parser.OperationTypeCreateUpdate: - operationName = fmt.Sprintf("CreateUpdate%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("CreateUpdate%s", normalize.Singularize(operationName)) case parser.OperationTypeUpdate: - operationName = fmt.Sprintf("Update%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("Update%s", normalize.Singularize(operationName)) case parser.OperationTypeDelete: if lastSegment.Type == parser.SegmentODataReference { - operationName = fmt.Sprintf("Remove%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("Remove%s", normalize.Singularize(operationName)) } else { - operationName = fmt.Sprintf("Delete%s", normalize.Singularize(resourceName)) + operationName = fmt.Sprintf("Delete%s", normalize.Singularize(operationName)) } } From 449383dae01f6852ee05232701f62b0feb226780 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 13:24:53 +0100 Subject: [PATCH 018/117] importer-msgraph-metadata: blacklist more services due to inscrutable/unsupported naming --- .../components/blacklisted/blacklist.go | 45 ++++++++++++++++++- 1 file changed, 43 insertions(+), 2 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go index aa0d8943c11..74b783b2621 100644 --- a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go +++ b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go @@ -1,13 +1,20 @@ package blacklisted import ( - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "fmt" "strings" + + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" ) func Resource(resource *parser.Resource) bool { if resource.Service == "Groups" { + // Repeating ID segments which are not supported + if resource.Name == "SiteSite" { + return true + } + // GroupSiteTermStore resources have repeating ID segments which are not supported at this time if strings.Contains(resource.Name, "TermStore") { return true @@ -20,13 +27,47 @@ func Resource(resource *parser.Resource) bool { } - if resource.Service == "Users" { + if resource.Service == "IdentityGovernance" { + + // Repeating ID segments which are not supported at this time + prefixes := []string{ + "EntitlementManagementAccessPackageResourceRoleScopeRoleResourceScopeResourceRole", + "EntitlementManagementAccessPackageResourceRoleScopeScopeResourceRoleResourceScope", + "EntitlementManagementCatalogResourceRoleResourceScopeResourceRole", + "EntitlementManagementCatalogResourceScopeResourceRoleResourceScope", + "EntitlementManagementResourceRequestCatalogResourceRoleResourceScopeResource", + "EntitlementManagementResourceRequestCatalogResourceScopeResourceRoleResourceScope", + "EntitlementManagementResourceRequestResourceRoleResourceScope", + "EntitlementManagementResourceRequestResourceScopeResourceRole", + "EntitlementManagementResourceRoleScopeRoleResourceScopeResourceRole", + "EntitlementManagementResourceRoleScopeScopeResourceRoleResourceScope", + } + for _, prefix := range prefixes { + if strings.HasPrefix(resource.Name, prefix) || strings.HasPrefix(resource.Name, fmt.Sprintf("%s%s", resource.Service, prefix)) { + return true + } + } + + } + + if resource.Service == "Me" || resource.Service == "Users" { // Onenote resources have repeating ID segments which are not supported at this time if strings.Contains(resource.Name, "Onenote") { return true } + // Repeating ID segments which are not supported at this time + prefixes := []string{ + "PendingAccessReviewInstanceDecisionInstanceStageDecision", + "PendingAccessReviewInstanceStageDecisionInstanceDecision", + } + for _, prefix := range prefixes { + if strings.HasPrefix(resource.Name, prefix) || strings.HasPrefix(resource.Name, fmt.Sprintf("%s%s", resource.Service, prefix)) { + return true + } + } + } return false From d174e891855337d27e6155bbd7beabe2492af677 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 15:16:22 +0100 Subject: [PATCH 019/117] importer-msgraph-metadata: fix blacklist to compare the category rather than the resource name --- .../components/blacklisted/blacklist.go | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go index 74b783b2621..672de85a53d 100644 --- a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go +++ b/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go @@ -10,18 +10,18 @@ import ( func Resource(resource *parser.Resource) bool { if resource.Service == "Groups" { - // Repeating ID segments which are not supported - if resource.Name == "SiteSite" { + // Has IDs with repeating segments, which are not supported + if resource.Category == "SiteSite" { return true } // GroupSiteTermStore resources have repeating ID segments which are not supported at this time - if strings.Contains(resource.Name, "TermStore") { + if strings.Contains(resource.Category, "TermStore") { return true } // Onenote resources have repeating ID segments which are not supported at this time - if strings.Contains(resource.Name, "Onenote") { + if strings.Contains(resource.Category, "Onenote") { return true } @@ -29,7 +29,7 @@ func Resource(resource *parser.Resource) bool { if resource.Service == "IdentityGovernance" { - // Repeating ID segments which are not supported at this time + // These contain IDs with repeating segments, which are not supported at this time prefixes := []string{ "EntitlementManagementAccessPackageResourceRoleScopeRoleResourceScopeResourceRole", "EntitlementManagementAccessPackageResourceRoleScopeScopeResourceRoleResourceScope", @@ -43,7 +43,7 @@ func Resource(resource *parser.Resource) bool { "EntitlementManagementResourceRoleScopeScopeResourceRoleResourceScope", } for _, prefix := range prefixes { - if strings.HasPrefix(resource.Name, prefix) || strings.HasPrefix(resource.Name, fmt.Sprintf("%s%s", resource.Service, prefix)) { + if strings.HasPrefix(resource.Category, prefix) || strings.HasPrefix(resource.Category, fmt.Sprintf("%s%s", resource.Service, prefix)) { return true } } @@ -53,17 +53,17 @@ func Resource(resource *parser.Resource) bool { if resource.Service == "Me" || resource.Service == "Users" { // Onenote resources have repeating ID segments which are not supported at this time - if strings.Contains(resource.Name, "Onenote") { + if strings.Contains(resource.Category, "Onenote") { return true } - // Repeating ID segments which are not supported at this time + // These contain IDs with repeating segments, which are not supported at this time prefixes := []string{ "PendingAccessReviewInstanceDecisionInstanceStageDecision", "PendingAccessReviewInstanceStageDecisionInstanceDecision", } for _, prefix := range prefixes { - if strings.HasPrefix(resource.Name, prefix) || strings.HasPrefix(resource.Name, fmt.Sprintf("%s%s", resource.Service, prefix)) { + if strings.HasPrefix(resource.Category, prefix) || strings.HasPrefix(resource.Category, fmt.Sprintf("%s%s", resource.Service, prefix)) { return true } } From e1472e98bb2d6d816176d04148a6ff4c7a5f0425 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 15:18:28 +0100 Subject: [PATCH 020/117] importer-msgraph-metadata: write initial documentation --- tools/importer-msgraph-metadata/README.md | 67 +++++++++++++++++++++++ 1 file changed, 67 insertions(+) create mode 100644 tools/importer-msgraph-metadata/README.md diff --git a/tools/importer-msgraph-metadata/README.md b/tools/importer-msgraph-metadata/README.md new file mode 100644 index 00000000000..374f7b19ffb --- /dev/null +++ b/tools/importer-msgraph-metadata/README.md @@ -0,0 +1,67 @@ +# Microsoft Graph API Specs Importer + +The Microsoft Graph API specs are in OpenAPI3 format. The data we parse is automatically generated from the original API Metadata which are in OData XML format. This translation to OpenAPI3 format is performed by a [.NET tool](https://github.com/microsoft/OpenAPI.NET.OData) published by Microsoft. + +You can read more about Microsoft Graph Metadata at https://learn.microsoft.com/en-us/graph/traverse-the-graph + +The metadata can be downloaded directly from the service at one of the following URLs: +- https://graph.microsoft.com/v1.0/$metadata +- https://graph.microsoft.com/beta/$metadata + +Both the metadata and OpenAPI3 specs are published on GitHub at [microsoftgraph/msgraph-metadata](https://github.com/microsoftgraph/msgraph-metadata); this is where we currently source this data as it's versioned and fits in with our git submodule approach. + +## API Versions + +MS Graph has two distinct API versions, which are the same across all services. These are `v1.0` and `beta`. You may also sometimes see references to a `v2.0` API, this appears to be internal and/or never released to the public, but it does leak through in some API responses. + +Each API version comprises many services, which are exposed via a unified frontend and can be found at various URL subtrees. All the services for each API version are described in a single metadata / OpenAPI3 file per version. + +When importing from these API versions, we normalize them to `stable` and `beta` as this makes for easier handling when the Go SDK is built. + +New features and services are _supposed_ to go into the `beta` API, with a period of public preview, before being added to the `v1.0` API. The latter is not supposed to receive breaking changes, however in practice this is clearly impossible and so it regularly receives breaking changes. + +Additionally, whilst Microsoft disclaims use of the `beta` API in production and recommends all customers use the `v1.0` API, there are lots of services that have yet to make it out of beta/preview, and the Portal uses the beta API literally everywhere. + +## Adding New Services + +To assist when adding new services, the importer has a `list-tags` subcommand which is intended to be run locally and which outputs a list of tags and subtags for each API version. Whilst subtags are not supported by the importer, this lets you see at a glance which tags are available and these are translated directly into Services. + +The importer reads from the `config/microsoft-graph.hcl` configuration file (from the root of the Pandora repository). Similarly to Resource Manager, this is where to configure services/tags for importing. Some tags are only supported in one API version, and others have different features in each API version. + +## The Importer + +We use the `github.com/getkin/kin-openapi` module to read the OpenAPI3 specs, and implement our own parser in the `components/parser` package. This shares many of the same paradigms as the `importer-rest-api-specs` tool, but adjusted for the different OpenAPI specification and the way in which OData objects are expressed in the translated spec. + +This tool has its own internal types for representing things like Services, Resources, Resource IDs, Models etc, these are all defined in the `components/parser` package. The core parsing logic of populating these types happens within that package, whilst the "business logic" of determining what to import and how to import it happens in the `internal/pipeline` package. + +One notable feature of Microsoft Graph compared to Resource Manager is that data types (models) are shared across the entire API. For example, the `users` service would make use of `directoryObject`, `group`, `servicePrincipal` and other models. Furthermore, internally within MS Graph most objects have some inheritance, forming a tree of models with behavior much like class inheritance. We intentionally don't express this when importing, and have elected to flatten all ancestor fields in a family tree. + +This global namespace of models leads to some excessive duplication if we define those models on a per-resource or per-service basis, so we consolidate them into a single namespace we refer to as "common types". In the generated SDK, this is output as separate packages called `common-types/stable` and `common-types/beta`, doing so greatly reduces the SDK footprint by an order of magnitude. + +### How the internal types map to data API types + +After importing and processing the API specs using the internal types in `components/parser`, they are then translated to the Data API native types to be persisted in the Data API. There are some things to be aware of in this translation: + +* The internal types for Resources have a `Category` and a `Name` - the `Category` maps to the resource name in the data API and the `Name` is used to build operation names. The reason for this approach is that each "resource" in the data API contains multiple Graph resources and/or representations of resources, so the importer organises these internally as "categories" to facilitate grouping of resources prior to translation. +* All models and constants described in the API spec are considered to be "common types", the only models which are local to a resource are those that represent complex request objects. +* Resource IDs are all considered to be resource-local - even though they are intentionally fully-namespaced and are all parsed out together prior to parsing out the resources, so they could in theory _could_ be pooled together as "common resource IDs". However, this would require refactoring of the SDK generator, and at this time (unlike with models), there is no significant overhead to duplicating these. + +## About OData + +[OData](https://www.odata.org/) is a data representation & navigation specification published by Microsoft and Oracle with low industry adoption. It is largely built for .NET and this is the only widely used language with proper support for it. Golang has no library support that we've found which is why we have [built this ourselves](https://github.com/hashicorp/go-azure-sdk/tree/main/sdk/odata) (whilst trying to keep it simple and generic). + +Whilst Resource Manager makes use of some OData primitives, it has been embraced in Microsoft Graph and is the primary means of navigating the API and expressing relationships between resource/objects. + +However, it is worth pointing out that MS Graph has made notably noncompliant revisions to their OData implementation. Particularly around the use of OData IDs which are _supposed_ to be fully qualified URIs, for example: + +``` +https://graph.microsoft.com/v1.0/directoryObjects/00000000-0000-0000-0000-000000000000 +``` + +However, these are increasingly received as part of API responses in other formats, like: + +``` +directoryObjects('00000000-0000-0000-0000-000000000000') +``` + +The base layer SDK nor the dataplane SDKs generated by Pandora are currently able to parse these or infer the related URLs, it is up to the implementor (e.g. Terraform providers) to do so. From 5f0d52641eb137671b094d1b8c5b3aa0795be906 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 16:59:15 +0100 Subject: [PATCH 021/117] generator-go-sdk: set the correct package for the base client --- .../internal/generator/data.go | 5 ++++ .../internal/generator/helpers.go | 10 +++++++ .../internal/generator/stage_meta_client.go | 11 +++---- .../internal/generator/templater_clients.go | 10 ++++--- .../generator/templater_meta_client.go | 30 +++++++++++-------- .../internal/generator/templater_methods.go | 25 ++++++++-------- 6 files changed, 57 insertions(+), 34 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index e937f6bfc0a..9dc47fe8c06 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -16,6 +16,9 @@ type GeneratorData struct { // the name of the package which should be used packageName string + // baseClientPackage is the name of the base client package to use for this SDK + baseClientPackage string + // commonTypes contains any common models and constants for this API version commonTypes models.CommonTypes @@ -95,6 +98,7 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { return GeneratorData{ apiVersion: i.VersionName, + baseClientPackage: baseClientPackageForSdk(i.Type), commonTypes: i.CommonTypes, commonTypesIncludePath: commonTypesIncludePath, commonTypesPackageName: commonTypesPackageName, @@ -145,6 +149,7 @@ func (i VersionGeneratorInput) generatorData(settings Settings) VersionGenerator return VersionGeneratorData{ GeneratorData: GeneratorData{ apiVersion: i.VersionName, + baseClientPackage: baseClientPackageForSdk(i.Type), commonTypes: i.CommonTypes, constants: i.CommonTypes.Constants, isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), diff --git a/tools/generator-go-sdk/internal/generator/helpers.go b/tools/generator-go-sdk/internal/generator/helpers.go index 172d086d8b8..6841b458219 100644 --- a/tools/generator-go-sdk/internal/generator/helpers.go +++ b/tools/generator-go-sdk/internal/generator/helpers.go @@ -30,6 +30,16 @@ func alternateCasingOnEveryLetter(input string) string { return output } +func baseClientPackageForSdk(input models.SourceDataType) string { + switch input { + case models.MicrosoftGraphSourceDataType: + return "msgraph" + case models.ResourceManagerSourceDataType: + return "resourcemanager" + } + return "client" +} + func capitalizeFirstLetter(input string) string { return strings.ToUpper(input[0:1]) + strings.ToLower(input[1:]) } diff --git a/tools/generator-go-sdk/internal/generator/stage_meta_client.go b/tools/generator-go-sdk/internal/generator/stage_meta_client.go index 68b34079ac4..1736d418978 100644 --- a/tools/generator-go-sdk/internal/generator/stage_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/stage_meta_client.go @@ -15,11 +15,12 @@ func (s *ServiceGenerator) metaClient(data VersionGeneratorData) error { var templater templaterForVersion if data.useNewBaseLayer { templater = metaClientTemplater{ - serviceName: data.servicePackageName, - apiVersion: data.versionPackageName, - resources: data.resources, - source: data.source, - sourceType: data.sourceType, + serviceName: data.servicePackageName, + apiVersion: data.versionPackageName, + baseClientPackage: data.baseClientPackage, + resources: data.resources, + source: data.source, + sourceType: data.sourceType, } } else { templater = metaClientAutorestTemplater{ diff --git a/tools/generator-go-sdk/internal/generator/templater_clients.go b/tools/generator-go-sdk/internal/generator/templater_clients.go index 5e01b90287d..85d8dbac1cf 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients.go @@ -16,18 +16,20 @@ func (c clientsTemplater) template(data GeneratorData) (*string, error) { template := fmt.Sprintf(`package %[1]s import ( + "github.com/hashicorp/go-azure-sdk/sdk/client" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" sdkEnv "github.com/hashicorp/go-azure-sdk/sdk/environments" ) -%[3]s +%[4]s type %[2]s struct { - Client *resourcemanager.Client + Client *%[3]s.Client } func New%[2]sWithBaseURI(sdkApi sdkEnv.Api) (*%[2]s, error) { - client, err := resourcemanager.NewResourceManagerClient(sdkApi, %[1]q, defaultApiVersion) + client, err := %[3]s.NewResourceManagerClient(sdkApi, %[1]q, defaultApiVersion) if err != nil { return nil, fmt.Errorf("instantiating %[2]s: %%+v", err) } @@ -35,6 +37,6 @@ func New%[2]sWithBaseURI(sdkApi sdkEnv.Api) (*%[2]s, error) { return &%[2]s{ Client: client, }, nil -}`, data.packageName, data.serviceClientName, *copyrightLines) +}`, data.packageName, data.serviceClientName, data.baseClientPackage, *copyrightLines) return &template, nil } diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client.go b/tools/generator-go-sdk/internal/generator/templater_meta_client.go index 1fa156c3b33..cb3d5d92156 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client.go @@ -9,11 +9,12 @@ import ( ) type metaClientTemplater struct { - serviceName string - apiVersion string - resources map[string]models.APIResource - source models.SourceDataOrigin - sourceType models.SourceDataType + serviceName string + apiVersion string + baseClientPackage string + resources map[string]models.APIResource + source models.SourceDataOrigin + sourceType models.SourceDataType } func (m metaClientTemplater) template() (*string, error) { @@ -52,29 +53,32 @@ configureFunc(%[1]s.Client) sort.Strings(fields) sort.Strings(imports) - packageName := fmt.Sprintf("v%s", strings.ReplaceAll(m.apiVersion, "-", "_")) + packageName := m.apiVersion + if strings.Contains(packageName, "-") { + packageName = fmt.Sprintf("v%s", strings.ReplaceAll(m.apiVersion, "-", "_")) + } out := fmt.Sprintf(`package %[1]s -%[2]s +%[3]s import ( "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" sdkEnv "github.com/hashicorp/go-azure-sdk/sdk/environments" - %[3]s + %[4]s ) type Client struct { - %[4]s + %[5]s } -func NewClientWithBaseURI(sdkApi sdkEnv.Api, configureFunc func(c *resourcemanager.Client)) (*Client, error) { - %[5]s +func NewClientWithBaseURI(sdkApi sdkEnv.Api, configureFunc func(c *%[2]s.Client)) (*Client, error) { + %[6]s return &Client{ - %[6]s + %[7]s }, nil } -`, packageName, *copyrightLines, strings.Join(imports, "\n"), strings.Join(fields, "\n"), strings.Join(clientInitialization, "\n"), strings.Join(assignments, "\n")) +`, packageName, m.baseClientPackage, *copyrightLines, strings.Join(imports, "\n"), strings.Join(fields, "\n"), strings.Join(clientInitialization, "\n"), strings.Join(assignments, "\n")) return &out, nil } diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index c6dce5a0b8f..7edc8316eeb 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -45,6 +45,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" "github.com/hashicorp/go-azure-sdk/sdk/client" "github.com/hashicorp/go-azure-sdk/sdk/client/pollers" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" "github.com/hashicorp/go-azure-sdk/sdk/odata" %[4]s @@ -226,12 +227,12 @@ func (c methodsPandoraTemplater) longRunningOperationTemplate(data GeneratorData } templated := fmt.Sprintf(` -%[8]s %[9]s %[10]s +%[11]s -// %[2]s ... -func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, err error) { +// %[3]s ... +func (c %[1]s) %[3]s(ctx context.Context %[4]s) (result %[3]sOperationResponse, err error) { opts := %[4]s req, err := c.Client.NewRequest(ctx, opts) @@ -239,7 +240,7 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, return } - %[5]s + %[6]s var resp *client.Response resp, err = req.Execute(ctx) @@ -251,9 +252,9 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, return } - %[6]s + %[7]s - result.Poller, err = resourcemanager.PollerFromResponse(resp, c.Client) + result.Poller, err = %[2]s.PollerFromResponse(resp, c.Client) if err != nil { return } @@ -261,20 +262,20 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, return } -// %[2]sThenPoll performs %[2]s then polls until it's completed -func (c %[1]s) %[2]sThenPoll(ctx context.Context %[3]s) error { - result, err := c.%[2]s(ctx %[7]s) +// %[3]sThenPoll performs %[3]s then polls until it's completed +func (c %[1]s) %[3]sThenPoll(ctx context.Context %[4]s) error { + result, err := c.%[3]s(ctx %[8]s) if err != nil { - return fmt.Errorf("performing %[2]s: %%+v", err) + return fmt.Errorf("performing %[3]s: %%+v", err) } if err := result.Poller.PollUntilDone(ctx); err != nil { - return fmt.Errorf("polling after %[2]s: %%+v", err) + return fmt.Errorf("polling after %[3]s: %%+v", err) } return nil } -`, data.serviceClientName, c.operationName, *methodArguments, *requestOptions, *marshalerCode, *unmarshalerCode, argumentsCode, *responseStruct, *optionsStruct, requestOptionStruct) +`, data.serviceClientName, data.baseClientPackage, c.operationName, *methodArguments, *requestOptions, *marshalerCode, *unmarshalerCode, argumentsCode, *responseStruct, *optionsStruct, requestOptionStruct) return &templated, nil } From 9b4b6659f126c601c55cb502389f8056ae0c7da6 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 30 Jul 2024 17:52:54 +0100 Subject: [PATCH 022/117] generator-go-sdk: reusable constructor for base clients --- tools/generator-go-sdk/internal/generator/templater_clients.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_clients.go b/tools/generator-go-sdk/internal/generator/templater_clients.go index 85d8dbac1cf..f0977f68315 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients.go @@ -29,7 +29,7 @@ type %[2]s struct { } func New%[2]sWithBaseURI(sdkApi sdkEnv.Api) (*%[2]s, error) { - client, err := %[3]s.NewResourceManagerClient(sdkApi, %[1]q, defaultApiVersion) + client, err := %[3]s.NewClient(sdkApi, %[1]q, defaultApiVersion) if err != nil { return nil, fmt.Errorf("instantiating %[2]s: %%+v", err) } From 1ff7cc32d1e6801b556330b5b4b8c1d3ff3c659b Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 1 Aug 2024 23:59:15 +0100 Subject: [PATCH 023/117] importer-msgraph-metadata: support resource IDs as common types --- .../repository/helpers_parse.go | 26 +++++--- .../repository/internal/helpers/known_data.go | 8 ++- .../repository/internal/models/operations.go | 3 + .../internal/models/resource_ids.go | 2 +- .../stages/common_types_resource_ids.go | 49 ++++++++++++++ .../repository/internal/stages/operations.go | 11 ++-- .../internal/stages/resource_ids.go | 2 +- .../internal/transforms/sdk_operation.go | 2 + .../repository/save_common_types.go | 5 ++ tools/data-api-sdk/v1/models/common_types.go | 9 ++- tools/data-api-sdk/v1/models/sdk_operation.go | 3 + .../components/parser/resourceids.go | 64 +++++++++++++++++- .../internal/cmd/import.go | 1 - .../internal/logging/log.go | 26 +++++++- .../internal/pipeline/interface.go | 8 +-- .../internal/pipeline/pipeline.go | 20 +++--- .../internal/pipeline/run_importer.go | 24 ++++--- .../pipeline/task_parse_resourceids.go | 65 ------------------- .../task_parse_service_resourceids.go | 12 ++++ ...ces.go => task_parse_service_resources.go} | 11 ++-- .../pipeline/task_persist_common_types.go | 3 +- .../internal/pipeline/task_persist_service.go | 3 +- .../pipeline/task_translate_service.go | 15 +++-- .../internal/pipeline/translate_models.go | 19 ++++-- 24 files changed, 262 insertions(+), 129 deletions(-) create mode 100644 tools/data-api-repository/repository/internal/stages/common_types_resource_ids.go delete mode 100644 tools/importer-msgraph-metadata/internal/pipeline/task_parse_resourceids.go create mode 100644 tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go rename tools/importer-msgraph-metadata/internal/pipeline/{task_parse_resources.go => task_parse_service_resources.go} (96%) diff --git a/tools/data-api-repository/repository/helpers_parse.go b/tools/data-api-repository/repository/helpers_parse.go index 3418fe58e90..1bdd07429e9 100644 --- a/tools/data-api-repository/repository/helpers_parse.go +++ b/tools/data-api-repository/repository/helpers_parse.go @@ -57,8 +57,9 @@ func parseCommonTypesWithin(workingDirectory string, logger hclog.Logger) (*map[ apiVersion = strings.TrimPrefix(apiVersion, "/") commonTypes := sdkModels.CommonTypes{ - Constants: map[string]sdkModels.SDKConstant{}, - Models: map[string]sdkModels.SDKModel{}, + Constants: map[string]sdkModels.SDKConstant{}, + Models: map[string]sdkModels.SDKModel{}, + ResourceIDs: make(map[string]sdkModels.ResourceID), } logger.Trace(fmt.Sprintf("Discovering the Common Type Constants within %q..", subDirectory)) @@ -75,7 +76,14 @@ func parseCommonTypesWithin(workingDirectory string, logger hclog.Logger) (*map[ } commonTypes.Models = *models - if len(commonTypes.Constants) > 0 || len(commonTypes.Models) > 0 { + logger.Trace(fmt.Sprintf("Discovering the Common Type Resource IDs within %q..", subDirectory)) + resourceIds, err := parseResourceIDsWithin(subDirectory, *constants, logger) + if err != nil { + return nil, fmt.Errorf("parsing the Common Type Models within %q: %+v", subDirectory, err) + } + commonTypes.ResourceIDs = *resourceIds + + if len(commonTypes.Constants) > 0 || len(commonTypes.Models) > 0 || len(commonTypes.ResourceIDs) > 0 { output[apiVersion] = commonTypes } } @@ -183,15 +191,17 @@ func parseAPIResourceWithin(workingDirectory, resourceName string, commonTypesFo } knownData := helpers.KnownData{ - Constants: *constants, - Models: *models, - ResourceIds: *resourceIds, - CommonTypeConstants: make(map[string]sdkModels.SDKConstant), - CommonTypeModels: make(map[string]sdkModels.SDKModel), + Constants: *constants, + Models: *models, + ResourceIds: *resourceIds, + CommonTypeConstants: make(map[string]sdkModels.SDKConstant), + CommonTypeModels: make(map[string]sdkModels.SDKModel), + CommonTypesResourceIds: make(map[string]sdkModels.ResourceID), } if commonTypesForThisAPIVersion != nil { knownData.CommonTypeConstants = commonTypesForThisAPIVersion.Constants knownData.CommonTypeModels = commonTypesForThisAPIVersion.Models + knownData.CommonTypesResourceIds = commonTypesForThisAPIVersion.ResourceIDs } logger.Trace(fmt.Sprintf("Parsing the Operations within %q..", workingDirectory)) diff --git a/tools/data-api-repository/repository/internal/helpers/known_data.go b/tools/data-api-repository/repository/internal/helpers/known_data.go index f2989f67270..1fcd7a190f5 100644 --- a/tools/data-api-repository/repository/internal/helpers/known_data.go +++ b/tools/data-api-repository/repository/internal/helpers/known_data.go @@ -10,8 +10,9 @@ type KnownData struct { Models map[string]sdkModels.SDKModel ResourceIds map[string]sdkModels.ResourceID - CommonTypeConstants map[string]sdkModels.SDKConstant - CommonTypeModels map[string]sdkModels.SDKModel + CommonTypeConstants map[string]sdkModels.SDKConstant + CommonTypeModels map[string]sdkModels.SDKModel + CommonTypesResourceIds map[string]sdkModels.ResourceID } func (d KnownData) ConstantExists(constantName string) bool { @@ -38,5 +39,6 @@ func (d KnownData) ModelExists(modelName string) bool { func (d KnownData) ResourceIDExists(resourceIdName string) bool { _, resourceIdExists := d.ResourceIds[resourceIdName] - return resourceIdExists + _, commonTypesResourceIdExists := d.CommonTypesResourceIds[resourceIdName] + return resourceIdExists || commonTypesResourceIdExists } diff --git a/tools/data-api-repository/repository/internal/models/operations.go b/tools/data-api-repository/repository/internal/models/operations.go index 2856e976f71..13d52688dc8 100644 --- a/tools/data-api-repository/repository/internal/models/operations.go +++ b/tools/data-api-repository/repository/internal/models/operations.go @@ -32,6 +32,9 @@ type Operation struct { // ResourceIdName specifies the name of the optional Resource ID used for this operation ResourceIdName *string `json:"resourceIdName,omitempty"` + // ResourceIdNameIsCommonType specifies whether the referenced ResourceIdName is a common type + ResourceIdNameIsCommonType *bool `json:"resourceIdNameIsCommonType,omitempty"` + // RequestObject specifies the optional ObjectDefinition to be specified in the Request RequestObject *ObjectDefinition `json:"requestObject,omitempty"` diff --git a/tools/data-api-repository/repository/internal/models/resource_ids.go b/tools/data-api-repository/repository/internal/models/resource_ids.go index dd7ffb62bd7..61fbeb4558d 100644 --- a/tools/data-api-repository/repository/internal/models/resource_ids.go +++ b/tools/data-api-repository/repository/internal/models/resource_ids.go @@ -17,7 +17,7 @@ type ResourceId struct { // during `terraform import` examples. Id string `json:"id"` // TODO: does this want renaming to `ExampleValue` to be clearer? - // Segments specifies the ordered list of ResourceIdSegments which comprise this Resource Id. + // Segments specifies the ordered list of ResourceIdSegments which comprise this ResourceId. // Typically, these comprise Static and UserSpecified Segment Types. Segments []ResourceIdSegment `json:"segments"` } diff --git a/tools/data-api-repository/repository/internal/stages/common_types_resource_ids.go b/tools/data-api-repository/repository/internal/stages/common_types_resource_ids.go new file mode 100644 index 00000000000..c758483ad87 --- /dev/null +++ b/tools/data-api-repository/repository/internal/stages/common_types_resource_ids.go @@ -0,0 +1,49 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package stages + +import ( + "fmt" + "path/filepath" + + "github.com/hashicorp/go-hclog" + "github.com/hashicorp/pandora/tools/data-api-repository/repository/internal/helpers" + "github.com/hashicorp/pandora/tools/data-api-repository/repository/internal/transforms" + sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" +) + +var _ Stage = CommonTypesResourceIDsStage{} + +type CommonTypesResourceIDsStage struct { + // APIVersion specifies the APIVersion within the Service where the Resource IDs exist. + APIVersion string + + // CommonTypesResourceIDs specifies a map of Resource ID Name (key) to ResourceID (value) that should + // be persisted. + CommonTypesResourceIDs map[string]sdkModels.ResourceID +} + +func (g CommonTypesResourceIDsStage) Name() string { + return "Common Types Resource IDs" +} + +func (g CommonTypesResourceIDsStage) Generate(input *helpers.FileSystem, logger hclog.Logger) error { + logger.Debug("Generating Common Types Resource IDs") + for resourceIDName, resourceIDValue := range g.CommonTypesResourceIDs { + logger.Trace(fmt.Sprintf("Generating Resource ID %q", resourceIDName)) + mapped, err := transforms.MapResourceIDToRepository(resourceIDName, resourceIDValue) + if err != nil { + return fmt.Errorf("mapping Resource ID %q: %+v", resourceIDName, err) + } + + // {workingDirectory}/common-types/{APIVersion}/ResourceId-{Name}.json + path := filepath.Join(helpers.CommonTypesDirectoryName, g.APIVersion, fmt.Sprintf("ResourceId-%s.json", resourceIDName)) + logger.Trace(fmt.Sprintf("Staging to %s", path)) + if err = input.Stage(path, *mapped); err != nil { + return fmt.Errorf("staging Resource ID %q: %+v", resourceIDName, err) + } + } + + return nil +} diff --git a/tools/data-api-repository/repository/internal/stages/operations.go b/tools/data-api-repository/repository/internal/stages/operations.go index ddcee120ecb..cd483d0e5d4 100644 --- a/tools/data-api-repository/repository/internal/stages/operations.go +++ b/tools/data-api-repository/repository/internal/stages/operations.go @@ -44,11 +44,12 @@ func (g OperationsStage) Generate(input *helpers.FileSystem, logger hclog.Logger logger.Trace(fmt.Sprintf("Generating Operation %q..", operationName)) knownData := helpers.KnownData{ - Constants: g.Constants, - Models: g.Models, - ResourceIds: map[string]sdkModels.ResourceID{}, - CommonTypeConstants: g.CommonTypesForThisAPIVersion.Constants, - CommonTypeModels: g.CommonTypesForThisAPIVersion.Models, + Constants: g.Constants, + Models: g.Models, + ResourceIds: map[string]sdkModels.ResourceID{}, + CommonTypeConstants: g.CommonTypesForThisAPIVersion.Constants, + CommonTypeModels: g.CommonTypesForThisAPIVersion.Models, + CommonTypesResourceIds: g.CommonTypesForThisAPIVersion.ResourceIDs, } operationDetails := g.Operations[operationName] diff --git a/tools/data-api-repository/repository/internal/stages/resource_ids.go b/tools/data-api-repository/repository/internal/stages/resource_ids.go index e996d0d6633..0a335619fd0 100644 --- a/tools/data-api-repository/repository/internal/stages/resource_ids.go +++ b/tools/data-api-repository/repository/internal/stages/resource_ids.go @@ -39,7 +39,7 @@ func (g ResourceIDsStage) Generate(input *helpers.FileSystem, logger hclog.Logge // {ServiceDirectory}/APIVersion/APIResource/ResourceId-{Name}.json path := filepath.Join(g.APIVersion, g.APIResource, fmt.Sprintf("ResourceId-%s.json", resourceIDName)) logger.Trace(fmt.Sprintf("Staging to %s", path)) - if err := input.Stage(path, *mapped); err != nil { + if err = input.Stage(path, *mapped); err != nil { return fmt.Errorf("staging Resource ID %q: %+v", resourceIDName, err) } } diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_operation.go b/tools/data-api-repository/repository/internal/transforms/sdk_operation.go index efdecaf8644..e82e3e70390 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_operation.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_operation.go @@ -35,6 +35,7 @@ func MapSDKOperationFromRepository(input repositoryModels.Operation, knownData h Options: options, RequestObject: nil, ResourceIDName: nil, + ResourceIDNameIsCommonType: input.ResourceIdNameIsCommonType, ResponseObject: nil, URISuffix: input.UriSuffix, } @@ -78,6 +79,7 @@ func MapSDKOperationToRepository(operationName string, input sdkModels.SDKOperat LongRunning: input.LongRunning, HTTPMethod: strings.ToUpper(input.Method), ResourceIdName: input.ResourceIDName, + ResourceIdNameIsCommonType: input.ResourceIDNameIsCommonType, UriSuffix: input.URISuffix, } diff --git a/tools/data-api-repository/repository/save_common_types.go b/tools/data-api-repository/repository/save_common_types.go index 6823c645f66..4a207b82fed 100644 --- a/tools/data-api-repository/repository/save_common_types.go +++ b/tools/data-api-repository/repository/save_common_types.go @@ -37,6 +37,11 @@ func (r *repositoryImpl) SaveCommonTypes(opts SaveCommonTypesOptions) error { CommonTypeConstants: commonTypes.Constants, CommonTypeModels: commonTypes.Models, }) + + items = append(items, stages.CommonTypesResourceIDsStage{ + APIVersion: apiVersion, + CommonTypesResourceIDs: commonTypes.ResourceIDs, + }) } fs := helpers.NewFileSystem() diff --git a/tools/data-api-sdk/v1/models/common_types.go b/tools/data-api-sdk/v1/models/common_types.go index e612b991dfb..7668d985e7c 100644 --- a/tools/data-api-sdk/v1/models/common_types.go +++ b/tools/data-api-sdk/v1/models/common_types.go @@ -9,10 +9,15 @@ type CommonTypes struct { // Constants specifies a map of Constant Name (key) to SDKConstant (value) which // describes each common Constant supported by this API. // NOTE: the Constant Name is a valid Identifier. - Constants map[string]SDKConstant `json:"constants"` + Constants map[string]SDKConstant // Models specifies a map of Model Name (key) to SDKModel (value) which // describes each common Model supported by this API. // NOTE: the Model Name is a valid Identifier. - Models map[string]SDKModel `json:"models"` + Models map[string]SDKModel + + // ResourceIDs specifies a map of Resource ID Name (key) to SDKOperation (value) + // which contains information about the common ResourceIDs available in this API. + // NOTE: the Resource ID Name is a valid Identifier. + ResourceIDs map[string]ResourceID } diff --git a/tools/data-api-sdk/v1/models/sdk_operation.go b/tools/data-api-sdk/v1/models/sdk_operation.go index 1cea0ceb08a..bd1b022ea4f 100644 --- a/tools/data-api-sdk/v1/models/sdk_operation.go +++ b/tools/data-api-sdk/v1/models/sdk_operation.go @@ -49,6 +49,9 @@ type SDKOperation struct { // 3. {uriSuffix} ResourceIDName *string `json:"resourceIdName"` + // ResourceIDNameIsCommonType specifies whether the referenced ResourceIdName is a common type + ResourceIDNameIsCommonType *bool `json:"resourceIdNameIsCommonType,omitempty"` + // ResponseObject optionally specifies the Object which is expected to be returned in the // HTTP Response. This is represented by an SDKObjectDefinition, which defines the shape // of the object. diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 7fc69527f20..936e1a8ef4b 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -8,9 +8,12 @@ import ( "sort" "strings" + "github.com/getkin/kin-openapi/openapi3" "github.com/hashicorp/go-azure-helpers/lang/pointer" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/tags" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) const ( @@ -62,7 +65,6 @@ func (ri ResourceIds) MatchIdOrAncestor(resourceId ResourceId) (*ResourceIdMatch type ResourceId struct { Name string Service string - Version string Segments []ResourceIdSegment } @@ -420,3 +422,63 @@ func NewResourceId(path string, tags []string) (id ResourceId) { return } + +func ParseResourceIDs(paths openapi3.Paths, serviceName *string) (resourceIds ResourceIds, err error) { + resourceIds = make(ResourceIds, 0) + for path, item := range paths { + operations := item.Operations() + operationTags := make([]string, 0) + + if serviceName != nil { + // Check tags and skip + skip := true + for _, operation := range operations { + if tags.Matches(*serviceName, operation.Tags) { + operationTags = append(operationTags, operation.Tags...) + skip = false + break + } + } + if skip { + continue + } + } + + id := NewResourceId(path, operationTags) + segmentsLastIndex := len(id.Segments) - 1 + + lastSegment := id.Segments[segmentsLastIndex] + if lastSegment.Type == SegmentODataReference { + lastSegment = id.Segments[segmentsLastIndex-1] + truncated := id.TruncateToLastSegmentOfTypeBeforeSegment([]ResourceIdSegmentType{}, segmentsLastIndex) + if truncated == nil { + err = fmt.Errorf("unable to truncate resource ID with OData Reference: %q", id.ID()) + return + } + id = *truncated + } + if lastSegment.Type != SegmentUserValue { + continue + } + + resourceIdName := "" + if r, ok := id.FindResourceIdName(); ok { + resourceIdName = normalize.Singularize(normalize.CleanName(*r)) + } + + if resourceIdName != "" { + logging.Infof(fmt.Sprintf("Found resource ID %q", resourceIdName)) + + id.Name = resourceIdName + + if serviceName != nil { + id.Service = normalize.CleanName(*serviceName) + } + + resourceIds = append(resourceIds, &id) + } + } + + return + +} diff --git a/tools/importer-msgraph-metadata/internal/cmd/import.go b/tools/importer-msgraph-metadata/internal/cmd/import.go index 7ad8189c38f..ce78bd36bd1 100644 --- a/tools/importer-msgraph-metadata/internal/cmd/import.go +++ b/tools/importer-msgraph-metadata/internal/cmd/import.go @@ -65,7 +65,6 @@ func (c ImportCommand) Run(args []string) int { input := pipeline.RunInput{ ProviderPrefix: "azuread", - Logger: logging.Log, ConfigFilePath: c.microsoftGraphConfigPath, MetadataDirectory: c.metadataDirectory, diff --git a/tools/importer-msgraph-metadata/internal/logging/log.go b/tools/importer-msgraph-metadata/internal/logging/log.go index 4fc7607cad8..ac01c2e3f7c 100644 --- a/tools/importer-msgraph-metadata/internal/logging/log.go +++ b/tools/importer-msgraph-metadata/internal/logging/log.go @@ -3,10 +3,34 @@ package logging -import "github.com/hashicorp/go-hclog" +import ( + "fmt" + + "github.com/hashicorp/go-hclog" +) var Log hclog.Logger func init() { Log = hclog.NewNullLogger() } + +func Debugf(msg string, args ...interface{}) { + Log.Debug(fmt.Sprintf(msg, args...)) +} + +func Errorf(msg string, args ...interface{}) { + Log.Error(fmt.Sprintf(msg, args...)) +} + +func Tracef(msg string, args ...interface{}) { + Log.Trace(fmt.Sprintf(msg, args...)) +} + +func Infof(msg string, args ...interface{}) { + Log.Info(fmt.Sprintf(msg, args...)) +} + +func Warnf(msg string, args ...interface{}) { + Log.Warn(fmt.Sprintf(msg, args...)) +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/interface.go b/tools/importer-msgraph-metadata/internal/pipeline/interface.go index 9a79571e452..858b4a94235 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/interface.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/interface.go @@ -6,15 +6,13 @@ package pipeline import ( "fmt" - "github.com/hashicorp/go-hclog" "github.com/hashicorp/pandora/tools/data-api-repository/repository" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) type RunInput struct { ProviderPrefix string - Logger hclog.Logger - CommonTypesDirectoryName string ConfigFilePath string MetadataDirectory string @@ -25,9 +23,7 @@ type RunInput struct { } func Run(input RunInput) error { - logger := input.Logger - - metadataGitSha, err := determineGitSha(input.MetadataDirectory, logger) + metadataGitSha, err := determineGitSha(input.MetadataDirectory, logging.Log) if err != nil { return fmt.Errorf("determining Git SHA at %q: %+v", input.MetadataDirectory, err) } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go b/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go index 425b28ddcde..2a1e64680d7 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go @@ -8,19 +8,19 @@ import ( "github.com/davecgh/go-spew/spew" "github.com/getkin/kin-openapi/openapi3" - "github.com/hashicorp/go-hclog" "github.com/hashicorp/pandora/tools/data-api-repository/repository" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) type pipeline struct { apiVersion string commonTypesForVersion sdkModels.CommonTypes repo repository.Repository - logger hclog.Logger metadataGitSha string outputDirectory string + resourceIds parser.ResourceIds spec *openapi3.T } @@ -37,14 +37,14 @@ func (p pipeline) ForService(serviceName string) pipelineForService { } func (p pipelineForService) RunImport(serviceTags []string, models parser.Models, constants parser.Constants) error { - p.logger.Info(fmt.Sprintf("Parsing resource IDs for %q", p.service)) - resourceIds, err := p.parseResourceIDs() - if err != nil { - return err - } + //logging.Infof(fmt.Sprintf("Parsing resource IDs for %q", p.service)) + //resourceIds, err := p.parseResourceIDs() + //if err != nil { + // return err + //} - p.logger.Info(fmt.Sprintf("Parsing resources for %q", p.service)) - resources, err := p.parseResources(resourceIds, models, constants) + logging.Infof(fmt.Sprintf("Parsing resources for %q", p.service)) + resources, err := p.parseResources(p.resourceIds, models, constants) if err != nil { return err } @@ -62,7 +62,7 @@ func (p pipelineForService) RunImport(serviceTags []string, models parser.Models if len(resource.Paths) > 0 { path = resource.Paths[0].ID() } - p.logger.Warn(spew.Sprintf("Resource with no category was encountered for %q at %q: %#v", p.service, path, *resource)) + logging.Warnf(spew.Sprintf("Resource with no category was encountered for %q at %q: %#v", p.service, path, *resource)) } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go b/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go index bfdbe5b5b92..45903115648 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go @@ -12,18 +12,17 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/tags" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" "github.com/hashicorp/pandora/tools/sdk/config/services" ) func runImporter(input RunInput, metadataGitSha string) error { - logger := input.Logger - config, err := services.LoadFromFile(input.ConfigFilePath) if err != nil { return fmt.Errorf("loading config: %+v", err) } - logger.Debug("Removing any existing API Definitions") + logging.Debugf("Removing any existing API Definitions") if err = input.Repo.PurgeExistingData(sdkModels.MicrosoftGraphMetaDataSourceDataOrigin); err != nil { return fmt.Errorf("removing existing API Definitions: %+v", err) } @@ -34,24 +33,31 @@ func runImporter(input RunInput, metadataGitSha string) error { return err } } - logger.Info("Finished!") + logging.Infof("Finished!") return nil } func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha string, config *services.Config) error { - input.Logger.Info(fmt.Sprintf("Loading OpenAPI3 definitions for API version %q", apiVersion)) + logging.Infof(fmt.Sprintf("Loading OpenAPI3 definitions for API version %q", apiVersion)) spec, err := openapi3.NewLoader().LoadFromFile(filepath.Join(input.MetadataDirectory, openApiFile)) if err != nil { return err } + logging.Infof(fmt.Sprintf("Parsing models and constants...")) models, constants, err := parser.Common(spec.Components.Schemas) if err != nil { return err } - commonTypesForApiVersion, err := translateModelsToDataApiSdkTypes(models, constants) + logging.Infof(fmt.Sprintf("Parsing resource IDs...")) + resourceIds, err := parser.ParseResourceIDs(spec.Paths, nil) + if err != nil { + return err + } + + commonTypesForApiVersion, err := translateModelsToDataApiSdkTypes(models, constants, resourceIds) if err != nil { return err } @@ -64,9 +70,9 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha p := &pipeline{ apiVersion: apiVersion, commonTypesForVersion: *commonTypesForApiVersion, - logger: input.Logger, metadataGitSha: metadataGitSha, outputDirectory: input.OutputDirectory, + resourceIds: resourceIds, repo: input.Repo, spec: spec, } @@ -105,11 +111,13 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } } - input.Logger.Info(fmt.Sprintf("Importing service %q for API version %q", service.Name, version)) + logging.Infof(fmt.Sprintf("Importing service %q for API version %q", service.Name, version)) if err = p.ForService(service.Directory).RunImport(serviceTags[service.Directory], models, constants); err != nil { return err } + + break } } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resourceids.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resourceids.go deleted file mode 100644 index 12025284dd9..00000000000 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resourceids.go +++ /dev/null @@ -1,65 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package pipeline - -import ( - "fmt" - - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/tags" -) - -func (p pipelineForService) parseResourceIDs() (resourceIds parser.ResourceIds, err error) { - resourceIds = make(parser.ResourceIds, 0) - for path, item := range p.spec.Paths { - operations := item.Operations() - operationTags := make([]string, 0) - - // Check tags and skip - skip := true - for _, operation := range operations { - if tags.Matches(p.service, operation.Tags) { - operationTags = append(operationTags, operation.Tags...) - skip = false - } - } - if skip { - continue - } - - id := parser.NewResourceId(path, operationTags) - segmentsLastIndex := len(id.Segments) - 1 - - lastSegment := id.Segments[segmentsLastIndex] - if lastSegment.Type == parser.SegmentODataReference { - lastSegment = id.Segments[segmentsLastIndex-1] - truncated := id.TruncateToLastSegmentOfTypeBeforeSegment([]parser.ResourceIdSegmentType{}, segmentsLastIndex) - if truncated == nil { - err = fmt.Errorf("unable to truncate resource ID with OData Reference (service %q, version %q): %q", p.service, p.apiVersion, id.ID()) - return - } - id = *truncated - } - if lastSegment.Type != parser.SegmentUserValue { - continue - } - - resourceIdName := "" - if r, ok := id.FindResourceIdName(); ok { - resourceIdName = normalize.Singularize(normalize.CleanName(*r)) - } - - if resourceIdName != "" { - p.logger.Info(fmt.Sprintf("Found resource ID %q (service %q, version %q)", resourceIdName, p.service, p.apiVersion)) - - id.Name = resourceIdName - id.Service = normalize.CleanName(p.service) - id.Version = p.apiVersion - resourceIds = append(resourceIds, &id) - } - } - - return -} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go new file mode 100644 index 00000000000..264b614f228 --- /dev/null +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go @@ -0,0 +1,12 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package pipeline + +import ( + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +func (p pipelineForService) parseResourceIDs() (resourceIds parser.ResourceIds, err error) { + return parser.ParseResourceIDs(p.spec.Paths, &p.service) +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resources.go similarity index 96% rename from tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go rename to tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resources.go index 3844f9bbbbd..e33cfd1dbd1 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resources.go @@ -12,6 +12,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/tags" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, models parser.Models, constants parser.Constants) (resources parser.Resources, err error) { @@ -37,7 +38,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model // Determine whether to skip a path containing unsupported segment types for idx, segment := range parsedPath.Segments { if segment.Type == parser.SegmentCast || segment.Type == parser.SegmentFunction { - p.logger.Debug(fmt.Sprintf("Skipping path containing %s at position %d for %q: %v", segment.Type, idx, p.service, path)) + logging.Debugf(fmt.Sprintf("Skipping path containing %s at position %d for %q: %v", segment.Type, idx, p.service, path)) skip = true break } @@ -52,7 +53,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model resourceName = *r } if resourceName == "" { - p.logger.Warn(fmt.Sprintf("Path with unknown name was encountered for %q: %v", p.service, path)) + logging.Warnf(fmt.Sprintf("Path with unknown name was encountered for %q: %v", p.service, path)) continue } @@ -66,7 +67,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model if _, ok := resources[resourceName]; !ok { // Create a new resource if not already encountered - p.logger.Info(fmt.Sprintf("Found new resource %q (category %q, service %q, version %q)", resourceName, resourceCategory, p.service, p.apiVersion)) + logging.Infof(fmt.Sprintf("Found new resource %q (category %q, service %q, version %q)", resourceName, resourceCategory, p.service, p.apiVersion)) resources[resourceName] = &parser.Resource{ Name: resourceName, @@ -110,7 +111,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model if uriSuffix != nil { if uriSuffixParsed := parser.NewResourceId(*uriSuffix, operationTags); uriSuffixParsed.HasUserValue() { - p.logger.Info(fmt.Sprintf("Skipping URI suffix containing user value in resource %q (category %q, service %q, version %q): %q", resourceName, resourceCategory, p.service, p.apiVersion, *uriSuffix)) + logging.Infof(fmt.Sprintf("Skipping URI suffix containing user value in resource %q (category %q, service %q, version %q): %q", resourceName, resourceCategory, p.service, p.apiVersion, *uriSuffix)) continue } } @@ -199,7 +200,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model // Skip unknown operations if operationType == parser.OperationTypeUnknown { - p.logger.Warn(fmt.Sprintf("Skipping unknown operation type for %q: %v", p.service, path)) + logging.Warnf(fmt.Sprintf("Skipping unknown operation type for %q: %v", p.service, path)) continue } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go index 72954721298..07fb3edf7e6 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go @@ -8,10 +8,11 @@ import ( "github.com/hashicorp/pandora/tools/data-api-repository/repository" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) func (p pipeline) PersistCommonTypesDefinitions() error { - p.logger.Info("Persisting Common Types Definitions..") + logging.Infof("Persisting Common Types Definitions..") commonTypes := map[string]sdkModels.CommonTypes{ p.apiVersion: p.commonTypesForVersion, diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go index e2a0e0f62d4..e8658b81214 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go @@ -9,11 +9,12 @@ import ( "github.com/hashicorp/go-azure-helpers/lang/pointer" "github.com/hashicorp/pandora/tools/data-api-repository/repository" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) func (p pipelineForService) persistApiDefinitions(sdkServices map[string]sdkModels.Service, commonTypes map[string]sdkModels.CommonTypes) error { for serviceName, service := range sdkServices { - p.logger.Info(fmt.Sprintf("Persisting API Definitions for Service %q..", serviceName)) + logging.Infof(fmt.Sprintf("Persisting API Definitions for Service %q..", serviceName)) opts := repository.SaveServiceOptions{ CommonTypes: commonTypes, diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index a7b8c623ea2..7652caaf5c8 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -6,6 +6,7 @@ package pipeline import ( "fmt" + "github.com/hashicorp/go-azure-helpers/lang/pointer" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/blacklisted" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" @@ -60,12 +61,13 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta if operation.ResourceId != nil { resourceIdName = &operation.ResourceId.Name - sdkResourceId, err := operation.ResourceId.DataApiSdkResourceId() - if err != nil { - return nil, err - } - - sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category].ResourceIDs[operation.ResourceId.Name] = *sdkResourceId + // No longer output resource IDs per service, they are now common types + //sdkResourceId, err := operation.ResourceId.DataApiSdkResourceId() + //if err != nil { + // return nil, err + //} + // + //sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category].ResourceIDs[operation.ResourceId.Name] = *sdkResourceId } var requestObject *sdkModels.SDKObjectDefinition @@ -170,6 +172,7 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta Options: nil, // TODO request options for odata queries etc RequestObject: requestObject, ResourceIDName: resourceIdName, + ResourceIDNameIsCommonType: pointer.To(true), ResponseObject: responseObject, URISuffix: operation.UriSuffix, } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go index 696baa53c25..350c8ea62ba 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go @@ -11,9 +11,10 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" ) -func translateModelsToDataApiSdkTypes(models parser.Models, constants parser.Constants) (*sdkModels.CommonTypes, error) { +func translateModelsToDataApiSdkTypes(models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) (*sdkModels.CommonTypes, error) { sdkConstantsMap := make(map[string]sdkModels.SDKConstant) sdkModelsMap := make(map[string]sdkModels.SDKModel) + sdkResourceIdsMap := make(map[string]sdkModels.ResourceID) for modelName, model := range models { sdkModel, err := model.DataApiSdkModel(models) @@ -31,15 +32,25 @@ func translateModelsToDataApiSdkTypes(models parser.Models, constants parser.Con constantValues[fmt.Sprintf("_%s", normalize.CleanName(value))] = value } - // TODO support additional types, if there are any sdkConstantsMap[constantName] = sdkModels.SDKConstant{ + // TODO support additional types, if there are any Type: sdkModels.StringSDKConstantType, Values: constantValues, } } + for _, resourceId := range resourceIds { + sdkResourceId, err := resourceId.DataApiSdkResourceId() + if err != nil { + return nil, err + } + + sdkResourceIdsMap[resourceId.Name] = *sdkResourceId + } + return &sdkModels.CommonTypes{ - Constants: sdkConstantsMap, - Models: sdkModelsMap, + Constants: sdkConstantsMap, + Models: sdkModelsMap, + ResourceIDs: sdkResourceIdsMap, }, nil } From ce35d2963bb53f21507035a01f0bd2a90cc27228 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 2 Aug 2024 00:00:57 +0100 Subject: [PATCH 024/117] generator-go-sdk: support resource IDs as common types, fix race conditions for common types generator --- .../generator-go-sdk/internal/cmd/generate.go | 92 +++++++++++++------ .../internal/generator/data.go | 12 +-- .../internal/generator/service.go | 59 ++++++------ .../internal/generator/settings.go | 7 +- .../internal/generator/stage_clients.go | 2 +- .../internal/generator/stage_common_types.go | 27 +++++- .../internal/generator/stage_constants.go | 2 +- .../internal/generator/stage_ids.go | 2 +- .../internal/generator/stage_meta_client.go | 2 +- .../internal/generator/stage_methods.go | 2 +- .../internal/generator/stage_models.go | 2 +- .../internal/generator/stage_predicates.go | 2 +- .../internal/generator/stage_readme.go | 2 +- .../internal/generator/stage_version.go | 2 +- .../internal/generator/templater.go | 8 +- .../internal/generator/templater_constant.go | 2 +- .../generator/templater_meta_client.go | 2 + .../internal/generator/templater_methods.go | 19 +++- .../internal/generator/templater_readme.go | 21 ++++- 19 files changed, 179 insertions(+), 88 deletions(-) diff --git a/tools/generator-go-sdk/internal/cmd/generate.go b/tools/generator-go-sdk/internal/cmd/generate.go index c0ea36444d0..d20e0f76ec3 100644 --- a/tools/generator-go-sdk/internal/cmd/generate.go +++ b/tools/generator-go-sdk/internal/cmd/generate.go @@ -58,24 +58,31 @@ func (g GenerateCommand) Run(args []string) int { }, } - input.settings.UseOldBaseLayerFor( - // @tombuildsstuff: New Services should now use the `hashicorp/go-azure-sdk` base layer by default - // instead of the base layer from `Azure/go-autorest` - as such this list is for compatibility purposes - // with services already used in `terraform-provider-azurerm`. These services will be gradually removed - // from this list to ensure they're migrated across to using `hashicorp/go-azure-sdk`s base layer. - - "FrontDoor", - "RecoveryServicesBackup", // error: generating Service "RecoveryServicesBackup" / Version "2023-04-01" / Resource "Operation": generating methods: templating methods (using hashicorp/go-azure-sdk): templating: building methods: building response struct template: existing model "ValidateOperationResponse" conflicts with the operation response model for "Validate" - "Subscription", - - // @tombuildsstuff: The Key Vault API has an issue where it requires that the EXACT casing returned in the Response - // is sent in the Request to update or remove a Key Vault Access Policy - and using other casings mean the update - // or removal fails - which is tracked in https://github.com/hashicorp/pandora/issues/3229. - // - // After testing it appears that `2023-07-01` doesn't suffer from this problem - as such we're going to leave - // `2023-02-01` on the older base layer and use the newer API Version as a divide to give us a clear migration path. - "KeyVault@2023-02-01", - ) + if g.sourceDataType == models.MicrosoftGraphSourceDataType { + input.settings.VersionsToGenerateCommonTypes = map[string]models.SourceDataOrigin{ + "stable": models.MicrosoftGraphMetaDataSourceDataOrigin, + "beta": models.MicrosoftGraphMetaDataSourceDataOrigin, + } + } else if g.sourceDataType == models.ResourceManagerSourceDataType { + input.settings.UseOldBaseLayerFor( + // @tombuildsstuff: New Services should now use the `hashicorp/go-azure-sdk` base layer by default + // instead of the base layer from `Azure/go-autorest` - as such this list is for compatibility purposes + // with services already used in `terraform-provider-azurerm`. These services will be gradually removed + // from this list to ensure they're migrated across to using `hashicorp/go-azure-sdk`s base layer. + + "FrontDoor", + "RecoveryServicesBackup", // error: generating Service "RecoveryServicesBackup" / Version "2023-04-01" / Resource "Operation": generating methods: templating methods (using hashicorp/go-azure-sdk): templating: building methods: building response struct template: existing model "ValidateOperationResponse" conflicts with the operation response model for "Validate" + "Subscription", + + // @tombuildsstuff: The Key Vault API has an issue where it requires that the EXACT casing returned in the Response + // is sent in the Request to update or remove a Key Vault Access Policy - and using other casings mean the update + // or removal fails - which is tracked in https://github.com/hashicorp/pandora/issues/3229. + // + // After testing it appears that `2023-07-01` doesn't suffer from this problem - as such we're going to leave + // `2023-02-01` on the older base layer and use the newer API Version as a divide to give us a clear migration path. + "KeyVault@2023-02-01", + ) + } var serviceNames string @@ -128,7 +135,8 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { } } - generatorService := generator.NewServiceGenerator(input.settings) + gen := generator.NewGenerator(input.settings) + for serviceName, service := range data.Services { logging.Debugf("Service %q", serviceName) if !service.Generate { @@ -139,6 +147,8 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { wg.Add(1) go func(serviceName string, service models.Service, input GeneratorInput) { defer wg.Done() + var err error + logging.Debugf("Service %q", serviceName) for versionNumber, versionDetails := range service.APIVersions { logging.Debugf(" Version %q", versionNumber) @@ -150,7 +160,7 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { for resourceName, resourceDetails := range versionDetails.Resources { logging.Debugf(" Resource %q", resourceName) - generatorData := generator.ServiceGeneratorInput{ + serviceGeneratorInput := generator.ServiceGeneratorInput{ CommonTypes: commonTypes, OutputDirectory: input.outputDirectory, ResourceDetails: resourceDetails, @@ -163,15 +173,15 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { VersionName: versionNumber, } logging.Debugf("Generating Service %q / Version %q / Resource %q", serviceName, versionNumber, resourceName) - if err := generatorService.Generate(generatorData); err != nil { + if err = gen.Generate(serviceGeneratorInput); err != nil { addErr(fmt.Errorf("generating Service %q / Version %q / Resource %q: %+v", serviceName, versionNumber, resourceName, err)) return } logging.Debugf("Generated Service %q / Version %q / Resource %q", serviceName, versionNumber, resourceName) } - // then output Common Types and Meta Client - generatorData := generator.VersionGeneratorInput{ + // then output the Meta Client + versionGeneratorInput := generator.VersionGeneratorInput{ OutputDirectory: input.outputDirectory, CommonTypes: commonTypes, ServiceName: serviceName, @@ -180,12 +190,12 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { Source: versionDetails.Source, Type: g.sourceDataType, } - generatorData.UseNewBaseLayer = false + versionGeneratorInput.UseNewBaseLayer = false if input.settings.ShouldUseNewBaseLayer(serviceName, versionNumber) { - generatorData.UseNewBaseLayer = true + versionGeneratorInput.UseNewBaseLayer = true } logging.Debugf("Generating Service %q / Version %q", serviceName, versionNumber) - if err := generatorService.GenerateForVersion(generatorData); err != nil { + if err = gen.GenerateForVersion(versionGeneratorInput); err != nil { addErr(fmt.Errorf("generating Service %q / Version %q: %+v", serviceName, versionNumber, err)) return } @@ -194,6 +204,36 @@ func (g GenerateCommand) run(ctx context.Context, input GeneratorInput) error { }(serviceName, service, input) } + for versionNumber, source := range input.settings.VersionsToGenerateCommonTypes { + wg.Add(1) + go func(versionNumber string, source models.SourceDataOrigin, input GeneratorInput) { + defer wg.Done() + var err error + + commonTypes, ok := data.CommonTypes[versionNumber] + if !ok { + logging.Debugf("No Common Types Found / Version %q", versionNumber) + return + } + + // then output Common Types + generatorData := generator.VersionGeneratorInput{ + OutputDirectory: input.outputDirectory, + CommonTypes: commonTypes, + VersionName: versionNumber, + Source: source, + Type: g.sourceDataType, + UseNewBaseLayer: true, + } + logging.Debugf("Generating Common Types / Version %q", versionNumber) + if err = gen.GenerateCommonTypes(generatorData); err != nil { + addErr(fmt.Errorf("generating Common Types / Version %q: %+v", versionNumber, err)) + return + } + logging.Debugf("Generated Common Types / Version %q", versionNumber) + }(versionNumber, source, input) + } + go func() { wg.Wait() waitDone <- struct{}{} diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 9dc47fe8c06..1c31db6f569 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -31,10 +31,6 @@ type GeneratorData struct { // the name of the Client e.g. MyThingClient serviceClientName string - // This is the output path for Resource IDs, which then gets aliased by packages - // for example {workingDir}/service/version/ids - idsOutputPath string - // This is the working directory where files should be output for this specific service // for example {workingDir}/{service}/{version}/{resource} resourceOutputPath string @@ -70,10 +66,6 @@ type GeneratorData struct { // whether this is a data plane SDK (omits certain Resource Manager specific features, currently used in ID parsers) isDataPlane bool - // development feature flag - this requires work in the Resource ID parser to handle name conflicts - // @tombuildsstuff: fix this - useIdAliases bool - // development feature flag - should this service use the new transport layer from `hashicorp/go-azure-sdk` // rather than the existing Autorest base layer? useNewBaseLayer bool @@ -85,7 +77,6 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { versionPackageName := strings.ToLower(i.VersionName) versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionPackageName) - idsPath := filepath.Join(versionOutputPath, "ids") resourceOutputPath := filepath.Join(versionOutputPath, resourcePackageName) useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) @@ -103,7 +94,6 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { commonTypesIncludePath: commonTypesIncludePath, commonTypesPackageName: commonTypesPackageName, constants: i.ResourceDetails.Constants, - idsOutputPath: idsPath, isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), models: i.ResourceDetails.Models, operations: i.ResourceDetails.Operations, @@ -114,7 +104,6 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { servicePackageName: strings.ToLower(i.ServiceName), source: i.Source, sourceType: i.Type, - useIdAliases: false, useNewBaseLayer: useNewBaseLayer, } } @@ -154,6 +143,7 @@ func (i VersionGeneratorInput) generatorData(settings Settings) VersionGenerator constants: i.CommonTypes.Constants, isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), models: i.CommonTypes.Models, + packageName: versionPackageName, servicePackageName: strings.ToLower(i.ServiceName), source: i.Source, sourceType: i.Type, diff --git a/tools/generator-go-sdk/internal/generator/service.go b/tools/generator-go-sdk/internal/generator/service.go index 59aab046bdf..282971dd498 100644 --- a/tools/generator-go-sdk/internal/generator/service.go +++ b/tools/generator-go-sdk/internal/generator/service.go @@ -7,18 +7,17 @@ import ( "fmt" "os" "os/exec" - "path/filepath" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/generator-go-sdk/internal/logging" ) -type ServiceGenerator struct { +type Generator struct { settings Settings } -func NewServiceGenerator(settings Settings) ServiceGenerator { - return ServiceGenerator{ +func NewGenerator(settings Settings) Generator { + return Generator{ settings: settings, } } @@ -36,17 +35,12 @@ type ServiceGeneratorInput struct { VersionName string } -func (s *ServiceGenerator) Generate(input ServiceGeneratorInput) error { +func (s *Generator) Generate(input ServiceGeneratorInput) error { data := input.generatorData(s.settings) if err := cleanAndRecreateWorkingDirectory(data.resourceOutputPath); err != nil { return fmt.Errorf("cleaning/recreating working directory %q: %+v", data.resourceOutputPath, err) } - if data.useIdAliases { - if err := ensureWorkingDirectoryExists(data.idsOutputPath); err != nil { - return fmt.Errorf("ensuring the ids working directory %q exists: %+v", data.idsOutputPath, err) - } - } stages := map[string]func(data GeneratorData) error{ "clients": s.clients, @@ -82,7 +76,26 @@ type VersionGeneratorInput struct { VersionName string } -func (s *ServiceGenerator) GenerateForVersion(input VersionGeneratorInput) error { +func (s *Generator) GenerateForVersion(input VersionGeneratorInput) error { + data := input.generatorData(s.settings) + + stages := map[string]func(data VersionGeneratorData) error{ + "metaClient": s.metaClient, + } + for name, stage := range stages { + logging.Debugf("Running Stage %q..", name) + if err := stage(data); err != nil { + return fmt.Errorf("generating %s: %+v", name, err) + } + } + + runGoFmt(data.versionOutputPath) + runGoImports(data.versionOutputPath) + + return nil +} + +func (s *Generator) GenerateCommonTypes(input VersionGeneratorInput) error { data := input.generatorData(s.settings) if err := cleanAndRecreateWorkingDirectory(data.commonTypesOutputPath); err != nil { @@ -91,7 +104,6 @@ func (s *ServiceGenerator) GenerateForVersion(input VersionGeneratorInput) error stages := map[string]func(data VersionGeneratorData) error{ "commonTypes": s.commonTypes, - "metaClient": s.metaClient, } for name, stage := range stages { logging.Debugf("Running Stage %q..", name) @@ -101,10 +113,7 @@ func (s *ServiceGenerator) GenerateForVersion(input VersionGeneratorInput) error } runGoFmt(data.commonTypesOutputPath) - runGoFmt(data.versionOutputPath) - runGoImports(data.commonTypesOutputPath) - runGoImports(data.versionOutputPath) return nil } @@ -122,22 +131,14 @@ func runGoImports(path string) { } func cleanAndRecreateWorkingDirectory(path string) error { - // first, ensure the directory exists - if err := os.MkdirAll(path, 0777); err != nil { - return fmt.Errorf("creating %q: %+v", path, err) - } - - // determine contents of output directory - pathsToDelete, err := filepath.Glob(filepath.Join(path, "*")) - if err != nil { - return fmt.Errorf("globbing files to delete: %+v", err) + // rm -r + if err := os.RemoveAll(path); err != nil { + return fmt.Errorf("deleting %q: %+v", path, err) } - // delete any contained files and directories - for _, pathToDelete := range pathsToDelete { - if err = os.RemoveAll(pathToDelete); err != nil { - return fmt.Errorf("deleting %q: %w", pathToDelete, err) - } + // then ensure the directory exists + if err := os.MkdirAll(path, 0777); err != nil { + return fmt.Errorf("creating %q: %+v", path, err) } return nil diff --git a/tools/generator-go-sdk/internal/generator/settings.go b/tools/generator-go-sdk/internal/generator/settings.go index 663937a06cc..65b77ce37b9 100644 --- a/tools/generator-go-sdk/internal/generator/settings.go +++ b/tools/generator-go-sdk/internal/generator/settings.go @@ -5,11 +5,14 @@ package generator import ( "fmt" + + "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) type Settings struct { - CommonTypesPackageName string - servicesUsingOldBaseLayer map[string]struct{} + CommonTypesPackageName string + VersionsToGenerateCommonTypes map[string]models.SourceDataOrigin + servicesUsingOldBaseLayer map[string]struct{} } func (s *Settings) UseOldBaseLayerFor(serviceNames ...string) { diff --git a/tools/generator-go-sdk/internal/generator/stage_clients.go b/tools/generator-go-sdk/internal/generator/stage_clients.go index a58a376d935..4dff7710651 100644 --- a/tools/generator-go-sdk/internal/generator/stage_clients.go +++ b/tools/generator-go-sdk/internal/generator/stage_clients.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) clients(data GeneratorData) error { +func (s *Generator) clients(data GeneratorData) error { if data.useNewBaseLayer { if err := s.writeToPathForResource(data.resourceOutputPath, "client.go", clientsTemplater{}, data); err != nil { return fmt.Errorf("templating client (using hashicorp/go-azure-sdk): %+v", err) diff --git a/tools/generator-go-sdk/internal/generator/stage_common_types.go b/tools/generator-go-sdk/internal/generator/stage_common_types.go index b4c4ea7c5c9..9a56273e81c 100644 --- a/tools/generator-go-sdk/internal/generator/stage_common_types.go +++ b/tools/generator-go-sdk/internal/generator/stage_common_types.go @@ -10,7 +10,7 @@ import ( "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) -func (s *ServiceGenerator) commonTypes(data VersionGeneratorData) error { +func (s *Generator) commonTypes(data VersionGeneratorData) error { if len(data.commonTypes.Constants) == 0 && len(data.commonTypes.Models) == 0 { return nil } @@ -44,5 +44,30 @@ func (s *ServiceGenerator) commonTypes(data VersionGeneratorData) error { } } + for idName, resourceData := range data.commonTypes.ResourceIDs { + if resourceData.CommonIDAlias != nil || len(resourceData.Segments) == 0 { + continue + } + + nameWithoutSuffix := strings.TrimSuffix(idName, "Id") // we suffix 'Id' and 'ID' in places + fileNamePrefix := strings.ToLower(nameWithoutSuffix) + pt := resourceIdTemplater{ + name: idName, + resource: resourceData, + constantDetails: data.constants, + } + if err := s.writeToPathForResource(data.commonTypesOutputPath, fmt.Sprintf("id_%s.go", fileNamePrefix), pt, data.GeneratorData); err != nil { + return fmt.Errorf("templating ids: %+v", err) + } + + tpt := resourceIdTestsTemplater{ + resourceName: idName, + resourceData: resourceData, + constantDetails: data.constants, + } + if err := s.writeToPathForResource(data.commonTypesOutputPath, fmt.Sprintf("id_%s_test.go", fileNamePrefix), tpt, data.GeneratorData); err != nil { + return fmt.Errorf("templating tests for id: %+v", err) + } + } return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_constants.go b/tools/generator-go-sdk/internal/generator/stage_constants.go index c863b33790c..cbf1ba89f1c 100644 --- a/tools/generator-go-sdk/internal/generator/stage_constants.go +++ b/tools/generator-go-sdk/internal/generator/stage_constants.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) constants(data GeneratorData) error { +func (s *Generator) constants(data GeneratorData) error { if len(data.constants) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_ids.go b/tools/generator-go-sdk/internal/generator/stage_ids.go index 8381e1511b1..375a895e351 100644 --- a/tools/generator-go-sdk/internal/generator/stage_ids.go +++ b/tools/generator-go-sdk/internal/generator/stage_ids.go @@ -8,7 +8,7 @@ import ( "strings" ) -func (s *ServiceGenerator) ids(data GeneratorData) error { +func (s *Generator) ids(data GeneratorData) error { outputDirectory := data.resourceOutputPath for idName, resourceData := range data.resourceIds { diff --git a/tools/generator-go-sdk/internal/generator/stage_meta_client.go b/tools/generator-go-sdk/internal/generator/stage_meta_client.go index 1736d418978..8e9a98b923b 100644 --- a/tools/generator-go-sdk/internal/generator/stage_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/stage_meta_client.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) metaClient(data VersionGeneratorData) error { +func (s *Generator) metaClient(data VersionGeneratorData) error { if len(data.resources) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_methods.go b/tools/generator-go-sdk/internal/generator/stage_methods.go index 852e37d577f..232349294b2 100644 --- a/tools/generator-go-sdk/internal/generator/stage_methods.go +++ b/tools/generator-go-sdk/internal/generator/stage_methods.go @@ -8,7 +8,7 @@ import ( "strings" ) -func (s *ServiceGenerator) methods(data GeneratorData) error { +func (s *Generator) methods(data GeneratorData) error { for operationName, operation := range data.operations { if data.useNewBaseLayer { diff --git a/tools/generator-go-sdk/internal/generator/stage_models.go b/tools/generator-go-sdk/internal/generator/stage_models.go index 5e43a0ce5d4..7b4ee4e6e0c 100644 --- a/tools/generator-go-sdk/internal/generator/stage_models.go +++ b/tools/generator-go-sdk/internal/generator/stage_models.go @@ -8,7 +8,7 @@ import ( "strings" ) -func (s *ServiceGenerator) models(data GeneratorData) error { +func (s *Generator) models(data GeneratorData) error { if len(data.models) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_predicates.go b/tools/generator-go-sdk/internal/generator/stage_predicates.go index 5db4c06d28a..4e11793118c 100644 --- a/tools/generator-go-sdk/internal/generator/stage_predicates.go +++ b/tools/generator-go-sdk/internal/generator/stage_predicates.go @@ -8,7 +8,7 @@ import ( "sort" ) -func (s *ServiceGenerator) predicates(data GeneratorData) error { +func (s *Generator) predicates(data GeneratorData) error { modelNames := make(map[string]string) for _, operation := range data.operations { if operation.FieldContainingPaginationDetails == nil { diff --git a/tools/generator-go-sdk/internal/generator/stage_readme.go b/tools/generator-go-sdk/internal/generator/stage_readme.go index 4c32b9c02dc..f1cd8779af4 100644 --- a/tools/generator-go-sdk/internal/generator/stage_readme.go +++ b/tools/generator-go-sdk/internal/generator/stage_readme.go @@ -8,7 +8,7 @@ import ( "sort" ) -func (s *ServiceGenerator) readmeFile(data GeneratorData) error { +func (s *Generator) readmeFile(data GeneratorData) error { if len(data.models) == 0 { return nil } diff --git a/tools/generator-go-sdk/internal/generator/stage_version.go b/tools/generator-go-sdk/internal/generator/stage_version.go index e95817a3068..e9b153d4d58 100644 --- a/tools/generator-go-sdk/internal/generator/stage_version.go +++ b/tools/generator-go-sdk/internal/generator/stage_version.go @@ -7,7 +7,7 @@ import ( "fmt" ) -func (s *ServiceGenerator) version(data GeneratorData) error { +func (s *Generator) version(data GeneratorData) error { if err := s.writeToPathForResource(data.resourceOutputPath, "version.go", versionTemplater{}, data); err != nil { return fmt.Errorf("templating version: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater.go b/tools/generator-go-sdk/internal/generator/templater.go index 3f1459afbd9..799730d67dd 100644 --- a/tools/generator-go-sdk/internal/generator/templater.go +++ b/tools/generator-go-sdk/internal/generator/templater.go @@ -19,7 +19,7 @@ type templaterForVersion interface { template() (*string, error) } -func (s *ServiceGenerator) writeToPathForResource(directory, filePath string, templater templaterForResource, data GeneratorData) error { +func (s *Generator) writeToPathForResource(directory, filePath string, templater templaterForResource, data GeneratorData) error { fileContents, err := templater.template(data) if err != nil { return fmt.Errorf("templating: %+v", err) @@ -42,7 +42,7 @@ func (s *ServiceGenerator) writeToPathForResource(directory, filePath string, te return nil } -func (s *ServiceGenerator) writeToPathForVersion(directory, filePath string, templater templaterForVersion) error { +func (s *Generator) writeToPathForVersion(directory, filePath string, templater templaterForVersion) error { fileContents, err := templater.template() if err != nil { return fmt.Errorf("templating: %+v", err) @@ -52,7 +52,9 @@ func (s *ServiceGenerator) writeToPathForVersion(directory, filePath string, tem // remove any existing file if it exists _ = os.Remove(fullFilePath) - file, err := os.Create(fullFilePath) + + // call os.OpenFile instead of os.Create to reduce spurious "file not found" errors when O_TRUNC is used + file, err := os.OpenFile(fullFilePath, os.O_WRONLY|os.O_CREATE, 0666) defer file.Close() if err != nil { return fmt.Errorf("opening %q: %+v", fullFilePath, err) diff --git a/tools/generator-go-sdk/internal/generator/templater_constant.go b/tools/generator-go-sdk/internal/generator/templater_constant.go index 19a796d4338..1540fed8d9b 100644 --- a/tools/generator-go-sdk/internal/generator/templater_constant.go +++ b/tools/generator-go-sdk/internal/generator/templater_constant.go @@ -124,7 +124,7 @@ func PossibleValuesFor%[1]s() []%[2]s { } func (t constantTemplater) parseFunction() (*string, error) { - // we only output the parse function if it's a String constant (in which case it'll be needed by the Normalizaation function) + // we only output the parse function if it's a String constant (in which case it'll be needed by the Normalization function) // or if the Constant is used in a Resource ID segment (since that calls this to parse the Resource ID segment) // we could output this always, but this reduces the TLOC we output, which is preferable. requiresParseFunction := t.details.Type == models.StringSDKConstantType || t.constantUsedInAResourceID diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client.go b/tools/generator-go-sdk/internal/generator/templater_meta_client.go index cb3d5d92156..00726c222ab 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client.go @@ -63,6 +63,8 @@ configureFunc(%[1]s.Client) %[3]s import ( + "github.com/hashicorp/go-azure-sdk/sdk/client" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" sdkEnv "github.com/hashicorp/go-azure-sdk/sdk/environments" %[4]s diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 7edc8316eeb..1cd764babb9 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -467,9 +467,22 @@ func (c methodsPandoraTemplater) argumentsTemplateForMethod(data GeneratorData) arguments := make([]string, 0) if c.operation.ResourceIDName != nil { idName := *c.operation.ResourceIDName - id, ok := data.resourceIds[idName] - if !ok { - return nil, fmt.Errorf("internal error: Resource ID %q was not found", idName) + var id models.ResourceID + if pointer.From(c.operation.ResourceIDNameIsCommonType) { + if data.commonTypesPackageName == nil { + return nil, fmt.Errorf("internal error: Common Type Resource ID %q encountered, but `commonTypesPackageName` was nil", idName) + } + var ok bool + if id, ok = data.commonTypes.ResourceIDs[idName]; !ok { + return nil, fmt.Errorf("internal error: Common Type Resource ID %q was not found", idName) + } + + idName = fmt.Sprintf("%s.%s", *data.commonTypesPackageName, idName) + } else { + var ok bool + if id, ok = data.resourceIds[idName]; !ok { + return nil, fmt.Errorf("internal error: Resource ID %q was not found", idName) + } } if id.CommonIDAlias != nil { idName = fmt.Sprintf("commonids.%sId", *id.CommonIDAlias) diff --git a/tools/generator-go-sdk/internal/generator/templater_readme.go b/tools/generator-go-sdk/internal/generator/templater_readme.go index b4cfa44960d..4e060b98e85 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme.go @@ -8,6 +8,7 @@ import ( "sort" "strings" + "github.com/hashicorp/go-azure-helpers/lang/pointer" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/helpers" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) @@ -109,9 +110,23 @@ func (r readmeTemplater) exampleUsageForOperation(packageName, clientName, opera } func (r readmeTemplater) resourceIdInitialization(operation models.SDKOperation, data GeneratorData) (*string, error) { - resourceId, ok := data.resourceIds[*operation.ResourceIDName] - if !ok { - return nil, fmt.Errorf("resource id %q was not found", *operation.ResourceIDName) + var resourceId models.ResourceID + idName := *operation.ResourceIDName + if pointer.From(operation.ResourceIDNameIsCommonType) { + if data.commonTypesPackageName == nil { + return nil, fmt.Errorf("internal error: Common Type Resource ID %q encountered, but `commonTypesPackageName` was nil", idName) + } + var ok bool + if resourceId, ok = data.commonTypes.ResourceIDs[idName]; !ok { + return nil, fmt.Errorf("internal error: Common Type Resource ID %q was not found", idName) + } + + idName = fmt.Sprintf("%s.%s", *data.commonTypesPackageName, idName) + } else { + var ok bool + if resourceId, ok = data.resourceIds[idName]; !ok { + return nil, fmt.Errorf("internal error: Resource ID %q was not found", idName) + } } resourceIdPackageName := data.packageName From 9490e9ee945310fd902142fd1c2fdc40aa56e951 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 2 Aug 2024 18:40:01 +0100 Subject: [PATCH 025/117] importer-msgraph-metadata: improve service translation, ensure ReferenceNameIsCommonType is set for model fields --- .../repository/internal/stages/models.go | 7 +- .../repository/save_service.go | 9 +- .../components/parser/types.go | 21 ++-- .../internal/pipeline/pipeline.go | 67 ++-------- .../internal/pipeline/run_importer.go | 117 ++++++++++++++---- ...e_resources.go => task_parse_resources.go} | 4 + .../task_parse_service_resourceids.go | 12 -- .../pipeline/task_persist_common_types.go | 4 +- .../internal/pipeline/task_persist_service.go | 24 ++-- .../pipeline/task_translate_service.go | 60 +++++---- .../internal/pipeline/translate_models.go | 13 +- 11 files changed, 177 insertions(+), 161 deletions(-) rename tools/importer-msgraph-metadata/internal/pipeline/{task_parse_service_resources.go => task_parse_resources.go} (99%) delete mode 100644 tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go diff --git a/tools/data-api-repository/repository/internal/stages/models.go b/tools/data-api-repository/repository/internal/stages/models.go index 9c2c07303a6..baa6d5bab43 100644 --- a/tools/data-api-repository/repository/internal/stages/models.go +++ b/tools/data-api-repository/repository/internal/stages/models.go @@ -22,6 +22,9 @@ type ModelsStage struct { // APIResource specifies the APIResource within the APIVersion where the Models exist. APIResource string + // CommonTypesForThisAPIVersion specifies the known CommonTypes for this APIVersion. + CommonTypesForThisAPIVersion sdkModels.CommonTypes + // Constants specifies the map of Constant Name (key) to SDKConstant (value) which should be // persisted. Constants map[string]sdkModels.SDKConstant @@ -50,8 +53,8 @@ func (g ModelsStage) Generate(input *helpers.FileSystem, logger hclog.Logger) er Constants: g.Constants, Models: g.Models, ResourceIds: map[string]sdkModels.ResourceID{}, - CommonTypeConstants: make(map[string]sdkModels.SDKConstant), - CommonTypeModels: make(map[string]sdkModels.SDKModel), + CommonTypeConstants: g.CommonTypesForThisAPIVersion.Constants, + CommonTypeModels: g.CommonTypesForThisAPIVersion.Models, } mapped, err := transforms.MapSDKModelToRepository(modelName, modelValue, parent, knownData) diff --git a/tools/data-api-repository/repository/save_service.go b/tools/data-api-repository/repository/save_service.go index 33122ee3dbf..4299f6d9401 100644 --- a/tools/data-api-repository/repository/save_service.go +++ b/tools/data-api-repository/repository/save_service.go @@ -71,10 +71,11 @@ func (r *repositoryImpl) SaveService(opts SaveServiceOptions) error { }) items = append(items, stages.ModelsStage{ - APIVersion: apiVersion, - APIResource: apiResourceName, - Constants: apiResourceDetails.Constants, - Models: apiResourceDetails.Models, + APIVersion: apiVersion, + APIResource: apiResourceName, + CommonTypesForThisAPIVersion: commonTypesForThisApiVersion, + Constants: apiResourceDetails.Constants, + Models: apiResourceDetails.Models, }) items = append(items, stages.OperationsStage{ diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 7c9ea68c815..f31f9c1bca7 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -49,12 +49,15 @@ func (m Models) Found(modelName string) bool { } // MergeDependants inspects the named model in m, then traverses allModels and appends any dependant models to m, recursively -func (m Models) MergeDependants(allModels Models, modelName string) error { +func (m Models) MergeDependants(allModels Models, modelName string, includeCommon bool) error { if !allModels.Found(modelName) { return fmt.Errorf("model not found: %q", modelName) } if _, ok := m[modelName]; !ok { + if !includeCommon && allModels[modelName].Common { + return nil + } m[modelName] = allModels[modelName] } @@ -71,7 +74,7 @@ func (m Models) MergeDependants(allModels Models, modelName string) error { return fmt.Errorf("dependant model not found: %q", modelName) } - if err := m.MergeDependants(allModels, *field.ModelName); err != nil { + if err := m.MergeDependants(allModels, *field.ModelName, includeCommon); err != nil { return err } } @@ -178,9 +181,10 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj } return &sdkModels.SDKObjectDefinition{ - NestedItem: nil, - ReferenceName: f.ModelName, - Type: sdkModels.ReferenceSDKObjectDefinitionType, + NestedItem: nil, + ReferenceName: f.ModelName, + ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), + Type: sdkModels.ReferenceSDKObjectDefinitionType, }, nil case DataTypeArray: @@ -191,9 +195,10 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - NestedItem: nil, - ReferenceName: f.ModelName, - Type: sdkModels.ReferenceSDKObjectDefinitionType, + NestedItem: nil, + ReferenceName: f.ModelName, + ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), + Type: sdkModels.ReferenceSDKObjectDefinitionType, }, ReferenceName: nil, Type: sdkModels.ListSDKObjectDefinitionType, diff --git a/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go b/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go index 2a1e64680d7..d13cf8a3e7a 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go @@ -4,24 +4,21 @@ package pipeline import ( - "fmt" - - "github.com/davecgh/go-spew/spew" "github.com/getkin/kin-openapi/openapi3" "github.com/hashicorp/pandora/tools/data-api-repository/repository" - sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) type pipeline struct { - apiVersion string - commonTypesForVersion sdkModels.CommonTypes - repo repository.Repository - metadataGitSha string - outputDirectory string - resourceIds parser.ResourceIds - spec *openapi3.T + apiVersion string + constants parser.Constants + repo repository.Repository + metadataGitSha string + models parser.Models + outputDirectory string + resources map[string]parser.Resources + resourceIds parser.ResourceIds + spec *openapi3.T } type pipelineForService struct { @@ -35,49 +32,3 @@ func (p pipeline) ForService(serviceName string) pipelineForService { service: serviceName, } } - -func (p pipelineForService) RunImport(serviceTags []string, models parser.Models, constants parser.Constants) error { - //logging.Infof(fmt.Sprintf("Parsing resource IDs for %q", p.service)) - //resourceIds, err := p.parseResourceIDs() - //if err != nil { - // return err - //} - - logging.Infof(fmt.Sprintf("Parsing resources for %q", p.service)) - resources, err := p.parseResources(p.resourceIds, models, constants) - if err != nil { - return err - } - if len(resources) == 0 { - return nil - } - - // Consistency checks for discovered resources - for resourceName, resource := range resources { - if resource == nil { - return fmt.Errorf("nil resource named %q was encountered for %q", resourceName, p.service) - } - if resource.Category == "" { - path := "(no path)" - if len(resource.Paths) > 0 { - path = resource.Paths[0].ID() - } - logging.Warnf(spew.Sprintf("Resource with no category was encountered for %q at %q: %#v", p.service, path, *resource)) - } - } - - sdkServices, err := p.translateServiceToDataApiSdkTypes(models, constants, resources) - if err != nil { - return err - } - - commonTypes := map[string]sdkModels.CommonTypes{ - p.apiVersion: p.commonTypesForVersion, - } - - if err = p.persistApiDefinitions(*sdkServices, commonTypes); err != nil { - return err - } - - return nil -} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go b/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go index 45903115648..f231371fae6 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go @@ -7,6 +7,7 @@ import ( "fmt" "path/filepath" + "github.com/davecgh/go-spew/spew" "github.com/getkin/kin-openapi/openapi3" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" @@ -39,48 +40,39 @@ func runImporter(input RunInput, metadataGitSha string) error { } func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha string, config *services.Config) error { + var err error + + p := &pipeline{ + apiVersion: apiVersion, + metadataGitSha: metadataGitSha, + outputDirectory: input.OutputDirectory, + resources: make(map[string]parser.Resources), + repo: input.Repo, + } + logging.Infof(fmt.Sprintf("Loading OpenAPI3 definitions for API version %q", apiVersion)) - spec, err := openapi3.NewLoader().LoadFromFile(filepath.Join(input.MetadataDirectory, openApiFile)) + p.spec, err = openapi3.NewLoader().LoadFromFile(filepath.Join(input.MetadataDirectory, openApiFile)) if err != nil { return err } logging.Infof(fmt.Sprintf("Parsing models and constants...")) - models, constants, err := parser.Common(spec.Components.Schemas) + p.models, p.constants, err = parser.Common(p.spec.Components.Schemas) if err != nil { return err } logging.Infof(fmt.Sprintf("Parsing resource IDs...")) - resourceIds, err := parser.ParseResourceIDs(spec.Paths, nil) - if err != nil { - return err - } - - commonTypesForApiVersion, err := translateModelsToDataApiSdkTypes(models, constants, resourceIds) + p.resourceIds, err = parser.ParseResourceIDs(p.spec.Paths, nil) if err != nil { return err } - serviceTags, err := tags.Parse(spec.Tags) + serviceTags, err := tags.Parse(p.spec.Tags) if err != nil { return err } - p := &pipeline{ - apiVersion: apiVersion, - commonTypesForVersion: *commonTypesForApiVersion, - metadataGitSha: metadataGitSha, - outputDirectory: input.OutputDirectory, - resourceIds: resourceIds, - repo: input.Repo, - spec: spec, - } - - if err = p.PersistCommonTypesDefinitions(); err != nil { - return err - } - for _, service := range config.Services { for _, version := range service.Available { if version == apiVersion { @@ -113,7 +105,7 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha logging.Infof(fmt.Sprintf("Importing service %q for API version %q", service.Name, version)) - if err = p.ForService(service.Directory).RunImport(serviceTags[service.Directory], models, constants); err != nil { + if err = p.ForService(service.Directory).RunImport(); err != nil { return err } @@ -122,5 +114,82 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } } + // Determine which resource IDs were actually used in resources + usedResourceIds := make(map[string]parser.ResourceId) + for _, resources := range p.resources { + for _, resource := range resources { + for _, operation := range resource.Operations { + if operation.ResourceId != nil { + usedResourceIds[operation.ResourceId.Name] = *operation.ResourceId + } + } + } + } + resourceIds := make(parser.ResourceIds, 0, len(usedResourceIds)) + for _, resourceId := range usedResourceIds { + resourceIds = append(resourceIds, &resourceId) + } + + commonTypesForApiVersion, err := translateCommonTypesToDataApiSdkTypes(p.models, p.constants, resourceIds) + if err != nil { + return err + } + + for service := range p.resources { + if err = p.ForService(service).PersistDefinitions(*commonTypesForApiVersion); err != nil { + return err + } + } + + if err = p.PersistCommonTypesDefinitions(*commonTypesForApiVersion); err != nil { + return err + } + + return nil +} + +func (p pipelineForService) RunImport() error { + logging.Infof(fmt.Sprintf("Parsing resources for %q", p.service)) + resources, err := p.parseResources(p.resourceIds, p.models, p.constants) + if err != nil { + return err + } + if len(resources) == 0 { + return nil + } + + p.resources[p.service] = resources + + // Consistency checks for discovered resources + for resourceName, resource := range resources { + if resource == nil { + return fmt.Errorf("nil resource named %q was encountered for %q", resourceName, p.service) + } + if resource.Category == "" { + path := "(no path)" + if len(resource.Paths) > 0 { + path = resource.Paths[0].ID() + } + logging.Warnf(spew.Sprintf("Resource with no category was encountered for %q at %q: %#v", p.service, path, *resource)) + } + } + + return nil +} + +func (p pipelineForService) PersistDefinitions(commonTypesForVersion sdkModels.CommonTypes) error { + sdkService, err := p.translateServiceToDataApiSdkTypes() + if err != nil { + return err + } + + commonTypes := map[string]sdkModels.CommonTypes{ + p.apiVersion: commonTypesForVersion, + } + + if err = p.persistApiDefinitions(*sdkService, commonTypes); err != nil { + return err + } + return nil } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go similarity index 99% rename from tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resources.go rename to tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index e33cfd1dbd1..6567b1b981f 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -32,6 +32,10 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + if skip { + continue + } + parsedPath := parser.NewResourceId(path, operationTags) lastSegment := parsedPath.Segments[len(parsedPath.Segments)-1] diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go deleted file mode 100644 index 264b614f228..00000000000 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_service_resourceids.go +++ /dev/null @@ -1,12 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package pipeline - -import ( - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -func (p pipelineForService) parseResourceIDs() (resourceIds parser.ResourceIds, err error) { - return parser.ParseResourceIDs(p.spec.Paths, &p.service) -} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go index 07fb3edf7e6..64d046ea11f 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_common_types.go @@ -11,11 +11,11 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -func (p pipeline) PersistCommonTypesDefinitions() error { +func (p pipeline) PersistCommonTypesDefinitions(commonTypesForVersion sdkModels.CommonTypes) error { logging.Infof("Persisting Common Types Definitions..") commonTypes := map[string]sdkModels.CommonTypes{ - p.apiVersion: p.commonTypesForVersion, + p.apiVersion: commonTypesForVersion, } opts := repository.SaveCommonTypesOptions{ diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go index e8658b81214..843b4c2d29c 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_persist_service.go @@ -12,21 +12,19 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -func (p pipelineForService) persistApiDefinitions(sdkServices map[string]sdkModels.Service, commonTypes map[string]sdkModels.CommonTypes) error { - for serviceName, service := range sdkServices { - logging.Infof(fmt.Sprintf("Persisting API Definitions for Service %q..", serviceName)) +func (p pipelineForService) persistApiDefinitions(sdkService sdkModels.Service, commonTypes map[string]sdkModels.CommonTypes) error { + logging.Infof(fmt.Sprintf("Persisting API Definitions for Service %q..", sdkService.Name)) - opts := repository.SaveServiceOptions{ - CommonTypes: commonTypes, - Service: service, - ServiceName: serviceName, - SourceCommitSHA: pointer.To(p.metadataGitSha), - SourceDataOrigin: sdkModels.MicrosoftGraphMetaDataSourceDataOrigin, - } + opts := repository.SaveServiceOptions{ + CommonTypes: commonTypes, + Service: sdkService, + ServiceName: sdkService.Name, + SourceCommitSHA: pointer.To(p.metadataGitSha), + SourceDataOrigin: sdkModels.MicrosoftGraphMetaDataSourceDataOrigin, + } - if err := p.repo.SaveService(opts); err != nil { - return fmt.Errorf("persisting Data API Definitions for Service %q: %+v", serviceName, err) - } + if err := p.repo.SaveService(opts); err != nil { + return fmt.Errorf("persisting Data API Definitions for Service %q: %+v", sdkService.Name, err) } return nil diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index 7652caaf5c8..a6fd177c657 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -14,27 +14,23 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" ) -func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, constants parser.Constants, resources parser.Resources) (*map[string]sdkModels.Service, error) { - sdkServices := make(map[string]sdkModels.Service) +func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Service, error) { + sdkService := sdkModels.Service{ + APIVersions: make(map[string]sdkModels.APIVersion), + Generate: true, + Name: normalize.CleanName(p.service), + ResourceProvider: nil, + TerraformDefinition: nil, + } - for _, resource := range resources { + for _, resource := range p.resources[p.service] { if blacklisted.Resource(resource) { continue } - // First scaffold all discovered services, version(s) and resources (categories) - if _, ok := sdkServices[resource.Service]; !ok { - sdkServices[resource.Service] = sdkModels.Service{ - APIVersions: make(map[string]sdkModels.APIVersion), - Generate: true, - Name: resource.Service, - ResourceProvider: nil, - TerraformDefinition: nil, - } - } - - if _, ok := sdkServices[resource.Service].APIVersions[resource.Version]; !ok { - sdkServices[resource.Service].APIVersions[resource.Version] = sdkModels.APIVersion{ + // First scaffold the version and resources (categories) + if _, ok := sdkService.APIVersions[resource.Version]; !ok { + sdkService.APIVersions[resource.Version] = sdkModels.APIVersion{ APIVersion: resource.Version, Generate: true, Preview: versions.IsPreview(resource.Version), @@ -43,8 +39,8 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta } } - if _, ok := sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category]; !ok { - sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category] = sdkModels.APIResource{ + if _, ok := sdkService.APIVersions[resource.Version].Resources[resource.Category]; !ok { + sdkService.APIVersions[resource.Version].Resources[resource.Category] = sdkModels.APIResource{ Constants: make(map[string]sdkModels.SDKConstant), Models: make(map[string]sdkModels.SDKModel), Operations: make(map[string]sdkModels.SDKOperation), @@ -74,16 +70,16 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta requestObjectIsCommonType := true if operation.RequestModel != nil { - if !models.Found(*operation.RequestModel) { + if !p.models.Found(*operation.RequestModel) { return nil, fmt.Errorf("request model %q was not found for operation: %s", *operation.RequestModel, operation.Name) } - if model := models[*operation.RequestModel]; !model.IsValid() { + if model := p.models[*operation.RequestModel]; !model.IsValid() { return nil, fmt.Errorf("request model %q was invalid for operation: %s", *operation.RequestModel, operation.Name) } else if !model.Common { requestObjectIsCommonType = false - if err := serviceModels.MergeDependants(models, *operation.RequestModel); err != nil { + if err := serviceModels.MergeDependants(p.models, *operation.RequestModel, false); err != nil { return nil, err } } @@ -106,18 +102,18 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta if response.Type != nil && *response.Type == parser.DataTypeModel && response.ModelName != nil { modelName := *response.ModelName - if !models.Found(modelName) { + if !p.models.Found(modelName) { return nil, fmt.Errorf("response model %q was not found for operation: %s", modelName, operation.Name) } - model := models[modelName] + model := p.models[modelName] if !model.IsValid() { return nil, fmt.Errorf("response model %q was invalid for operation: %s", modelName, operation.Name) } else if !model.Common { responseObjectIsCommonType = false - if err := serviceModels.MergeDependants(models, modelName); err != nil { + if err := serviceModels.MergeDependants(p.models, modelName, false); err != nil { return nil, err } } @@ -129,12 +125,12 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta responseObjectIsCommonType = true modelName = *value.ModelName - if !models.Found(modelName) { + if !p.models.Found(modelName) { return nil, fmt.Errorf("nested response model %q was not found for operation: %s", modelName, operation.Name) } else if !model.Common { responseObjectIsCommonType = false - if err := serviceModels.MergeDependants(models, modelName); err != nil { + if err := serviceModels.MergeDependants(p.models, modelName, false); err != nil { return nil, err } } @@ -163,7 +159,7 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta } } - sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category].Operations[operation.Name] = sdkModels.SDKOperation{ + sdkService.APIVersions[resource.Version].Resources[resource.Category].Operations[operation.Name] = sdkModels.SDKOperation{ ContentType: contentType, ExpectedStatusCodes: expectedStatusCodes, FieldContainingPaginationDetails: operation.PaginationField, @@ -179,24 +175,24 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta } for modelName, model := range serviceModels { - sdkModel, err := model.DataApiSdkModel(models) + sdkModel, err := model.DataApiSdkModel(p.models) if err != nil { return nil, err } - sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category].Models[modelName] = *sdkModel + sdkService.APIVersions[resource.Version].Resources[resource.Category].Models[modelName] = *sdkModel for _, field := range model.Fields { if field.ConstantName != nil { constantValues := make(map[string]string) - if constant, ok := constants[*field.ConstantName]; ok { + if constant, ok := p.constants[*field.ConstantName]; ok { for _, value := range constant.Enum { constantValues[normalize.CleanName(value)] = value } } // TODO support additional types, if there are any - sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category].Constants[*field.ConstantName] = sdkModels.SDKConstant{ + sdkService.APIVersions[resource.Version].Resources[resource.Category].Constants[*field.ConstantName] = sdkModels.SDKConstant{ Type: sdkModels.StringSDKConstantType, Values: constantValues, } @@ -205,5 +201,5 @@ func (p pipeline) translateServiceToDataApiSdkTypes(models parser.Models, consta } } - return &sdkServices, nil + return &sdkService, nil } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go index 350c8ea62ba..8ae5de0cc27 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go @@ -11,18 +11,19 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" ) -func translateModelsToDataApiSdkTypes(models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) (*sdkModels.CommonTypes, error) { +func translateCommonTypesToDataApiSdkTypes(models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) (*sdkModels.CommonTypes, error) { sdkConstantsMap := make(map[string]sdkModels.SDKConstant) sdkModelsMap := make(map[string]sdkModels.SDKModel) sdkResourceIdsMap := make(map[string]sdkModels.ResourceID) for modelName, model := range models { - sdkModel, err := model.DataApiSdkModel(models) - if err != nil { - return nil, err + if model.Common { + sdkModel, err := model.DataApiSdkModel(models) + if err != nil { + return nil, err + } + sdkModelsMap[modelName] = *sdkModel } - sdkModelsMap[modelName] = *sdkModel - } for constantName, constant := range constants { From e23cfda232d89e8f88f359cc445003a98bbb537d Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 4 Aug 2024 00:57:55 +0100 Subject: [PATCH 026/117] data-api: support for native odata options --- .../repository/internal/models/operations.go | 3 +++ .../sdk_operation_object_definition.go | 21 ++++++++++++------- .../transforms/sdk_operation_option.go | 2 ++ .../golang_type_for_sdk_object_definition.go | 5 +---- .../v1/models/sdk_operation_option.go | 3 +++ 5 files changed, 23 insertions(+), 11 deletions(-) diff --git a/tools/data-api-repository/repository/internal/models/operations.go b/tools/data-api-repository/repository/internal/models/operations.go index 13d52688dc8..7114e73a658 100644 --- a/tools/data-api-repository/repository/internal/models/operations.go +++ b/tools/data-api-repository/repository/internal/models/operations.go @@ -52,6 +52,9 @@ type Option struct { // (e.g. `If-Match`, `x-ms-client-request-id`) HeaderName *string `json:"headerName,omitempty"` + // ODataFieldName specifies the name for the OData query string parameter associated with this Option. + ODataFieldName *string `json:"odataFieldName,omitempty"` + // Optional specifies whether this Option could be specified in the Request // NOTE: this is intentionally not inferred from Required to allow retrieving HTTP Header // and QueryString values in the Response in the future. diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go b/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go index ba3b75e491d..88e1d1cf707 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go @@ -5,6 +5,7 @@ package transforms import ( "fmt" + "strings" "github.com/hashicorp/pandora/tools/data-api-repository/repository/internal/helpers" repositoryModels "github.com/hashicorp/pandora/tools/data-api-repository/repository/internal/models" @@ -24,13 +25,15 @@ func mapSDKOperationOptionObjectDefinitionFromRepository(input repositoryModels. } if input.ReferenceName != nil { - isConstant := knownData.ConstantExists(*input.ReferenceName) - isModel := knownData.ModelExists(*input.ReferenceName) - if !isConstant && !isModel { - return nil, fmt.Errorf("the Reference %q was not found as either a Constant or a Model", *input.ReferenceName) - } - if isConstant && isModel { - return nil, fmt.Errorf("the Reference %q was found as both a Constant or a Model", *input.ReferenceName) + if !strings.HasPrefix(*input.ReferenceName, "odata.") { + isConstant := knownData.ConstantExists(*input.ReferenceName) + isModel := knownData.ModelExists(*input.ReferenceName) + if !isConstant && !isModel { + return nil, fmt.Errorf("the Reference %q was not found as either a Constant or a Model", *input.ReferenceName) + } + if isConstant && isModel { + return nil, fmt.Errorf("the Reference %q was found as both a Constant or a Model", *input.ReferenceName) + } } output.ReferenceName = input.ReferenceName @@ -93,6 +96,10 @@ func validateSDKOperationOptionObjectDefinition(input repositoryModels.OptionObj return fmt.Errorf("a Reference must be specified for a %q type but didn't get one", string(input.Type)) } + if strings.HasPrefix(*input.ReferenceName, "odata.") { + return nil + } + isConstant := knownData.ConstantExists(*input.ReferenceName) isModel := knownData.ModelExists(*input.ReferenceName) if !isConstant && !isModel { diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_operation_option.go b/tools/data-api-repository/repository/internal/transforms/sdk_operation_option.go index 36dfa7d5b02..5af0009d20f 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_operation_option.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_operation_option.go @@ -19,6 +19,7 @@ func mapSDKOperationOptionFromRepository(input repositoryModels.Option, knownDat return &sdkModels.SDKOperationOption{ HeaderName: input.HeaderName, + ODataFieldName: input.ODataFieldName, QueryStringName: input.QueryString, ObjectDefinition: *objectDefinition, Required: input.Required, @@ -33,6 +34,7 @@ func mapSDKOperationOptionToRepository(fieldName string, input sdkModels.SDKOper option := repositoryModels.Option{ HeaderName: input.HeaderName, + ODataFieldName: input.ODataFieldName, QueryString: input.QueryStringName, Field: fieldName, ObjectDefinition: *objectDefinition, diff --git a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go index 0ea495b912e..35aa1ae18d9 100644 --- a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go +++ b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go @@ -60,10 +60,7 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa out := *input.ReferenceName if golangPackageName != nil { out = fmt.Sprintf("%s.%s", *golangPackageName, out) - } else if input.ReferenceNameIsCommonType != nil && *input.ReferenceNameIsCommonType { - if commonTypesPackageName == nil { - return nil, fmt.Errorf("ReferenceName %q is indicated to be a common type, but common types package name was not specified", *input.ReferenceName) - } + } else if input.ReferenceNameIsCommonType != nil && *input.ReferenceNameIsCommonType && commonTypesPackageName != nil { out = fmt.Sprintf("%s.%s", *commonTypesPackageName, out) } diff --git a/tools/data-api-sdk/v1/models/sdk_operation_option.go b/tools/data-api-sdk/v1/models/sdk_operation_option.go index 5da98662f69..9219a169ac9 100644 --- a/tools/data-api-sdk/v1/models/sdk_operation_option.go +++ b/tools/data-api-sdk/v1/models/sdk_operation_option.go @@ -9,6 +9,9 @@ type SDKOperationOption struct { // HeaderName specifies the name of the HTTP Header associated with this Option. HeaderName *string `json:"headerName,omitempty"` + // ODataFieldName specifies the name for the OData query string parameter associated with this Option. + ODataFieldName *string `json:"odataFieldName,omitempty"` + // QueryStringName specifies the name for the QueryString key associated with this Option. QueryStringName *string `json:"queryStringName,omitempty"` From ee6fb32fcdad6bfc4e45b016e4a2498955bece39 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 4 Aug 2024 00:59:11 +0100 Subject: [PATCH 027/117] importer-msgraph-metadata: handle bad models by ignoring them, support odata, query and header options --- .../components/normalize/normalize.go | 3 + .../components/parser/resourceids.go | 5 +- .../components/parser/resources.go | 60 ++++++++ .../components/parser/types.go | 55 ++++++- .../components/tags/tags.go | 5 +- .../internal/pipeline/run_importer.go | 21 ++- .../internal/pipeline/task_cleanup_models.go | 31 ++++ .../internal/pipeline/task_parse_resources.go | 46 +++++- ...ate_models.go => task_translate_models.go} | 16 +- .../pipeline/task_translate_service.go | 137 +++++++++++++++++- tools/importer-msgraph-metadata/main.go | 4 +- 11 files changed, 357 insertions(+), 26 deletions(-) create mode 100644 tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go rename tools/importer-msgraph-metadata/internal/pipeline/{translate_models.go => task_translate_models.go} (73%) diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/normalize.go index 110f26c5cdf..171300cc503 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize.go @@ -78,6 +78,9 @@ func CleanName(name string) string { // known issue where CloudPC appears with inconsistent casing name = regexp.MustCompile("(?i)CloudPc").ReplaceAllString(name, "CloudPC") + // Orderby should be OrderBy + name = regexp.MustCompile("^Orderby").ReplaceAllString(name, "OrderBy") + return name } diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 936e1a8ef4b..5ccb7c96fba 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -28,7 +28,7 @@ type ResourceIdMatch struct { Remainder *ResourceId } -// MatchIdOrAncestor returns a ResourceIdMatch containing a matching/ancestor ResourceId and/or a remaininder value, or nil if no +// MatchIdOrAncestor returns a ResourceIdMatch containing a matching/ancestor ResourceId and/or a remainder value, or nil if no // match/ancestor was found. A match is a ResourceId that represents the same path, and an ancestor is any ResourceId that // represents a shorter matching path. Where multiple ancestors are found, the most granular (i.e. the longest) ancestor is returned. // @@ -445,6 +445,9 @@ func ParseResourceIDs(paths openapi3.Paths, serviceName *string) (resourceIds Re } id := NewResourceId(path, operationTags) + if len(id.Segments) == 0 { + continue + } segmentsLastIndex := len(id.Segments) - 1 lastSegment := id.Segments[segmentsLastIndex] diff --git a/tools/importer-msgraph-metadata/components/parser/resources.go b/tools/importer-msgraph-metadata/components/parser/resources.go index 5d0cdf94885..755ea85ff10 100644 --- a/tools/importer-msgraph-metadata/components/parser/resources.go +++ b/tools/importer-msgraph-metadata/components/parser/resources.go @@ -4,7 +4,10 @@ package parser import ( + "fmt" "net/http" + + sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) type Resource struct { @@ -23,12 +26,69 @@ type Operation struct { ResourceId *ResourceId UriSuffix *string RequestModel *string + RequestHeaders *Headers + RequestParams *Params RequestType *DataType Responses Responses PaginationField *string Tags []string } +type Header struct { + Name string + Type *DataType +} + +func (h Header) DataApiSdkObjectDefinition() (*sdkModels.SDKOperationOptionObjectDefinition, error) { + if h.Type == nil { + return nil, fmt.Errorf("param %q has no Type", h.Name) + } + + return &sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: nil, + Type: h.Type.DataApiSdkOperationOptionObjectDefinitionType(), + }, nil +} + +type Headers []Header + +type Param struct { + Name string + Type *DataType + ItemType *DataType +} + +func (p Param) DataApiSdkObjectDefinition() (*sdkModels.SDKOperationOptionObjectDefinition, error) { + if p.Type == nil { + return nil, fmt.Errorf("param %q has no Type", p.Name) + } + + if *p.Type == DataTypeArray { + if p.ItemType != nil { + return &sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: &sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: nil, + Type: p.ItemType.DataApiSdkOperationOptionObjectDefinitionType(), + }, + ReferenceName: nil, + Type: sdkModels.ListSDKOperationOptionObjectDefinitionType, + }, nil + } + + return nil, nil + } + + return &sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: nil, + Type: p.Type.DataApiSdkOperationOptionObjectDefinitionType(), + }, nil +} + +type Params []Param + type Response struct { Status int ContentType *string diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index f31f9c1bca7..62c33c5205e 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -11,6 +11,7 @@ import ( "github.com/hashicorp/go-azure-helpers/lang/pointer" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" "golang.org/x/text/cases" "golang.org/x/text/language" ) @@ -119,7 +120,9 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { } if objectDefinition == nil { - return nil, fmt.Errorf("could not determine SDKObjectDefinition for field: %s", fieldName) + //return nil, fmt.Errorf("could not determine SDKObjectDefinition for field: %s", fieldName) + logging.Warnf("could not determine SDKObjectDefinition for field %q, skipping", fieldName) + continue } sdkFields[fieldName] = sdkModels.SDKField{ @@ -137,6 +140,10 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { } } + if len(sdkFields) == 0 { + return nil, nil + } + // TODO support discriminated types (good example: conditional access named locations) return &sdkModels.SDKModel{ DiscriminatedValue: nil, @@ -176,8 +183,12 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj return nil, fmt.Errorf("field type Model encountered without model name") } - if !models.Found(*f.ModelName) || !models[*f.ModelName].IsValid() { - return nil, fmt.Errorf("field type Model encountered with invalid referenced model") + if !models.Found(*f.ModelName) { + return nil, fmt.Errorf("field type Model encountered with unknown referenced model") + } + + if !models[*f.ModelName].IsValid() { + logging.Warnf("skipping field %q with type Model as the referenced model %q is invalid", f.Title, *f.ModelName) } return &sdkModels.SDKObjectDefinition{ @@ -189,8 +200,12 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj case DataTypeArray: if f.ModelName != nil { - if !models.Found(*f.ModelName) || !models[*f.ModelName].IsValid() { - return nil, fmt.Errorf("field type Array[Model] encountered with invalid referenced model") + if !models.Found(*f.ModelName) { + return nil, fmt.Errorf("field type Array[Model] encountered with unknown referenced model") + } + + if !models[*f.ModelName].IsValid() { + logging.Warnf("skipping field %q with type Array[Model] as the referenced model %q is invalid", f.Title, *f.ModelName) } return &sdkModels.SDKObjectDefinition{ @@ -288,6 +303,27 @@ func (ft DataType) DataApiSdkObjectDefinitionType() sdkModels.SDKObjectDefinitio return sdkModels.StringSDKObjectDefinitionType } +func (ft DataType) DataApiSdkOperationOptionObjectDefinitionType() sdkModels.SDKOperationOptionObjectDefinitionType { + switch ft { + case DataTypeString, DataTypeBase64, DataTypeDuration, DataTypeUuid: + return sdkModels.StringSDKOperationOptionObjectDefinitionType + case DataTypeInteger64, DataTypeInteger32, DataTypeInteger16, DataTypeInteger8, DataTypeIntegerUnsigned64, + DataTypeIntegerUnsigned32, DataTypeIntegerUnsigned16, DataTypeIntegerUnsigned8: + return sdkModels.IntegerSDKOperationOptionObjectDefinitionType + case DataTypeFloat64, DataTypeFloat32: + return sdkModels.FloatSDKOperationOptionObjectDefinitionType + case DataTypeBool: + return sdkModels.BooleanSDKOperationOptionObjectDefinitionType + case DataTypeDate, DataTypeDateTime, DataTypeTime: + return sdkModels.StringSDKOperationOptionObjectDefinitionType + case DataTypeBinary: + return sdkModels.StringSDKOperationOptionObjectDefinitionType + } + + // Fall back to string where the type is not known + return sdkModels.StringSDKOperationOptionObjectDefinitionType +} + // FieldType parses the schemaType and schemaFormat from the OpenAPI spec for a given field, and returns the appropriate DataType func FieldType(schemaType, schemaFormat string, hasModel bool) *DataType { var ret DataType @@ -696,10 +732,19 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta } } + if field.Type == nil { + logging.Warnf("skipping field %q in model %q because Type is nil", name, field.Title) + continue + } + model.Fields[normalize.CleanName(jsonField)] = &field } } + if !model.IsValid() { + delete(models, name) + } + return models, constants } diff --git a/tools/importer-msgraph-metadata/components/tags/tags.go b/tools/importer-msgraph-metadata/components/tags/tags.go index ad416472107..e8c6a6a4b7b 100644 --- a/tools/importer-msgraph-metadata/components/tags/tags.go +++ b/tools/importer-msgraph-metadata/components/tags/tags.go @@ -5,13 +5,14 @@ package tags import ( "fmt" - "github.com/getkin/kin-openapi/openapi3" "strings" + + "github.com/getkin/kin-openapi/openapi3" ) type ServiceTags map[string][]string -// parsetags returns a ServiceTags (map[string][]string) for the provided openapi3.Tags. Note that the resulting ServiceTags +// Parse returns a ServiceTags (map[string][]string) for the provided openapi3.Tags. Note that the resulting ServiceTags // item values are not capitalized, singularized or otherwise cleaned up. func Parse(tags openapi3.Tags) (services ServiceTags, err error) { services = make(map[string][]string, 0) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go b/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go index f231371fae6..0fcc6c19cde 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go @@ -50,19 +50,24 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha repo: input.Repo, } - logging.Infof(fmt.Sprintf("Loading OpenAPI3 definitions for API version %q", apiVersion)) + logging.Infof("Loading OpenAPI3 definitions for API version %q", apiVersion) p.spec, err = openapi3.NewLoader().LoadFromFile(filepath.Join(input.MetadataDirectory, openApiFile)) if err != nil { return err } - logging.Infof(fmt.Sprintf("Parsing models and constants...")) + logging.Infof("Parsing models and constants...") p.models, p.constants, err = parser.Common(p.spec.Components.Schemas) if err != nil { return err } - logging.Infof(fmt.Sprintf("Parsing resource IDs...")) + logging.Infof("Cleaning up models...") + if err = p.cleanupModels(); err != nil { + return err + } + + logging.Infof("Parsing resource IDs...") p.resourceIds, err = parser.ParseResourceIDs(p.spec.Paths, nil) if err != nil { return err @@ -103,7 +108,7 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } } - logging.Infof(fmt.Sprintf("Importing service %q for API version %q", service.Name, version)) + logging.Infof("Importing service %q for API version %q", service.Name, version) if err = p.ForService(service.Directory).RunImport(); err != nil { return err @@ -125,12 +130,12 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } } } - resourceIds := make(parser.ResourceIds, 0, len(usedResourceIds)) + p.resourceIds = make(parser.ResourceIds, 0, len(usedResourceIds)) for _, resourceId := range usedResourceIds { - resourceIds = append(resourceIds, &resourceId) + p.resourceIds = append(p.resourceIds, &resourceId) } - commonTypesForApiVersion, err := translateCommonTypesToDataApiSdkTypes(p.models, p.constants, resourceIds) + commonTypesForApiVersion, err := p.translateCommonTypesToDataApiSdkTypes() if err != nil { return err } @@ -149,7 +154,7 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } func (p pipelineForService) RunImport() error { - logging.Infof(fmt.Sprintf("Parsing resources for %q", p.service)) + logging.Infof("Parsing resources for %q", p.service) resources, err := p.parseResources(p.resourceIds, p.models, p.constants) if err != nil { return err diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go b/tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go new file mode 100644 index 00000000000..c81c6617e6b --- /dev/null +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go @@ -0,0 +1,31 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package pipeline + +func (p pipeline) cleanupModels() error { + // First delete all invalid models (i.e. those without fields) + modelsToDelete := make([]string, 0) + for modelName, model := range p.models { + if !model.IsValid() { + modelsToDelete = append(modelsToDelete, modelName) + } + } + + for _, modelName := range modelsToDelete { + delete(p.models, modelName) + } + + // Look for invalid references due to deleted models, and remove them + for _, model := range p.models { + for _, field := range model.Fields { + if field.ModelName != nil { + if !p.models.Found(*field.ModelName) { + field.ModelName = nil + } + } + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 6567b1b981f..aa90c7db510 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -280,7 +280,11 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model // Unique object for this operation modelName = fmt.Sprintf("%sRequest", operationName) models, constants = parser.Schemas(*schema, modelName, models, constants, false) - requestModel = &modelName + + // Only assign requestModel if the parsed model was valid and added + if models.Found(modelName) { + requestModel = &modelName + } break } } @@ -289,6 +293,44 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + // Determine request options + var requestHeaders *parser.Headers + var requestParams *parser.Params + headers := make(parser.Headers, 0) + params := make(parser.Params, 0) + for _, param := range operation.Parameters { + if param.Value == nil { + continue + } + + result, _ := parser.FlattenSchemaRef(param.Value.Schema, nil) + paramType := parser.FieldType(result.Type, result.Format, false) + + switch param.Value.In { + case "header": + headers = append(headers, parser.Header{ + Name: param.Value.Name, + Type: paramType, + }) + case "query": + var itemType *parser.DataType + if paramType != nil && *paramType == parser.DataTypeArray { + itemType = parser.FieldType(result.Type, result.Format, false) + } + params = append(params, parser.Param{ + Name: param.Value.Name, + Type: paramType, + ItemType: itemType, + }) + } + } + if len(headers) > 0 { + requestHeaders = &headers + } + if len(params) > 0 { + requestParams = ¶ms + } + if operationType == parser.OperationTypeCreate || operationType == parser.OperationTypeUpdate || operationType == parser.OperationTypeCreateUpdate { if resourceId != nil && len(resourceId.Segments) > 0 && resourceId.Segments[len(resourceId.Segments)-1].Value == "$ref" { requestModel = pointer.To("DirectoryObject") @@ -302,6 +344,8 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model ResourceId: resourceId, UriSuffix: uriSuffix, RequestModel: requestModel, + RequestHeaders: requestHeaders, + RequestParams: requestParams, RequestType: requestType, Responses: responses, PaginationField: paginationField, diff --git a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_models.go similarity index 73% rename from tools/importer-msgraph-metadata/internal/pipeline/translate_models.go rename to tools/importer-msgraph-metadata/internal/pipeline/task_translate_models.go index 8ae5de0cc27..9eb554e4bbf 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/translate_models.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_models.go @@ -8,25 +8,29 @@ import ( sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -func translateCommonTypesToDataApiSdkTypes(models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) (*sdkModels.CommonTypes, error) { +func (p pipeline) translateCommonTypesToDataApiSdkTypes() (*sdkModels.CommonTypes, error) { sdkConstantsMap := make(map[string]sdkModels.SDKConstant) sdkModelsMap := make(map[string]sdkModels.SDKModel) sdkResourceIdsMap := make(map[string]sdkModels.ResourceID) - for modelName, model := range models { + for modelName, model := range p.models { if model.Common { - sdkModel, err := model.DataApiSdkModel(models) + sdkModel, err := model.DataApiSdkModel(p.models) if err != nil { return nil, err } + if sdkModel == nil { + logging.Warnf("skipping invalid model %q as it has no fields", modelName) + continue + } sdkModelsMap[modelName] = *sdkModel } } - for constantName, constant := range constants { + for constantName, constant := range p.constants { constantValues := make(map[string]string) for _, value := range constant.Enum { // prefix constant value names with underscore to prevent naming conflicts with similarly named models in the generated SDK @@ -40,7 +44,7 @@ func translateCommonTypesToDataApiSdkTypes(models parser.Models, constants parse } } - for _, resourceId := range resourceIds { + for _, resourceId := range p.resourceIds { sdkResourceId, err := resourceId.DataApiSdkResourceId() if err != nil { return nil, err diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index a6fd177c657..e1d1c61dfe8 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -5,6 +5,7 @@ package pipeline import ( "fmt" + "strings" "github.com/hashicorp/go-azure-helpers/lang/pointer" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" @@ -12,6 +13,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Service, error) { @@ -95,6 +97,139 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } } + options := make(map[string]sdkModels.SDKOperationOption) + + if operation.RequestHeaders != nil { + for _, header := range *operation.RequestHeaders { + if strings.EqualFold(header.Name, "ConsistencyLevel") { + options[normalize.CleanName(header.Name)] = sdkModels.SDKOperationOption{ + ODataFieldName: &header.Name, + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: pointer.To("odata.ConsistencyLevel"), + Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, + }, + } + continue + } + + objectDefinition, err := header.DataApiSdkObjectDefinition() + if err != nil { + return nil, err + } + + if objectDefinition == nil { + logging.Warnf("could not determine SDKOperationOptionObjectDefinition for header %q, skipping", header.Name) + continue + } + + options[normalize.CleanName(header.Name)] = sdkModels.SDKOperationOption{ + HeaderName: &header.Name, + Required: false, + ObjectDefinition: *objectDefinition, + } + } + } + + if operation.RequestParams != nil { + for _, param := range *operation.RequestParams { + switch normalize.CleanName(param.Name) { + case "Count": + options["Count"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("Count"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + Type: sdkModels.BooleanSDKOperationOptionObjectDefinitionType, + }, + } + + case "Expand": + options["Expand"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("Expand"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: pointer.To("odata.Expand"), + Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, + }, + } + + case "Format": + options["Format"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("Format"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: pointer.To("odata.Format"), + Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, + }, + } + + case "OrderBy": + options["OrderBy"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("OrderBy"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: nil, + ReferenceName: pointer.To("odata.OrderBy"), + Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, + }, + } + + case "Select": + options["Select"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("Select"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + NestedItem: &sdkModels.SDKOperationOptionObjectDefinition{ + Type: sdkModels.StringSDKOperationOptionObjectDefinitionType, + }, + Type: sdkModels.ListSDKOperationOptionObjectDefinitionType, + }, + } + + case "Filter", "Search": + options[normalize.CleanName(param.Name)] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To(normalize.CleanName(param.Name)), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + Type: sdkModels.StringSDKOperationOptionObjectDefinitionType, + }, + } + + case "Skip", "Top": + // Don't set here, we handle this implicitly for list operations + + default: + objectDefinition, err := param.DataApiSdkObjectDefinition() + if err != nil { + return nil, err + } + + if objectDefinition == nil { + logging.Warnf("could not determine SDKOperationOptionObjectDefinition for param %q, skipping", param.Name) + continue + } + + options[normalize.CleanName(param.Name)] = sdkModels.SDKOperationOption{ + QueryStringName: ¶m.Name, + Required: false, + ObjectDefinition: *objectDefinition, + } + } + } + } + + if operation.Type == parser.OperationTypeList { + options["Top"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("Top"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + Type: sdkModels.IntegerSDKOperationOptionObjectDefinitionType, + }, + } + + options["Skip"] = sdkModels.SDKOperationOption{ + ODataFieldName: pointer.To("Skip"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + Type: sdkModels.IntegerSDKOperationOptionObjectDefinitionType, + }, + } + } + var responseObject *sdkModels.SDKObjectDefinition responseObjectIsCommonType := true @@ -165,7 +300,7 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv FieldContainingPaginationDetails: operation.PaginationField, LongRunning: false, Method: operation.Method, - Options: nil, // TODO request options for odata queries etc + Options: options, RequestObject: requestObject, ResourceIDName: resourceIdName, ResourceIDNameIsCommonType: pointer.To(true), diff --git a/tools/importer-msgraph-metadata/main.go b/tools/importer-msgraph-metadata/main.go index ecdc3b884c8..5c07d70c840 100644 --- a/tools/importer-msgraph-metadata/main.go +++ b/tools/importer-msgraph-metadata/main.go @@ -16,7 +16,7 @@ import ( const ( metadataDirectory = "../../submodules/msgraph-metadata" microsoftGraphConfig = "../../config/microsoft-graph.hcl" - openApiFilePattern = "transformed_%s_metadata.xml.yaml" + openApiFilePattern = "openapi/%s/default.yaml" outputDirectory = "../../api-definitions" ) @@ -27,7 +27,7 @@ func main() { } logging.Log = hclog.New(loggingOpts) - c := cli.NewCLI("importer-msgraph-metadata", "0.2.2") + c := cli.NewCLI("importer-msgraph-metadata", "0.3.0") c.Args = os.Args[1:] c.Commands = map[string]cli.CommandFactory{ "import": cmd.NewImportCommand(metadataDirectory, microsoftGraphConfig, openApiFilePattern, outputDirectory), From 0e8e9a928307ec30d8808dc414ce915502c87f0f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 4 Aug 2024 00:59:40 +0100 Subject: [PATCH 028/117] generator-go-sdk: support for generating odata options --- .../internal/generator/templater_methods.go | 16 ++++++++++++++-- .../internal/generator/templater_models.go | 4 ++-- tools/generator-go-sdk/main.go | 2 +- 3 files changed, 17 insertions(+), 5 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 1cd764babb9..1b9a8ca119f 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -771,6 +771,7 @@ func (c methodsPandoraTemplater) optionsStruct(data GeneratorData) (*string, err } properties := make([]string, 0) + odataAssignments := make([]string, 0) queryStringAssignments := make([]string, 0) headerAssignments := make([]string, 0) @@ -780,6 +781,15 @@ func (c methodsPandoraTemplater) optionsStruct(data GeneratorData) (*string, err return nil, fmt.Errorf("determining golang type name for option %q's ObjectDefinition: %+v", optionName, err) } properties = append(properties, fmt.Sprintf("%s *%s", optionName, *optionType)) + if option.ODataFieldName != nil { + value := fmt.Sprintf("*o.%s", *option.ODataFieldName) + if option.ObjectDefinition.Type == models.IntegerSDKOperationOptionObjectDefinitionType { + value = fmt.Sprintf("int(%s)", value) + } + odataAssignments = append(odataAssignments, fmt.Sprintf(`if o.%[1]s != nil { + out.%[2]s = %[3]s +}`, optionName, *option.ODataFieldName, value)) + } if option.HeaderName != nil { headerAssignments = append(headerAssignments, fmt.Sprintf(`if o.%[1]s != nil { out.Append("%[2]s", fmt.Sprintf("%%v", *o.%[1]s)) @@ -793,6 +803,7 @@ func (c methodsPandoraTemplater) optionsStruct(data GeneratorData) (*string, err } sort.Strings(properties) + sort.Strings(odataAssignments) sort.Strings(headerAssignments) sort.Strings(queryStringAssignments) @@ -813,14 +824,15 @@ func (o %[1]s) ToHeaders() *client.Headers { func (o %[1]s) ToOData() *odata.Query { out := odata.Query{} +%[4]s return &out } func (o %[1]s) ToQuery() *client.QueryParams { out := client.QueryParams{} -%[4]s +%[5]s return &out } -`, optionsStructName, strings.Join(properties, "\n"), strings.Join(headerAssignments, "\n"), strings.Join(queryStringAssignments, "\n")) +`, optionsStructName, strings.Join(properties, "\n"), strings.Join(headerAssignments, "\n"), strings.Join(odataAssignments, "\n"), strings.Join(queryStringAssignments, "\n")) return &out, nil } diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index c37fc2e5d67..fa3a56ebbb7 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -87,7 +87,7 @@ type Raw%[1]sImpl struct { for _, fieldName := range fields { fieldDetails := c.model.Fields[fieldName] fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil) + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) } @@ -138,7 +138,7 @@ type Raw%[1]sImpl struct { for _, fieldName := range parentFields { fieldDetails := parent.Fields[fieldName] fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, nil) + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, data.commonTypesPackageName) if err != nil { return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) } diff --git a/tools/generator-go-sdk/main.go b/tools/generator-go-sdk/main.go index 7efa9484a5c..6a59ca747ea 100644 --- a/tools/generator-go-sdk/main.go +++ b/tools/generator-go-sdk/main.go @@ -10,7 +10,7 @@ import ( "github.com/hashicorp/go-azure-helpers/lang/pointer" "github.com/hashicorp/go-hclog" - v1 "github.com/hashicorp/pandora/tools/data-api-sdk/v1" + "github.com/hashicorp/pandora/tools/data-api-sdk/v1" "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/generator-go-sdk/internal/cmd" "github.com/hashicorp/pandora/tools/generator-go-sdk/internal/logging" From 37645a2df80c5333571727b8606cfb9a228ff73a Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 4 Aug 2024 01:00:48 +0100 Subject: [PATCH 029/117] update to latest msgraph-metadata --- submodules/msgraph-metadata | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/submodules/msgraph-metadata b/submodules/msgraph-metadata index 7b1feb7cf42..1c4023e9efb 160000 --- a/submodules/msgraph-metadata +++ b/submodules/msgraph-metadata @@ -1 +1 @@ -Subproject commit 7b1feb7cf42dd1f08941b349209c1cf8602f47c8 +Subproject commit 1c4023e9efb074f6685e04571282b6c803a5d97d From 39e9fec7be1288a84c26d18879f1b4b7229731c2 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 11:35:12 +0100 Subject: [PATCH 030/117] importer-msgraph-metadata: fix singularization rules to avoid collision between `/members/$ref` and `/members/{id}/$ref` --- .../components/parser/resourceids.go | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 5ccb7c96fba..ba7e444387c 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -130,7 +130,7 @@ func (r ResourceId) IsMatchOrAncestor(r2 ResourceId) (ResourceId, bool) { } // FullyQualifiedResourceName returns a human-readable name for the ResourceId, using all segments, each segment singularized -// except when the following segment is a plural function, or the first known verb is encountered. +// except when the following segment is an OData reference or plural function, or the first known verb is encountered. // e.g. // if r represents `/applications/{applicationId}/synchronization/jobs/{synchronizationJobId}/schema`, the returned name // will be `ApplicationSynchronizationJob` @@ -149,19 +149,21 @@ func (r ResourceId) FullyQualifiedResourceName(suffixQualification *string) (*st if i == len(r.Segments)+1 { // Don't singularize the final segment, so it can still be reliably verb-matched when parsing operations later shouldSingularize = false + } - } else if len(r.Segments) > i+1 && r.Segments[i+1].Type != SegmentUserValue { + if len(r.Segments) > i+1 && r.Segments[i+1].Type != SegmentUserValue { // Look for a verb match in the next segment, if it exists and is not a SegmentUserValue // Example: in the following ID, we want to _not_ singularize the `jobs` label // /applications/{applicationId}/synchronization/jobs/validateCredentials if _, ok := normalize.Verbs.Match(normalize.CleanName(r.Segments[i+1].Value)); ok { shouldSingularize = false } + } - } else if len(r.Segments) > i+1 && r.Segments[i+1].Type == SegmentODataReference && r.Segments[i+1].Value == "$count" { - // $count indicates a plural entity whereas $ref does not + if len(r.Segments) > i+1 && r.Segments[i+1].Type == SegmentODataReference && (r.Segments[i+1].Value == "$count" || r.Segments[i+1].Value == "$ref") { + // $count and $ref indicate a plural entity (noting that this only applies when the current + // segment is a label and not user-specified). shouldSingularize = false - } } From 3b1c2d6993d1e190f2b0ea0056de1a18f17762d8 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 12:30:15 +0100 Subject: [PATCH 031/117] importer-msgraph-metadata: singularization fix for "access", casing fixes for "OAuth" and "Id" --- .../components/normalize/normalize.go | 26 ++++++++++++++++--- 1 file changed, 22 insertions(+), 4 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/normalize.go index 171300cc503..86d2effb0d6 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize.go @@ -13,15 +13,27 @@ import ( ) func Singularize(name string) string { + // "access" is already singular + if len(name) >= 5 && name[len(name)-5:] == "ccess" { + return name + } + + // "properties", "entities" etc if len(name) >= 3 && name[len(name)-3:] == "ies" { return fmt.Sprintf("%sy", name[:len(name)-3]) } + + // "premises" etc if len(name) >= 3 && name[len(name)-3:] == "ses" { return name[:len(name)-2] } + + // all other words, remove the trailing "s" if len(name) >= 1 && name[len(name)-1:] == "s" { return name[:len(name)-1] } + + // else just return the original name return name } @@ -69,18 +81,24 @@ func CleanName(name string) string { trailing := regexp.MustCompile(`[^a-zA-Z0-9]`).ReplaceAllString(name[len(leading):], "") name = leading + trailing - // Odata should be OData - name = regexp.MustCompile("^Odata").ReplaceAllString(name, "OData") + // CloudPc should be CloudPC, also fixes inconsistent casing + name = regexp.MustCompile("(?i)CloudPc").ReplaceAllString(name, "CloudPC") // Innererror should be InnerError name = regexp.MustCompile("^Innererror").ReplaceAllString(name, "InnerError") - // known issue where CloudPC appears with inconsistent casing - name = regexp.MustCompile("(?i)CloudPc").ReplaceAllString(name, "CloudPC") + // Oauth should be OAuth + name = regexp.MustCompile("^Oauth").ReplaceAllString(name, "OAuth") + + // Odata should be OData + name = regexp.MustCompile("^Odata").ReplaceAllString(name, "OData") // Orderby should be OrderBy name = regexp.MustCompile("^Orderby").ReplaceAllString(name, "OrderBy") + // (trailing) ID should be Id for compatibility with Go SDK generator + name = regexp.MustCompile("([a-z])ID$").ReplaceAllString(name, "${1}Id") + return name } From 5d1ba6fbc53b6b919afc186bd9bbdd8ccffe678d Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 13:35:26 +0100 Subject: [PATCH 032/117] generator-go-sdk: add missing imports --- tools/generator-go-sdk/internal/generator/service.go | 10 ++-------- .../internal/generator/templater_clients.go | 2 ++ .../internal/generator/templater_meta_client.go | 2 ++ .../internal/generator/templater_models.go | 12 +++++++++--- 4 files changed, 15 insertions(+), 11 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/service.go b/tools/generator-go-sdk/internal/generator/service.go index 282971dd498..6f9b99eb7e4 100644 --- a/tools/generator-go-sdk/internal/generator/service.go +++ b/tools/generator-go-sdk/internal/generator/service.go @@ -131,21 +131,15 @@ func runGoImports(path string) { } func cleanAndRecreateWorkingDirectory(path string) error { - // rm -r + // rm -r 💥 if err := os.RemoveAll(path); err != nil { return fmt.Errorf("deleting %q: %+v", path, err) } - // then ensure the directory exists - if err := os.MkdirAll(path, 0777); err != nil { - return fmt.Errorf("creating %q: %+v", path, err) - } - - return nil + return ensureWorkingDirectoryExists(path) } func ensureWorkingDirectoryExists(path string) error { - // TODO: make these less exciting if err := os.MkdirAll(path, 0777); err != nil { if !os.IsExist(err) { return fmt.Errorf("creating %q: %+v", path, err) diff --git a/tools/generator-go-sdk/internal/generator/templater_clients.go b/tools/generator-go-sdk/internal/generator/templater_clients.go index f0977f68315..8aed0d9cca3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients.go @@ -16,6 +16,8 @@ func (c clientsTemplater) template(data GeneratorData) (*string, error) { template := fmt.Sprintf(`package %[1]s import ( + "fmt" + "github.com/hashicorp/go-azure-sdk/sdk/client" "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client.go b/tools/generator-go-sdk/internal/generator/templater_meta_client.go index 00726c222ab..d168f6b1b3f 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client.go @@ -63,6 +63,8 @@ configureFunc(%[1]s.Client) %[3]s import ( + "fmt" + "github.com/hashicorp/go-azure-sdk/sdk/client" "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index fa3a56ebbb7..46335da66e3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -36,6 +36,11 @@ func (c modelsTemplater) template(data GeneratorData) (*string, error) { return nil, fmt.Errorf("generating functions: %+v", err) } + commonTypesInclude := "" + if data.commonTypesIncludePath != nil { + commonTypesInclude = fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s"`, data.sourceType, *data.commonTypesIncludePath) + } + template := fmt.Sprintf(`package %[1]s import ( @@ -49,12 +54,13 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + %[2]s ) -%[4]s -%[2]s +%[5]s %[3]s -`, data.packageName, *structCode, *methods, *copyrightLines) +%[4]s +`, data.packageName, commonTypesInclude, *structCode, *methods, *copyrightLines) return &template, nil } From 53758b60780f4420700e77eae1068f199edb6c86 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 16:47:45 +0100 Subject: [PATCH 033/117] generator-go-sdk: fix package/directory name for API version when outputting meta client / common types --- .../internal/generator/data.go | 17 +++++++++----- .../internal/generator/stage_meta_client.go | 13 ++++++----- .../generator/templater_meta_client.go | 22 ++++++++----------- 3 files changed, 27 insertions(+), 25 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 1c31db6f569..6ad258cf910 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -74,10 +74,10 @@ type GeneratorData struct { func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { resourcePackageName := strings.ToLower(i.ResourceName) servicePackageName := strings.ToLower(i.ServiceName) - versionPackageName := strings.ToLower(i.VersionName) + versionDirectoryName := strings.ToLower(i.VersionName) - versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionPackageName) - resourceOutputPath := filepath.Join(versionOutputPath, resourcePackageName) + versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionDirectoryName) + resourceOutputPath := filepath.Join(versionOutputPath, versionDirectoryName) useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) @@ -122,16 +122,20 @@ type VersionGeneratorData struct { // for example {workingDir}/{service}/{version} versionOutputPath string - // the name of the version as a package + // the directory name of the package for the API version + versionDirectoryName string + + // the package name for the API version versionPackageName string } func (i VersionGeneratorInput) generatorData(settings Settings) VersionGeneratorData { servicePackageName := strings.ToLower(i.ServiceName) + versionDirectoryName := strings.ToLower(i.VersionName) versionPackageName := strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_")) - commonTypesOutputPath := filepath.Join(i.OutputDirectory, settings.CommonTypesPackageName, versionPackageName) - versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionPackageName) + commonTypesOutputPath := filepath.Join(i.OutputDirectory, settings.CommonTypesPackageName, versionDirectoryName) + versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionDirectoryName) useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) @@ -152,6 +156,7 @@ func (i VersionGeneratorInput) generatorData(settings Settings) VersionGenerator commonTypesOutputPath: commonTypesOutputPath, resources: i.Resources, versionOutputPath: versionOutputPath, + versionDirectoryName: versionDirectoryName, versionPackageName: versionPackageName, } } diff --git a/tools/generator-go-sdk/internal/generator/stage_meta_client.go b/tools/generator-go-sdk/internal/generator/stage_meta_client.go index 8e9a98b923b..d96473bf567 100644 --- a/tools/generator-go-sdk/internal/generator/stage_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/stage_meta_client.go @@ -15,12 +15,13 @@ func (s *Generator) metaClient(data VersionGeneratorData) error { var templater templaterForVersion if data.useNewBaseLayer { templater = metaClientTemplater{ - serviceName: data.servicePackageName, - apiVersion: data.versionPackageName, - baseClientPackage: data.baseClientPackage, - resources: data.resources, - source: data.source, - sourceType: data.sourceType, + serviceName: data.servicePackageName, + apiVersionDirectoryName: data.versionDirectoryName, + apiVersionPackageName: data.versionPackageName, + baseClientPackage: data.baseClientPackage, + resources: data.resources, + source: data.source, + sourceType: data.sourceType, } } else { templater = metaClientAutorestTemplater{ diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client.go b/tools/generator-go-sdk/internal/generator/templater_meta_client.go index d168f6b1b3f..fd97d1926d5 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client.go @@ -9,12 +9,13 @@ import ( ) type metaClientTemplater struct { - serviceName string - apiVersion string - baseClientPackage string - resources map[string]models.APIResource - source models.SourceDataOrigin - sourceType models.SourceDataType + apiVersionDirectoryName string + apiVersionPackageName string + baseClientPackage string + resources map[string]models.APIResource + serviceName string + source models.SourceDataOrigin + sourceType models.SourceDataType } func (m metaClientTemplater) template() (*string, error) { @@ -36,7 +37,7 @@ func (m metaClientTemplater) template() (*string, error) { for _, resourceName := range resourceNames { variableName := fmt.Sprintf("%s%sClient", strings.ToLower(string(resourceName[0])), resourceName[1:]) - imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s/%s/%s"`, m.sourceType, strings.ToLower(m.serviceName), m.apiVersion, strings.ToLower(resourceName))) + imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s/%s/%s"`, m.sourceType, strings.ToLower(m.serviceName), m.apiVersionDirectoryName, strings.ToLower(resourceName))) fields = append(fields, fmt.Sprintf("%[1]s *%[2]s.%[1]sClient", resourceName, strings.ToLower(resourceName))) clientInitializationTemplate := fmt.Sprintf(`%[1]s, err := %[2]s.New%[3]sClientWithBaseURI(sdkApi) if err != nil { @@ -53,11 +54,6 @@ configureFunc(%[1]s.Client) sort.Strings(fields) sort.Strings(imports) - packageName := m.apiVersion - if strings.Contains(packageName, "-") { - packageName = fmt.Sprintf("v%s", strings.ReplaceAll(m.apiVersion, "-", "_")) - } - out := fmt.Sprintf(`package %[1]s %[3]s @@ -83,6 +79,6 @@ func NewClientWithBaseURI(sdkApi sdkEnv.Api, configureFunc func(c *%[2]s.Client) %[7]s }, nil } -`, packageName, m.baseClientPackage, *copyrightLines, strings.Join(imports, "\n"), strings.Join(fields, "\n"), strings.Join(clientInitialization, "\n"), strings.Join(assignments, "\n")) +`, m.apiVersionPackageName, m.baseClientPackage, *copyrightLines, strings.Join(imports, "\n"), strings.Join(fields, "\n"), strings.Join(clientInitialization, "\n"), strings.Join(assignments, "\n")) return &out, nil } From 066c7ab33d3cc097776e17473b4b6a2c116594d3 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 17:19:06 +0100 Subject: [PATCH 034/117] generator-go-sdk: unit test updates --- .../generator/templater_clients_test.go | 8 +++- .../generator/templater_constants_test.go | 12 ++++- .../generator/templater_id_parser_test.go | 8 ++-- .../internal/generator/templater_methods.go | 2 +- .../templater_methods_discriminators_test.go | 44 +++++++++++-------- ...er_methods_long_running_operations_test.go | 9 ++-- .../generator/templater_methods_test.go | 5 +++ .../generator/templater_readme_test.go | 21 +++++++++ 8 files changed, 79 insertions(+), 30 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_clients_test.go b/tools/generator-go-sdk/internal/generator/templater_clients_test.go index 7567c6ac9f6..7ce791b10b3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_clients_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_clients_test.go @@ -9,6 +9,7 @@ import ( func TestTemplateClient(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "somepackage", serviceClientName: "ExampleClient", source: AccTestLicenceType, @@ -22,6 +23,9 @@ func TestTemplateClient(t *testing.T) { expected := `package somepackage import ( + "fmt" + "github.com/hashicorp/go-azure-sdk/sdk/client" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" sdkEnv "github.com/hashicorp/go-azure-sdk/sdk/environments" ) @@ -29,11 +33,11 @@ import ( // acctests licence placeholder type ExampleClient struct { - Client *resourcemanager.Client + Client *testclient.Client } func NewExampleClientWithBaseURI(sdkApi sdkEnv.Api) (*ExampleClient, error) { - client, err := resourcemanager.NewResourceManagerClient(sdkApi, "somepackage", defaultApiVersion) + client, err := testclient.NewClient(sdkApi, "somepackage", defaultApiVersion) if err != nil { return nil, fmt.Errorf("instantiating ExampleClient: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_constants_test.go b/tools/generator-go-sdk/internal/generator/templater_constants_test.go index bd97d7773f4..ab30a546a15 100644 --- a/tools/generator-go-sdk/internal/generator/templater_constants_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_constants_test.go @@ -34,7 +34,11 @@ func TestTemplateConstantsSingle(t *testing.T) { } expected := `package somepackage -import "strings" +import ( + "encoding/json" + "fmt" + "strings" +) // acctests licence placeholder // template for first @@ -64,7 +68,11 @@ func TestTemplateConstantsMultiple(t *testing.T) { } expected := `package somepackage -import "strings" +import ( + "encoding/json" + "fmt" + "strings" +) // acctests licence placeholder // template for first diff --git a/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go b/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go index c08cbbb08a4..40fdf2f6bc3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_id_parser_test.go @@ -71,7 +71,7 @@ var _ resourceids.ResourceId = &BasicTestId{} } id := BasicTestId{} - if err := id.FromParseResult(*parsed); err != nil { + if err = id.FromParseResult(*parsed); err != nil { return nil, err } @@ -88,7 +88,7 @@ var _ resourceids.ResourceId = &BasicTestId{} } id := BasicTestId{} - if err := id.FromParseResult(*parsed); err != nil { + if err = id.FromParseResult(*parsed); err != nil { return nil, err } @@ -220,7 +220,7 @@ func ParseConstantOnlyID(input string) (*ConstantOnlyId, error) { } id := ConstantOnlyId{} - if err := id.FromParseResult(*parsed); err != nil { + if err = id.FromParseResult(*parsed); err != nil { return nil, err } @@ -237,7 +237,7 @@ func ParseConstantOnlyIDInsensitively(input string) (*ConstantOnlyId, error) { } id := ConstantOnlyId{} - if err := id.FromParseResult(*parsed); err != nil { + if err = id.FromParseResult(*parsed); err != nil { return nil, err } diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 1b9a8ca119f..d78c812c0b7 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -233,7 +233,7 @@ func (c methodsPandoraTemplater) longRunningOperationTemplate(data GeneratorData // %[3]s ... func (c %[1]s) %[3]s(ctx context.Context %[4]s) (result %[3]sOperationResponse, err error) { - opts := %[4]s + opts := %[5]s req, err := c.Client.NewRequest(ctx, opts) if err != nil { diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go index bf514e4c2ca..f8c11b921a9 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go @@ -20,6 +20,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsImplementation_Get(t *tes // Discriminated Parent's `unmarshal` function shouldn't be called. input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -62,15 +63,16 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsImplementation_Get(t *tes expected := ` package chubbypandas import ( -"context" -"fmt" -"net/http" -"net/url" -"github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" -"github.com/hashicorp/go-azure-sdk/sdk/client" -"github.com/hashicorp/go-azure-sdk/sdk/client/pollers" -"github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" -"github.com/hashicorp/go-azure-sdk/sdk/odata" + "context" + "fmt" + "net/http" + "net/url" + "github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" + "github.com/hashicorp/go-azure-sdk/sdk/client" + "github.com/hashicorp/go-azure-sdk/sdk/client/pollers" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" + "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" + "github.com/hashicorp/go-azure-sdk/sdk/odata" ) // acctests licence placeholder type GetOperationResponse struct { @@ -122,6 +124,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsParent_Get(t *testing.T) // In this instance the `unmarshal` function for the Parent Type should be called as a part // of the Response. input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -164,15 +167,16 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsParent_Get(t *testing.T) package chubbypandas import ( -"context" -"fmt" -"net/http" -"net/url" -"github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" -"github.com/hashicorp/go-azure-sdk/sdk/client" -"github.com/hashicorp/go-azure-sdk/sdk/client/pollers" -"github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" -"github.com/hashicorp/go-azure-sdk/sdk/odata" + "context" + "fmt" + "net/http" + "net/url" + "github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" + "github.com/hashicorp/go-azure-sdk/sdk/client" + "github.com/hashicorp/go-azure-sdk/sdk/client/pollers" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" + "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" + "github.com/hashicorp/go-azure-sdk/sdk/odata" ) // acctests licence placeholder @@ -229,6 +233,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsImplementation_List(t *te // should be output - and the Discriminated Parent's `unmarshal` function shouldn't be called. input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -280,6 +285,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" "github.com/hashicorp/go-azure-sdk/sdk/client" "github.com/hashicorp/go-azure-sdk/sdk/client/pollers" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" "github.com/hashicorp/go-azure-sdk/sdk/odata" ) @@ -385,6 +391,7 @@ func TestTemplateMethods_Discriminator_ResponseObjectIsParent_List(t *testing.T) // should be output - and the (Discriminated Parent's) `unmarshal` function should be called. input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbypandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -435,6 +442,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" "github.com/hashicorp/go-azure-sdk/sdk/client" "github.com/hashicorp/go-azure-sdk/sdk/client/pollers" + "github.com/hashicorp/go-azure-sdk/sdk/client/msgraph" "github.com/hashicorp/go-azure-sdk/sdk/client/resourcemanager" "github.com/hashicorp/go-azure-sdk/sdk/odata" ) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go index 0a2a0249a70..380565c2c20 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_long_running_operations_test.go @@ -11,6 +11,7 @@ import ( func TestTemplateMethodsLROCreate(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -77,7 +78,7 @@ func (c pandaClient) Create(ctx context.Context , id PandaPop, input string) (re return } - result.Poller, err = resourcemanager.PollerFromResponse(resp, c.Client) + result.Poller, err = testclient.PollerFromResponse(resp, c.Client) if err != nil { return } @@ -104,6 +105,7 @@ func (c pandaClient) CreateThenPoll(ctx context.Context , id PandaPop, input str func TestTemplateMethodsLROReboot(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -161,7 +163,7 @@ func (c pandaClient) Reboot(ctx context.Context , id PandaPop) (result RebootOpe return } - result.Poller, err = resourcemanager.PollerFromResponse(resp, c.Client) + result.Poller, err = testclient.PollerFromResponse(resp, c.Client) if err != nil { return } @@ -195,6 +197,7 @@ func TestTemplateMethodsLRODoesNotCallUnmarshal(t *testing.T) { // Poller - since (for the moment) consumers will need to decide when `Unmarshal` should // be called on the LRO Result, if needed at all. input := GeneratorData{ + baseClientPackage: "testclient", packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -280,7 +283,7 @@ func (c pandaClient) Create(ctx context.Context , id PandaPop, input Example) (r return } - result.Poller, err = resourcemanager.PollerFromResponse(resp, c.Client) + result.Poller, err = testclient.PollerFromResponse(resp, c.Client) if err != nil { return } diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_test.go index 9773931ccfc..f733d4bf580 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_test.go @@ -15,6 +15,7 @@ import ( func TestTemplateMethodsGet(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -88,6 +89,7 @@ func (c pandaClient) Get(ctx context.Context , id PandaPop) (result GetOperation func TestTemplateMethodsGetAsTextPowerShell(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "skinnyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -166,6 +168,7 @@ func (c pandaClient) Get(ctx context.Context , id PandaPop) (result GetOperation func TestTemplateMethodsListWithDiscriminatedType(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -334,6 +337,7 @@ func (c pandaClient) ListCompleteMatchingPredicate(ctx context.Context, id Panda func TestTemplateMethodsListWithSimpleType(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, @@ -455,6 +459,7 @@ func (c pandaClient) ListComplete(ctx context.Context, id PandaPop) (result List func TestTemplateMethodsListWithObject(t *testing.T) { input := GeneratorData{ + baseClientPackage: "testclient", packageName: "chubbyPandas", serviceClientName: "pandaClient", source: AccTestLicenceType, diff --git a/tools/generator-go-sdk/internal/generator/templater_readme_test.go b/tools/generator-go-sdk/internal/generator/templater_readme_test.go index 242ba8888af..e493a7f318a 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme_test.go @@ -41,6 +41,7 @@ client.Client.Authorizer = authorizer apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, }) if err != nil { t.Fatalf("generating readme: %+v", err) @@ -98,6 +99,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -164,6 +166,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { CommonIDAlias: pointer.To("ManagedDisk"), @@ -237,6 +240,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -324,6 +328,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -404,6 +409,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -468,6 +474,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{}, }) if err != nil { @@ -533,6 +540,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{}, }) if err != nil { @@ -593,6 +601,7 @@ for _, item := range items { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -661,6 +670,7 @@ for _, item := range items { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { CommonIDAlias: pointer.To("ManagedDisk"), @@ -736,6 +746,7 @@ for _, item := range items { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -825,6 +836,7 @@ for _, item := range items { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -907,6 +919,7 @@ for _, item := range items { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -980,6 +993,7 @@ for _, item := range items { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{}, }) if err != nil { @@ -1035,6 +1049,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id); err != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -1098,6 +1113,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id); err != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { CommonIDAlias: pointer.To("ManagedDisk"), @@ -1168,6 +1184,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id, payload); err != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -1252,6 +1269,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id, payload, disks.DefaultSomeLong apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -1328,6 +1346,7 @@ if err := client.SomeLongRunningThenPoll(ctx, id, disks.DefaultSomeLongRunningOp apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{ "Disk": { Segments: []models.ResourceIDSegment{ @@ -1390,6 +1409,7 @@ if err := client.SomeLongRunningThenPoll(ctx); err != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{}, }) if err != nil { @@ -1466,6 +1486,7 @@ if model := read.Model; model != nil { apiVersion: "2022-02-01", servicePackageName: "compute", serviceClientName: "DisksClient", + sourceType: models.ResourceManagerSourceDataType, resourceIds: map[string]models.ResourceID{}, }) if err != nil { From c39cc5c98f63cdeae4b76304632bdbc459a3e354 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 19:05:58 +0100 Subject: [PATCH 035/117] generator-go-sdk: fix meta client package naming for numerically versioned APIs --- .../generator-go-sdk/internal/generator/data.go | 7 ++++++- .../internal/generator/stage_meta_client.go | 13 +++++++------ .../generator/templater_meta_client_autorest.go | 17 ++++++++--------- 3 files changed, 21 insertions(+), 16 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 6ad258cf910..dc0daa80404 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -6,6 +6,7 @@ package generator import ( "fmt" "path/filepath" + "regexp" "strings" "github.com/hashicorp/go-azure-helpers/lang/pointer" @@ -132,11 +133,15 @@ type VersionGeneratorData struct { func (i VersionGeneratorInput) generatorData(settings Settings) VersionGeneratorData { servicePackageName := strings.ToLower(i.ServiceName) versionDirectoryName := strings.ToLower(i.VersionName) - versionPackageName := strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_")) commonTypesOutputPath := filepath.Join(i.OutputDirectory, settings.CommonTypesPackageName, versionDirectoryName) versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionDirectoryName) + versionPackageName := strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_")) + if regexp.MustCompile(`^[0-9]+`).MatchString(versionPackageName) { + versionPackageName = fmt.Sprintf("v%s", versionPackageName) + } + useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) return VersionGeneratorData{ diff --git a/tools/generator-go-sdk/internal/generator/stage_meta_client.go b/tools/generator-go-sdk/internal/generator/stage_meta_client.go index d96473bf567..f94adf34fe3 100644 --- a/tools/generator-go-sdk/internal/generator/stage_meta_client.go +++ b/tools/generator-go-sdk/internal/generator/stage_meta_client.go @@ -15,21 +15,22 @@ func (s *Generator) metaClient(data VersionGeneratorData) error { var templater templaterForVersion if data.useNewBaseLayer { templater = metaClientTemplater{ - serviceName: data.servicePackageName, apiVersionDirectoryName: data.versionDirectoryName, apiVersionPackageName: data.versionPackageName, baseClientPackage: data.baseClientPackage, resources: data.resources, + serviceName: data.servicePackageName, source: data.source, sourceType: data.sourceType, } } else { templater = metaClientAutorestTemplater{ - serviceName: data.servicePackageName, - apiVersion: data.versionPackageName, - resources: data.resources, - source: data.source, - sourceType: data.sourceType, + apiVersionDirectoryName: data.versionDirectoryName, + apiVersionPackageName: data.versionPackageName, + resources: data.resources, + serviceName: data.servicePackageName, + source: data.source, + sourceType: data.sourceType, } } diff --git a/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go b/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go index d5d07aa3553..969b3ad609b 100644 --- a/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go +++ b/tools/generator-go-sdk/internal/generator/templater_meta_client_autorest.go @@ -9,11 +9,12 @@ import ( ) type metaClientAutorestTemplater struct { - serviceName string - apiVersion string - resources map[string]models.APIResource - source models.SourceDataOrigin - sourceType models.SourceDataType + apiVersionDirectoryName string + apiVersionPackageName string + resources map[string]models.APIResource + serviceName string + source models.SourceDataOrigin + sourceType models.SourceDataType } func (m metaClientAutorestTemplater) template() (*string, error) { @@ -35,7 +36,7 @@ func (m metaClientAutorestTemplater) template() (*string, error) { for _, resourceName := range resourceNames { variableName := fmt.Sprintf("%s%sClient", strings.ToLower(string(resourceName[0])), resourceName[1:]) - imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s/%s/%s"`, m.sourceType, strings.ToLower(m.serviceName), m.apiVersion, strings.ToLower(resourceName))) + imports = append(imports, fmt.Sprintf(`"github.com/hashicorp/go-azure-sdk/%s/%s/%s/%s"`, m.sourceType, strings.ToLower(m.serviceName), m.apiVersionDirectoryName, strings.ToLower(resourceName))) fields = append(fields, fmt.Sprintf("%[1]s *%[2]s.%[1]sClient", resourceName, strings.ToLower(resourceName))) clientInitializationTemplate := fmt.Sprintf(` %[1]s := %[2]s.New%[3]sClientWithBaseURI(endpoint) @@ -50,8 +51,6 @@ configureAuthFunc(&%[1]s.Client) sort.Strings(fields) sort.Strings(imports) - packageName := fmt.Sprintf("v%s", strings.ReplaceAll(m.apiVersion, "-", "_")) - out := fmt.Sprintf(`package %[1]s %[2]s @@ -72,6 +71,6 @@ func NewClientWithBaseURI(endpoint string, configureAuthFunc func(c *autorest.Cl %[6]s } } -`, packageName, *copyrightLines, strings.Join(imports, "\n"), strings.Join(fields, "\n"), strings.Join(clientInitialization, "\n"), strings.Join(assignments, "\n")) +`, m.apiVersionPackageName, *copyrightLines, strings.Join(imports, "\n"), strings.Join(fields, "\n"), strings.Join(clientInitialization, "\n"), strings.Join(assignments, "\n")) return &out, nil } From 88552699ab1df89e16efb9f7267f6db0c4ce0dc6 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 5 Aug 2024 19:51:30 +0100 Subject: [PATCH 036/117] generator-go-sdk: fix output path for resource packages --- tools/generator-go-sdk/internal/generator/data.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index dc0daa80404..cd346cc0852 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -78,7 +78,7 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { versionDirectoryName := strings.ToLower(i.VersionName) versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionDirectoryName) - resourceOutputPath := filepath.Join(versionOutputPath, versionDirectoryName) + resourceOutputPath := filepath.Join(versionOutputPath, resourcePackageName) useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) From 7467e1d682a961f45d6f8de7912d1cea0856850d Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 6 Aug 2024 01:41:38 +0100 Subject: [PATCH 037/117] importer-msgraph-metadata: rename file --- .../components/blacklisted/{blacklist.go => resource.go} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename tools/importer-msgraph-metadata/components/blacklisted/{blacklist.go => resource.go} (100%) diff --git a/tools/importer-msgraph-metadata/components/blacklisted/blacklist.go b/tools/importer-msgraph-metadata/components/blacklisted/resource.go similarity index 100% rename from tools/importer-msgraph-metadata/components/blacklisted/blacklist.go rename to tools/importer-msgraph-metadata/components/blacklisted/resource.go From 78cba7b4d3c694e84da70afea64340436e6fa9ab Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 6 Aug 2024 13:08:45 +0100 Subject: [PATCH 038/117] generator-go-sdk: more API version name/directory fixes --- .../generator-go-sdk/internal/cmd/generate.go | 3 ++ .../internal/generator/data.go | 49 ++++++++++++------- .../internal/generator/settings.go | 8 +++ .../internal/generator/templater_readme.go | 2 +- .../internal/generator/templater_version.go | 11 +++-- 5 files changed, 50 insertions(+), 23 deletions(-) diff --git a/tools/generator-go-sdk/internal/cmd/generate.go b/tools/generator-go-sdk/internal/cmd/generate.go index d20e0f76ec3..8c46041d351 100644 --- a/tools/generator-go-sdk/internal/cmd/generate.go +++ b/tools/generator-go-sdk/internal/cmd/generate.go @@ -59,6 +59,9 @@ func (g GenerateCommand) Run(args []string) int { } if g.sourceDataType == models.MicrosoftGraphSourceDataType { + input.settings.CanonicalApiVersions = map[string]string{ + "stable": "v1.0", + } input.settings.VersionsToGenerateCommonTypes = map[string]models.SourceDataOrigin{ "stable": models.MicrosoftGraphMetaDataSourceDataOrigin, "beta": models.MicrosoftGraphMetaDataSourceDataOrigin, diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index cd346cc0852..789362bb6d0 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -43,6 +43,10 @@ type GeneratorData struct { // API Version is the default API version for this Resource apiVersion string + // canonicalApiVersion is the upstream API version, which is set when different to the internal API version in Pandora + // example of this would be MS Graph v1.0, which Pandora internally refers to as "stable" + canonicalApiVersion *string + // models is a map of Model Name (key) to SDKModel (value) describing // the models used in this Resource models map[string]models.SDKModel @@ -64,6 +68,12 @@ type GeneratorData struct { // sourceType is the source data type and is the SDK package name sourceType models.SourceDataType + // the directory name of the package for the API version + versionDirectoryName string + + // the package name for the API version + versionPackageName string + // whether this is a data plane SDK (omits certain Resource Manager specific features, currently used in ID parsers) isDataPlane bool @@ -80,6 +90,11 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { versionOutputPath := filepath.Join(i.OutputDirectory, servicePackageName, versionDirectoryName) resourceOutputPath := filepath.Join(versionOutputPath, resourcePackageName) + versionPackageName := strings.ToLower(strings.ReplaceAll(i.VersionName, "-", "_")) + if regexp.MustCompile(`^[0-9]+`).MatchString(versionPackageName) { + versionPackageName = fmt.Sprintf("v%s", versionPackageName) + } + useNewBaseLayer := settings.ShouldUseNewBaseLayer(i.ServiceName, i.VersionName) var commonTypesPackageName, commonTypesIncludePath *string @@ -90,6 +105,7 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { return GeneratorData{ apiVersion: i.VersionName, + canonicalApiVersion: settings.CanonicalApiVersion(i.VersionName), baseClientPackage: baseClientPackageForSdk(i.Type), commonTypes: i.CommonTypes, commonTypesIncludePath: commonTypesIncludePath, @@ -122,12 +138,6 @@ type VersionGeneratorData struct { // This is the directory for the API version where the meta client should be output // for example {workingDir}/{service}/{version} versionOutputPath string - - // the directory name of the package for the API version - versionDirectoryName string - - // the package name for the API version - versionPackageName string } func (i VersionGeneratorInput) generatorData(settings Settings) VersionGeneratorData { @@ -146,22 +156,23 @@ func (i VersionGeneratorInput) generatorData(settings Settings) VersionGenerator return VersionGeneratorData{ GeneratorData: GeneratorData{ - apiVersion: i.VersionName, - baseClientPackage: baseClientPackageForSdk(i.Type), - commonTypes: i.CommonTypes, - constants: i.CommonTypes.Constants, - isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), - models: i.CommonTypes.Models, - packageName: versionPackageName, - servicePackageName: strings.ToLower(i.ServiceName), - source: i.Source, - sourceType: i.Type, - useNewBaseLayer: useNewBaseLayer, + apiVersion: i.VersionName, + canonicalApiVersion: settings.CanonicalApiVersion(i.VersionName), + baseClientPackage: baseClientPackageForSdk(i.Type), + commonTypes: i.CommonTypes, + constants: i.CommonTypes.Constants, + isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), + models: i.CommonTypes.Models, + packageName: versionPackageName, + servicePackageName: strings.ToLower(i.ServiceName), + source: i.Source, + sourceType: i.Type, + useNewBaseLayer: useNewBaseLayer, + versionDirectoryName: versionDirectoryName, + versionPackageName: versionPackageName, }, commonTypesOutputPath: commonTypesOutputPath, resources: i.Resources, versionOutputPath: versionOutputPath, - versionDirectoryName: versionDirectoryName, - versionPackageName: versionPackageName, } } diff --git a/tools/generator-go-sdk/internal/generator/settings.go b/tools/generator-go-sdk/internal/generator/settings.go index 65b77ce37b9..ee4d720e606 100644 --- a/tools/generator-go-sdk/internal/generator/settings.go +++ b/tools/generator-go-sdk/internal/generator/settings.go @@ -10,11 +10,19 @@ import ( ) type Settings struct { + CanonicalApiVersions map[string]string CommonTypesPackageName string VersionsToGenerateCommonTypes map[string]models.SourceDataOrigin servicesUsingOldBaseLayer map[string]struct{} } +func (s *Settings) CanonicalApiVersion(version string) *string { + if v, ok := s.CanonicalApiVersions[version]; ok { + return &v + } + return nil +} + func (s *Settings) UseOldBaseLayerFor(serviceNames ...string) { if s.servicesUsingOldBaseLayer == nil { s.servicesUsingOldBaseLayer = map[string]struct{}{} diff --git a/tools/generator-go-sdk/internal/generator/templater_readme.go b/tools/generator-go-sdk/internal/generator/templater_readme.go index 4e060b98e85..809a9afff26 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme.go @@ -62,7 +62,7 @@ This readme covers example usages, but further information on [using this SDK ca '''go %[5]s ''' -`, data.sourceType, data.servicePackageName, data.apiVersion, data.packageName, strings.Join(importLines, "\n")) +`, data.sourceType, data.servicePackageName, data.versionDirectoryName, data.packageName, strings.Join(importLines, "\n")) } func (r readmeTemplater) clientInitialization(packageName, clientName string) string { diff --git a/tools/generator-go-sdk/internal/generator/templater_version.go b/tools/generator-go-sdk/internal/generator/templater_version.go index ce35d8e2145..ced240f89c9 100644 --- a/tools/generator-go-sdk/internal/generator/templater_version.go +++ b/tools/generator-go-sdk/internal/generator/templater_version.go @@ -13,17 +13,22 @@ func (c versionTemplater) template(data GeneratorData) (*string, error) { return nil, fmt.Errorf("retrieving copyright lines: %+v", err) } + apiVersion := data.apiVersion + if data.canonicalApiVersion != nil { + apiVersion = *data.canonicalApiVersion + } + template := fmt.Sprintf(`package %[1]s import "fmt" -%[3]s +%[4]s const defaultApiVersion = "%[2]s" func userAgent() string { - return fmt.Sprintf("hashicorp/go-azure-sdk/%[1]s/%%s", defaultApiVersion) + return "hashicorp/go-azure-sdk/%[1]s/%[3]s" } -`, data.packageName, data.apiVersion, *copyrightLines) +`, data.packageName, apiVersion, data.apiVersion, *copyrightLines) return &template, nil } From 569b3ca83f25ff6e92b873d11e124193ab01a003 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 6 Aug 2024 14:04:00 +0100 Subject: [PATCH 039/117] importer-msgraph-metadata: more renames --- .../pipeline/{run_importer.go => importer.go} | 0 .../internal/pipeline/interface.go | 26 ++++++++++++++ .../internal/pipeline/pipeline.go | 34 ------------------- 3 files changed, 26 insertions(+), 34 deletions(-) rename tools/importer-msgraph-metadata/internal/pipeline/{run_importer.go => importer.go} (100%) delete mode 100644 tools/importer-msgraph-metadata/internal/pipeline/pipeline.go diff --git a/tools/importer-msgraph-metadata/internal/pipeline/run_importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go similarity index 100% rename from tools/importer-msgraph-metadata/internal/pipeline/run_importer.go rename to tools/importer-msgraph-metadata/internal/pipeline/importer.go diff --git a/tools/importer-msgraph-metadata/internal/pipeline/interface.go b/tools/importer-msgraph-metadata/internal/pipeline/interface.go index 858b4a94235..0d804aa5b1c 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/interface.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/interface.go @@ -6,7 +6,9 @@ package pipeline import ( "fmt" + "github.com/getkin/kin-openapi/openapi3" "github.com/hashicorp/pandora/tools/data-api-repository/repository" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) @@ -30,3 +32,27 @@ func Run(input RunInput) error { return runImporter(input, *metadataGitSha) } + +type pipeline struct { + apiVersion string + constants parser.Constants + repo repository.Repository + metadataGitSha string + models parser.Models + outputDirectory string + resources map[string]parser.Resources + resourceIds parser.ResourceIds + spec *openapi3.T +} + +type pipelineForService struct { + pipeline + service string +} + +func (p pipeline) ForService(serviceName string) pipelineForService { + return pipelineForService{ + pipeline: p, + service: serviceName, + } +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go b/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go deleted file mode 100644 index d13cf8a3e7a..00000000000 --- a/tools/importer-msgraph-metadata/internal/pipeline/pipeline.go +++ /dev/null @@ -1,34 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package pipeline - -import ( - "github.com/getkin/kin-openapi/openapi3" - "github.com/hashicorp/pandora/tools/data-api-repository/repository" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -type pipeline struct { - apiVersion string - constants parser.Constants - repo repository.Repository - metadataGitSha string - models parser.Models - outputDirectory string - resources map[string]parser.Resources - resourceIds parser.ResourceIds - spec *openapi3.T -} - -type pipelineForService struct { - pipeline - service string -} - -func (p pipeline) ForService(serviceName string) pipelineForService { - return pipelineForService{ - pipeline: p, - service: serviceName, - } -} From e8a1d3be39590e55fa1cee2dae7d5d13b80d76c6 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 11 Aug 2024 03:14:21 +0100 Subject: [PATCH 040/117] data-api: add a `nullable` option for data types --- .../internal/models/object_definition.go | 3 +++ .../transforms/sdk_field_object_definition.go | 2 ++ .../golang_type_for_sdk_object_definition.go | 18 ++++++++++++++++++ .../v1/models/sdk_object_definition.go | 3 +++ 4 files changed, 26 insertions(+) diff --git a/tools/data-api-repository/repository/internal/models/object_definition.go b/tools/data-api-repository/repository/internal/models/object_definition.go index d25ce9fe449..949a846a6e8 100644 --- a/tools/data-api-repository/repository/internal/models/object_definition.go +++ b/tools/data-api-repository/repository/internal/models/object_definition.go @@ -8,6 +8,9 @@ type ObjectDefinition struct { // ObjectDefinitionType defines what kind of ObjectDefinition this is, such as a Reference, String or List Type ObjectDefinitionType `json:"type"` + // Nullable specifies that this type should be unset by sending `null` as the JSON value. + Nullable bool `json:"nullable"` + // ReferenceName is the name of the Constant or Model that this is a reference to ReferenceName *string `json:"referenceName"` diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go b/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go index 5aa61ffbc55..ee64e54cdc8 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_field_object_definition.go @@ -17,6 +17,7 @@ func mapSDKFieldObjectDefinitionFromRepository(input repositoryModels.ObjectDefi return nil, fmt.Errorf("internal-error: missing a mapping for the ObjectDefinitionType %q", string(input.Type)) } output := sdkModels.SDKObjectDefinition{ + Nullable: input.Nullable, ReferenceName: input.ReferenceName, ReferenceNameIsCommonType: input.ReferenceNameIsCommonType, Type: typeVal, @@ -39,6 +40,7 @@ func mapSDKFieldObjectDefinitionToRepository(input sdkModels.SDKObjectDefinition } output := repositoryModels.ObjectDefinition{ + Nullable: input.Nullable, Type: typeVal, ReferenceName: nil, NestedItem: nil, diff --git a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go index 35aa1ae18d9..bac359bbbda 100644 --- a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go +++ b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go @@ -67,6 +67,24 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa return pointer.To(out), nil } + if input.Nullable { + nullableSdkObjectDefinitionTypesToValues := map[models.SDKObjectDefinitionType]string{ + // Simple Types + models.BooleanSDKObjectDefinitionType: "nullable.Type[bool]", + models.DateTimeSDKObjectDefinitionType: "nullable.Type[string]", // intentional since we have cast methods one way or the other + models.FloatSDKObjectDefinitionType: "nullable.Type[float64]", + models.IntegerSDKObjectDefinitionType: "nullable.Type[int64]", + models.StringSDKObjectDefinitionType: "nullable.Type[string]", + + // Complex Types + models.LocationSDKObjectDefinitionType: "nullable.Type[string]", + } + + if v, ok := nullableSdkObjectDefinitionTypesToValues[input.Type]; ok { + return pointer.To(v), nil + } + } + sdkObjectDefinitionTypesToValues := map[models.SDKObjectDefinitionType]string{ // Simple Types models.BooleanSDKObjectDefinitionType: "bool", diff --git a/tools/data-api-sdk/v1/models/sdk_object_definition.go b/tools/data-api-sdk/v1/models/sdk_object_definition.go index bbaadd4c91e..bafe6af10d1 100644 --- a/tools/data-api-sdk/v1/models/sdk_object_definition.go +++ b/tools/data-api-sdk/v1/models/sdk_object_definition.go @@ -28,4 +28,7 @@ type SDKObjectDefinition struct { // Simple type (e.g. a String/Integer), a Reference to a Constant/Model or a more complex object // (e.g. a CommonSchema type [such as a SystemAssignedIdentity] - or a List/Dictionary). Type SDKObjectDefinitionType `json:"type"` + + // Nullable specifies that this type should be unset by sending `null` as the JSON value. + Nullable bool `json:"nullable"` } From c6b5c01ccd799ae16c1c9602c0aaac1a5e644f83 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 11 Aug 2024 03:14:46 +0100 Subject: [PATCH 041/117] generator-go-sdk: support for generating models with fields having nullable types --- .../generator-go-sdk/internal/generator/templater_models.go | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 46335da66e3..217785ca349 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -54,6 +54,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" %[2]s ) @@ -224,7 +225,9 @@ func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDe } // TODO: proper support for ReadOnly fields, which is likely to necessitate a custom marshal func if isOptional || fieldDetails.ReadOnly { - fieldType = fmt.Sprintf("*%s", fieldType) + if !strings.HasPrefix(fieldType, "nullable.") { + fieldType = fmt.Sprintf("*%s", fieldType) + } jsonDetails += ",omitempty" } From da2d5639948c1c39c8c3c0061f1b2135786a40f0 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 11 Aug 2024 03:15:16 +0100 Subject: [PATCH 042/117] importer-msgraph-metadata: support for nullable field types --- .../components/parser/types.go | 61 +++++++++++-------- .../internal/pipeline/task_parse_resources.go | 9 +++ 2 files changed, 43 insertions(+), 27 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 62c33c5205e..9c9f51bf3a2 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -132,9 +132,10 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { JsonName: field.JsonField, ObjectDefinition: *objectDefinition, + ReadOnly: field.ReadOnly, + // TODO work these out Optional: true, - ReadOnly: false, Required: false, Sensitive: false, } @@ -154,20 +155,24 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { } type ModelField struct { - Title string - Type *DataType - Description string - Default interface{} - ItemType *DataType - ConstantName *string - ModelName *string - JsonField string + Title string + Type *DataType + Description string + Default interface{} + ReadOnly bool + WriteOnly bool + Nullable bool + AllowEmptyValue bool + ItemType *DataType + ConstantName *string + ModelName *string + JsonField string } func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObjectDefinition, error) { if f.ConstantName != nil { return &sdkModels.SDKObjectDefinition{ - NestedItem: nil, + Nullable: f.Nullable, ReferenceName: f.ConstantName, Type: sdkModels.ReferenceSDKObjectDefinitionType, }, nil @@ -192,7 +197,7 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj } return &sdkModels.SDKObjectDefinition{ - NestedItem: nil, + Nullable: f.Nullable, ReferenceName: f.ModelName, ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), Type: sdkModels.ReferenceSDKObjectDefinitionType, @@ -210,7 +215,7 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - NestedItem: nil, + Nullable: f.Nullable, ReferenceName: f.ModelName, ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), Type: sdkModels.ReferenceSDKObjectDefinitionType, @@ -224,24 +229,21 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj // TODO validate constant exists return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - NestedItem: nil, + Nullable: f.Nullable, ReferenceName: f.ConstantName, Type: sdkModels.ReferenceSDKObjectDefinitionType, }, - ReferenceName: nil, - Type: sdkModels.ListSDKObjectDefinitionType, + Type: sdkModels.ListSDKObjectDefinitionType, }, nil } if f.ItemType != nil { return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - NestedItem: nil, - ReferenceName: nil, - Type: f.ItemType.DataApiSdkObjectDefinitionType(), + Nullable: f.Nullable, + Type: f.ItemType.DataApiSdkObjectDefinitionType(), }, - ReferenceName: nil, - Type: sdkModels.ListSDKObjectDefinitionType, + Type: sdkModels.ListSDKObjectDefinitionType, }, nil } @@ -249,9 +251,8 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj } return &sdkModels.SDKObjectDefinition{ - NestedItem: nil, - ReferenceName: nil, - Type: f.Type.DataApiSdkObjectDefinitionType(), + Nullable: f.Nullable, + Type: f.Type.DataApiSdkObjectDefinitionType(), }, nil } @@ -681,12 +682,18 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta models[name] = &model for jsonField, schemaRef := range input.Schemas { + // maintainer note: this is a good place to add a breakpoint for inspecting model fields and constants as they are processed + // example breakpoint condition: name=="AdministrativeUnit" if schema := schemaRef.Value; schema != nil { field := ModelField{ - Title: cases.Title(language.AmericanEnglish, cases.NoLower).String(jsonField), - Description: schema.Description, - Default: schema.Default, - JsonField: jsonField, + Title: cases.Title(language.AmericanEnglish, cases.NoLower).String(jsonField), + Description: schema.Description, + Default: schema.Default, + ReadOnly: schemaRef.Value.ReadOnly, + WriteOnly: schemaRef.Value.WriteOnly, + Nullable: schemaRef.Value.Nullable, + AllowEmptyValue: schemaRef.Value.AllowEmptyValue, + JsonField: jsonField, } if field.Title == "" { diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index aa90c7db510..afc40d4335d 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -36,6 +36,11 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model continue } + // maintainer note: this is a good place to add a breakpoint for inspecting individual resources and their operations + // example breakpoint conditions: + // p.service=="administrativeUnits" + // path=="/applications/{application-id}/owners" + // len(path)>62 && path[:62]=="/directory/administrativeUnits/{administrativeUnit-id}/members" parsedPath := parser.NewResourceId(path, operationTags) lastSegment := parsedPath.Segments[len(parsedPath.Segments)-1] @@ -69,6 +74,10 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + // maintainer note: this is a good place to add a breakpoint for inspecting individual resources and their operations + // example breakpoint conditions: + // resourceName=="ApplicationOwner" + // len(resourceName)>27 && resourceName[:27]=="DirectoryAdministrativeUnit" if _, ok := resources[resourceName]; !ok { // Create a new resource if not already encountered logging.Infof(fmt.Sprintf("Found new resource %q (category %q, service %q, version %q)", resourceName, resourceCategory, p.service, p.apiVersion)) From 8584a9e79c29f92573f671336b6ce295a21f2d9c Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 11 Aug 2024 03:42:01 +0100 Subject: [PATCH 043/117] importer-msgraph-metadata: add an explicit `ODataId` field to every model --- .../components/parser/types.go | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 9c9f51bf3a2..9b9cdebd47c 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -672,8 +672,18 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta return models, constants } + // Add an explicit ODataId field to each model, since it is inconsistently defined in the API specs. This won't be + // a valid field for every model, but it's impossible to tell which models support it, and it's effectively + // harmless to leave it in so long as it has the `omitempty` struct tag in the generated SDK. model := Model{ - Fields: make(map[string]*ModelField), + Fields: map[string]*ModelField{ + "ODataId": { + Title: "ODataId", + Type: pointer.To(DataTypeString), + Default: "", + JsonField: "@odata.id", + }, + }, Common: common, Prefix: input.Prefix, } From 76465fe041698dec17964b6f9ff1a38e06db9ba7 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 12 Aug 2024 02:02:56 +0100 Subject: [PATCH 044/117] generator-go-sdk: log gofmt/goimports invocations --- tools/generator-go-sdk/internal/generator/service.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/tools/generator-go-sdk/internal/generator/service.go b/tools/generator-go-sdk/internal/generator/service.go index 6f9b99eb7e4..3da6c946b97 100644 --- a/tools/generator-go-sdk/internal/generator/service.go +++ b/tools/generator-go-sdk/internal/generator/service.go @@ -119,12 +119,14 @@ func (s *Generator) GenerateCommonTypes(input VersionGeneratorInput) error { } func runGoFmt(path string) { + logging.Debugf("Running gofmt -w %s..", path) cmd := exec.Command("gofmt", "-w", path) _ = cmd.Start() _ = cmd.Wait() } func runGoImports(path string) { + logging.Debugf("Running goimports -w %s..", path) cmd := exec.Command("goimports", "-w", path) _ = cmd.Start() _ = cmd.Wait() From a7efc77598db52cae381c6ae969c7dbe09589b86 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 12 Aug 2024 02:05:08 +0100 Subject: [PATCH 045/117] importer-msgraph-metadata: add an `@odata.bind` field for referenced `DirectoryObject` models --- .../components/parser/types.go | 28 +++++++++++++++++++ 1 file changed, 28 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 9b9cdebd47c..7ba8d3d3a3f 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -712,11 +712,13 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta result, _ := FlattenSchemaRef(schemaRef, nil) + // Determine any enumeration for the field enum := parseEnum(schema.Enum) if result != nil && len(result.Enum) > 0 && len(enum) == 0 { enum = parseEnum(result.Enum) } + // Find the corresponding model when the field refers to it, and set the model name for the field if result != nil && result.Title != "" && result.Schemas != nil { if _, ok := models[result.Title]; !ok { models, constants = Schemas(*result, result.Title, models, constants, common) @@ -724,16 +726,19 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta field.ModelName = &result.Title } + // Determine the item type for collections if schema.Items != nil && schema.Items.Value != nil && schema.Items.Value.Type != "" { field.ItemType = FieldType(schema.Items.Value.Type, schema.Items.Value.Format, field.ModelName != nil) } + // Match the field type to the referenced object, or set a basic type if result != nil && schema.Type == "" && schema.Format == "" && (result.Type != "" || result.Format != "") { field.Type = FieldType(result.Type, result.Format, field.ModelName != nil) } else { field.Type = FieldType(schema.Type, schema.Format, field.ModelName != nil) } + // Set the field type for enums; typically strings for constants, but could be any valid type if result != nil && field.Type != nil && *field.Type == DataTypeArray && len(enum) > 0 && (result.Type != "" || result.Format != "") { field.ItemType = FieldType(result.Type, result.Format, field.ModelName != nil) } @@ -743,6 +748,7 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta // This leads to some excessively long constant names, it is what it is. field.ConstantName = pointer.To(name + field.Title) + // Add the enumeration as a constant constants[*field.ConstantName] = &Constant{ Enum: enum, Type: field.Type, @@ -754,10 +760,32 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta continue } + // Insert an "@odata.bind" field where a field or collection refers to a DirectoryObject. The MS Graph + // OpenAPI spec unfortunately does not document relationships between entities. + if field.ModelName != nil && strings.EqualFold(*field.ModelName, "DirectoryObject") { + bindFieldName := fmt.Sprintf("%s_ODataBind", normalize.CleanName(jsonField)) + + var fieldType, itemType *DataType + + fieldType = pointer.To(DataTypeString) + if *field.Type == DataTypeArray { + fieldType = pointer.To(DataTypeArray) + itemType = pointer.To(DataTypeString) + } + + model.Fields[bindFieldName] = &ModelField{ + Title: bindFieldName, + Type: fieldType, + ItemType: itemType, + JsonField: fmt.Sprintf("%s@odata.bind", jsonField), + } + } + model.Fields[normalize.CleanName(jsonField)] = &field } } + // Abandon the model if it has no fields if !model.IsValid() { delete(models, name) } From 63b5705946a4fe88a90e50f9797103f4d147a9a5 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 12 Aug 2024 02:06:20 +0100 Subject: [PATCH 046/117] importer-msgraph-metadata: add data workaround for missing fields in `Application` model in beta API See: https://github.com/microsoftgraph/msgraph-metadata/issues/273 --- .../components/parser/types.go | 8 +-- .../workarounds/workaround_application.go | 55 +++++++++++++++++++ .../components/workarounds/workarounds.go | 42 ++++++++++++++ .../internal/pipeline/importer.go | 6 ++ 4 files changed, 107 insertions(+), 4 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_application.go create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workarounds.go diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 7ba8d3d3a3f..8d25db2d2e2 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -121,7 +121,7 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { if objectDefinition == nil { //return nil, fmt.Errorf("could not determine SDKObjectDefinition for field: %s", fieldName) - logging.Warnf("could not determine SDKObjectDefinition for field %q, skipping", fieldName) + logging.Warnf("Could not determine SDKObjectDefinition for field %q, skipping", fieldName) continue } @@ -193,7 +193,7 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj } if !models[*f.ModelName].IsValid() { - logging.Warnf("skipping field %q with type Model as the referenced model %q is invalid", f.Title, *f.ModelName) + logging.Warnf("Skipping field %q with type Model as the referenced model %q is invalid", f.Title, *f.ModelName) } return &sdkModels.SDKObjectDefinition{ @@ -210,7 +210,7 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj } if !models[*f.ModelName].IsValid() { - logging.Warnf("skipping field %q with type Array[Model] as the referenced model %q is invalid", f.Title, *f.ModelName) + logging.Warnf("Skipping field %q with type Array[Model] as the referenced model %q is invalid", f.Title, *f.ModelName) } return &sdkModels.SDKObjectDefinition{ @@ -756,7 +756,7 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta } if field.Type == nil { - logging.Warnf("skipping field %q in model %q because Type is nil", name, field.Title) + logging.Warnf("Skipping field %q in model %q because Type is nil", name, field.Title) continue } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go new file mode 100644 index 00000000000..bf20406379d --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go @@ -0,0 +1,55 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" +) + +var _ workaround = workaroundApplication{} + +// workaroundApplication works around missing fields in the Application model for the beta API. +// 1. Missing `oauth2RequirePostResponse` field. +// Upstream PR: https://github.com/microsoftgraph/msgraph-metadata/issues/273 +// 2. Missing `applicationTemplateId` field. +type workaroundApplication struct{} + +func (workaroundApplication) IsApplicable(apiVersion, modelName string, model *parser.Model) bool { + return apiVersion == versions.ApiVersionBeta && modelName == "Application" +} + +func (workaroundApplication) Name() string { + return "Application / missing fields in beta" +} + +func (workaroundApplication) Process(model *parser.Model) error { + // Add the `oauth2RequirePostResponse` field if missing + if _, ok := model.Fields["OAuth2RequirePostResponse"]; !ok { + model.Fields["OAuth2RequirePostResponse"] = &parser.ModelField{ + Title: "OAuth2RequirePostResponse", + Description: "Specifies whether, as part of OAuth 2.0 token requests, Microsoft Entra ID allows POST requests, as opposed to GET requests. The default is false, which specifies that only GET requests are allowed.", + Type: pointer.To(parser.DataTypeBool), + JsonField: "oauth2RequirePostResponse", + } + } + + // Add the `applicationTemplateId` field if missing + if _, ok := model.Fields["ApplicationTemplateId"]; !ok { + fmt.Println("fo") + + model.Fields["ApplicationTemplateId"] = &parser.ModelField{ + Title: "ApplicationTemplateId", + Description: "Unique identifier of the applicationTemplate. Supports $filter (eq, not, ne). Read-only. null if the app wasn't created from an application template.", + Type: pointer.To(parser.DataTypeString), + JsonField: "applicationTemplateId", + Nullable: true, + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go new file mode 100644 index 00000000000..1b232d9e8f1 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -0,0 +1,42 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" +) + +var workarounds = []workaround{ + workaroundApplication{}, +} + +type workaround interface { + // IsApplicable determines whether this workaround is applicable for this AzureApiDefinition + IsApplicable(string, string, *parser.Model) bool + + // Name returns the Service Name and associated Pull Request number + Name() string + + // Process takes the apiDefinition and applies the Workaround to this AzureApiDefinition + Process(*parser.Model) error +} + +func ApplyWorkarounds(apiVersion string, models parser.Models) error { + logging.Tracef("Processing Data Workarounds..") + for modelName, model := range models { + for _, fix := range workarounds { + if fix.IsApplicable(apiVersion, modelName, model) { + logging.Tracef("Applying Data Workaround %q to Model %q", fix.Name(), modelName) + if err := fix.Process(model); err != nil { + return fmt.Errorf("applying Data Workaround %q to Model %q: %v", fix.Name(), modelName, err) + } + } + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go index 0fcc6c19cde..214017e296d 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/importer.go @@ -13,6 +13,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/tags" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/workarounds" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" "github.com/hashicorp/pandora/tools/sdk/config/services" ) @@ -67,6 +68,11 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha return err } + logging.Infof("Applying workarounds for invalid model definitions..") + if err = workarounds.ApplyWorkarounds(p.apiVersion, p.models); err != nil { + return err + } + logging.Infof("Parsing resource IDs...") p.resourceIds, err = parser.ParseResourceIDs(p.spec.Paths, nil) if err != nil { From 5b4b818fcf3ccd748bd2d3605193e3c7e6f83d27 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 01:46:50 +0100 Subject: [PATCH 047/117] generator-go-sdk: support unmarshaling discriminated types directly for operations --- .../internal/generator/templater_methods.go | 124 +++++++++++------- 1 file changed, 75 insertions(+), 49 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index d78c812c0b7..c4c8ea2d467 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -602,33 +602,55 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin return pointer.To(""), nil } - if c.operation.ResponseObject != nil { - golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) - if err != nil { - return nil, fmt.Errorf("determing golang type name for response object: %+v", err) + if c.operation.ResponseObject == nil { + return &output, nil + } + + golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) + if err != nil { + return nil, fmt.Errorf("determing golang type name for response object: %+v", err) + } + typeName := *golangTypeName + + discriminatedTypeParentName := "" + + var model *models.SDKModel + modelPackage := "" + modelName := typeName + if s := strings.SplitN(modelName, ".", 2); len(s) == 2 { + modelPackage = s[0] + modelName = s[1] + } + if m, ok := data.models[modelName]; ok { + model = &m + } else if m, ok = data.commonTypes.Models[modelName]; ok { + model = &m + } + + if model != nil { + // it's either a parent model + if model.FieldNameContainingDiscriminatedValue != nil { + discriminatedTypeParentName = modelName + } + // or an implementation referencing a parent + if model.ParentTypeName != nil { + discriminatedTypeParentName = *model.ParentTypeName } - typeName := *golangTypeName - discriminatedTypeParentName := "" - if model, ok := data.models[typeName]; ok { - // it's either a parent model - if model.FieldNameContainingDiscriminatedValue != nil { - discriminatedTypeParentName = typeName - } - // or an implementation referencing a parent - if model.ParentTypeName != nil { - discriminatedTypeParentName = *model.ParentTypeName - } + if model.DiscriminatedValue != nil { + // in this instance this would be a discriminated implementation present in the response object + // as such we should use that directly, rather than calling the parents unmarshal function + discriminatedTypeParentName = "" + } + } - if model.DiscriminatedValue != nil { - // in this instance this would be a discriminated implementation present in the response object - // as such we should use that directly, rather than calling the parents unmarshal function - discriminatedTypeParentName = "" - } + if c.operation.FieldContainingPaginationDetails != nil { + unmarshaler := fmt.Sprintf("unmarshal%sImplementation", modelName) + if modelPackage != "" { + unmarshaler = fmt.Sprintf("%s.unmarshal%sImplementation", modelPackage, modelName) } - if c.operation.FieldContainingPaginationDetails != nil { - output = fmt.Sprintf(` + output = fmt.Sprintf(` var values struct { Values *[]%[1]s %[2]s } @@ -639,10 +661,10 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin result.Model = values.Values `, typeName, "`json:\"value\"`") - if discriminatedTypeParentName != "" { - output = fmt.Sprintf(` + if discriminatedTypeParentName != "" { + output = fmt.Sprintf(` var values struct { - Values *[]json.RawMessage %[2]s + Values *[]json.RawMessage %[3]s } if err = resp.Unmarshal(&values); err != nil { return @@ -651,7 +673,7 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin temp := make([]%[1]s, 0) if values.Values != nil { for i, v := range *values.Values { - val, err := unmarshal%[1]sImplementation(v) + val, err := %[2]s(v) if err != nil { err = fmt.Errorf("unmarshalling item %%d for %[1]s (%%q): %%+v", i, v, err) return result, err @@ -660,49 +682,53 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin } } result.Model = &temp -`, typeName, "`json:\"value\"`") - } +`, typeName, unmarshaler, "`json:\"value\"`") + } + + return &output, nil + } - return &output, nil + // when this is a Discriminated Type (either the Parent or the Implementation, call the `unmarshal` func + // for the relevant Parent + if discriminatedTypeParentName != "" { + unmarshaler := fmt.Sprintf("unmarshal%sImplementation", discriminatedTypeParentName) + if modelPackage != "" { + unmarshaler = fmt.Sprintf("%s.unmarshal%sImplementation", modelPackage, discriminatedTypeParentName) } - // when this is a Discriminated Type (either the Parent or the Implementation, call the `unmarshal` func - // for the relevant Parent - if discriminatedTypeParentName != "" { - output = fmt.Sprintf(` + output = fmt.Sprintf(` var respObj json.RawMessage if err = resp.Unmarshal(&respObj); err != nil { return } - model, err := unmarshal%sImplementation(respObj) + model, err := %s(respObj) if err != nil { return } result.Model = &model -`, discriminatedTypeParentName) - } else { - responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) - if err != nil { - return nil, fmt.Errorf("determing golang type name for response object: %+v", err) - } - if responseModelType != nil { - if c.operation.FieldContainingPaginationDetails != nil { - output = fmt.Sprintf(` +`, unmarshaler) + } else { + responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) + if err != nil { + return nil, fmt.Errorf("determing golang type name for response object: %+v", err) + } + if responseModelType != nil { + if c.operation.FieldContainingPaginationDetails != nil { + output = fmt.Sprintf(` var model []%s`, *responseModelType) - } else { - output = fmt.Sprintf(` + } else { + output = fmt.Sprintf(` var model %s`, *responseModelType) - } - output = fmt.Sprintf(`%s - result.Model = &model`, output) } output = fmt.Sprintf(`%s + result.Model = &model`, output) + } + output = fmt.Sprintf(`%s if err = resp.Unmarshal(result.Model); err != nil { return } `, output) - } } return &output, nil From b44f8e371073b0f9183fde8922c01a181a86364d Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 01:48:55 +0100 Subject: [PATCH 048/117] importer-msgraph-metadata: support for discriminated types --- .../components/parser/types.go | 83 +++++++++++-------- 1 file changed, 49 insertions(+), 34 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 8d25db2d2e2..dda87e52d74 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -12,8 +12,6 @@ import ( sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" - "golang.org/x/text/cases" - "golang.org/x/text/language" ) /* =================== @@ -98,9 +96,12 @@ type Constant struct { } type Model struct { - Fields map[string]*ModelField - Common bool - Prefix string + Fields map[string]*ModelField + Common bool + Prefix string + TypeField *string + TypeValue *string + ParentModelName *string } func (m *Model) IsValid() bool { @@ -126,7 +127,7 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { } sdkFields[fieldName] = sdkModels.SDKField{ - ContainsDiscriminatedValue: false, + ContainsDiscriminatedValue: field.DiscriminatedValue, DateFormat: nil, Description: field.Description, JsonName: field.JsonField, @@ -145,28 +146,28 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { return nil, nil } - // TODO support discriminated types (good example: conditional access named locations) return &sdkModels.SDKModel{ - DiscriminatedValue: nil, - FieldNameContainingDiscriminatedValue: nil, + DiscriminatedValue: m.TypeValue, + FieldNameContainingDiscriminatedValue: m.TypeField, Fields: sdkFields, - ParentTypeName: nil, + ParentTypeName: m.ParentModelName, }, nil } type ModelField struct { - Title string - Type *DataType - Description string - Default interface{} - ReadOnly bool - WriteOnly bool - Nullable bool - AllowEmptyValue bool - ItemType *DataType - ConstantName *string - ModelName *string - JsonField string + Title string + Type *DataType + Description string + Default interface{} + ReadOnly bool + WriteOnly bool + Nullable bool + AllowEmptyValue bool + DiscriminatedValue bool + ItemType *DataType + ConstantName *string + ModelName *string + JsonField string } func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObjectDefinition, error) { @@ -672,9 +673,9 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta return models, constants } - // Add an explicit ODataId field to each model, since it is inconsistently defined in the API specs. This won't be - // a valid field for every model, but it's impossible to tell which models support it, and it's effectively - // harmless to leave it in so long as it has the `omitempty` struct tag in the generated SDK. + // Add an explicit ODataId and ODataType field to each model, since it is inconsistently defined in the API specs. + // This won't be valid for every model, but it's impossible to tell which models support them, and it's effectively + // harmless to leave these in so long as they have the `omitempty` struct tag in the generated SDK. model := Model{ Fields: map[string]*ModelField{ "ODataId": { @@ -683,6 +684,12 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta Default: "", JsonField: "@odata.id", }, + "ODataType": { + Title: "ODataType", + Type: pointer.To(DataTypeString), + Default: "", + JsonField: "@odata.type", + }, }, Common: common, Prefix: input.Prefix, @@ -696,7 +703,7 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta // example breakpoint condition: name=="AdministrativeUnit" if schema := schemaRef.Value; schema != nil { field := ModelField{ - Title: cases.Title(language.AmericanEnglish, cases.NoLower).String(jsonField), + Title: normalize.CleanName(jsonField), Description: schema.Description, Default: schema.Default, ReadOnly: schemaRef.Value.ReadOnly, @@ -726,9 +733,22 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta field.ModelName = &result.Title } - // Determine the item type for collections - if schema.Items != nil && schema.Items.Value != nil && schema.Items.Value.Type != "" { - field.ItemType = FieldType(schema.Items.Value.Type, schema.Items.Value.Format, field.ModelName != nil) + // Handle items for collections + if schema.Items != nil && schema.Items.Value != nil { + itemsResult, _ := FlattenSchemaRef(schema.Items, nil) + + // Determine any type for the items + if schema.Items.Value.Type != "" || schema.Items.Value.Format != "" { + field.ItemType = FieldType(schema.Items.Value.Type, schema.Items.Value.Format, field.ModelName != nil) + } + + // Find the corresponding model when the field (items) refers to it, and set the model name for the field items + if itemsResult != nil && itemsResult.Title != "" && itemsResult.Type == "object" { + if _, ok := models[result.Title]; !ok { + models, constants = Schemas(*result, result.Title, models, constants, common) + } + field.ModelName = &result.Title + } } // Match the field type to the referenced object, or set a basic type @@ -785,11 +805,6 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta } } - // Abandon the model if it has no fields - if !model.IsValid() { - delete(models, name) - } - return models, constants } From c61162353f45e1892af65f076367a6426c4e0596 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 01:50:24 +0100 Subject: [PATCH 049/117] importer-msgraph-metadata: workaround for making NamedLocation a discriminated type, since there is no useful data surfaced in the specs --- .../workarounds/workaround_application.go | 15 ++- .../workarounds/workaround_iprange.go | 97 +++++++++++++++++++ .../workarounds/workaround_namedlocation.go | 69 +++++++++++++ .../components/workarounds/workarounds.go | 21 ++-- .../internal/pipeline/importer.go | 8 +- 5 files changed, 188 insertions(+), 22 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go index bf20406379d..d27aa820d9b 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go @@ -19,15 +19,20 @@ var _ workaround = workaroundApplication{} // 2. Missing `applicationTemplateId` field. type workaroundApplication struct{} -func (workaroundApplication) IsApplicable(apiVersion, modelName string, model *parser.Model) bool { - return apiVersion == versions.ApiVersionBeta && modelName == "Application" -} - func (workaroundApplication) Name() string { return "Application / missing fields in beta" } -func (workaroundApplication) Process(model *parser.Model) error { +func (workaroundApplication) Process(apiVersion string, models parser.Models, constants parser.Constants) error { + if apiVersion != versions.ApiVersionBeta { + return nil + } + + model, ok := models["Application"] + if !ok { + return fmt.Errorf("`Application` model not found") + } + // Add the `oauth2RequirePostResponse` field if missing if _, ok := model.Fields["OAuth2RequirePostResponse"]; !ok { model.Fields["OAuth2RequirePostResponse"] = &parser.ModelField{ diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go new file mode 100644 index 00000000000..f66abd30778 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go @@ -0,0 +1,97 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ workaround = workaroundIPRange{} + +// workaroundIPRange implements discrimination +type workaroundIPRange struct{} + +func (workaroundIPRange) Name() string { + return "IPRange / discrimination" +} + +func (workaroundIPRange) Process(apiVersion string, models parser.Models, constants parser.Constants) error { + // microsoft.graph.ipRange model is empty and is omitted by the OpenAPI parser + models["IPRange"] = &parser.Model{ + Fields: map[string]*parser.ModelField{ + "ODataId": { + Title: "ODataId", + Type: pointer.To(parser.DataTypeString), + Default: "", + JsonField: "@odata.id", + }, + "ODataType": { + Title: "ODataType", + Type: pointer.To(parser.DataTypeString), + ConstantName: pointer.To("IPRangeType"), + Default: "", + DiscriminatedValue: true, + JsonField: "@odata.type", + }, + }, + Common: true, + TypeField: pointer.To("ODataType"), + } + + if _, ok := constants["IPRangeType"]; ok { + return fmt.Errorf("`IPRangeType` constant already defined") + } + + // Add constant values for the discriminated type value + constants["IPRangeType"] = &parser.Constant{ + Enum: []string{ + "#microsoft.graph.iPv4CidrRange", + "#microsoft.graph.iPv4Range", + "#microsoft.graph.iPv6CidrRange", + "#microsoft.graph.iPv6Range", + }, + Type: pointer.To(parser.DataTypeString), + } + + // Set the parent model and discriminated type value for IPv4CIDRRange + model, ok := models["IPv4CIDRRange"] + if !ok { + return fmt.Errorf("`IPv4CIDRRange` model not found") + } + model.ParentModelName = pointer.To("IPRange") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.iPv4CidrRange") + + // Set the parent model and discriminated type value for IPv4Range + model, ok = models["IPv4Range"] + if !ok { + return fmt.Errorf("`IPv4Range` model not found") + } + model.ParentModelName = pointer.To("IPRange") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.iPv4Range") + + // Set the parent model and discriminated type value for IPv6CIDRRange + model, ok = models["IPv6CIDRRange"] + if !ok { + return fmt.Errorf("`IPv6CIDRRange` model not found") + } + model.ParentModelName = pointer.To("IPRange") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.iPv6CidrRange") + + // Set the parent model and discriminated type value for IPv6Range + model, ok = models["IPv6Range"] + if !ok { + return fmt.Errorf("`IPv6Range` model not found") + } + model.ParentModelName = pointer.To("IPRange") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.iPv6Range") + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go new file mode 100644 index 00000000000..8a66c1994a8 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go @@ -0,0 +1,69 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ workaround = workaroundNamedLocation{} + +// workaroundNamedLocation implements discrimination +type workaroundNamedLocation struct{} + +func (workaroundNamedLocation) Name() string { + return "NamedLocation / discrimination" +} + +func (workaroundNamedLocation) Process(apiVersion string, models parser.Models, constants parser.Constants) error { + model, ok := models["NamedLocation"] + if !ok { + return fmt.Errorf("`NamedLocation` model not found") + } + + if _, ok = model.Fields["ODataType"]; !ok { + return fmt.Errorf("`ODataType` field not found in `NamedLocation` model") + } + + // Set the constant reference and discriminated flag for the @odata.type field + model.Fields["ODataType"].ConstantName = pointer.To("NamedLocationType") + model.Fields["ODataType"].DiscriminatedValue = true + model.TypeField = pointer.To("ODataType") + + if _, ok = constants["NamedLocationType"]; ok { + return fmt.Errorf("`NamedLocationType` constant already defined") + } + + // Add constant values for the discriminated type value + constants["NamedLocationType"] = &parser.Constant{ + Enum: []string{ + "#microsoft.graph.countryNamedLocation", + "#microsoft.graph.ipNamedLocation", + }, + Type: pointer.To(parser.DataTypeString), + } + + // Set the parent model and discriminated type value for CountryNamedLocation + model, ok = models["CountryNamedLocation"] + if !ok { + return fmt.Errorf("`CountryNamedLocation` model not found") + } + model.ParentModelName = pointer.To("NamedLocation") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.countryNamedLocation") + + // Set the parent model and discriminated type value for CountryNamedLocation + model, ok = models["IPNamedLocation"] + if !ok { + return fmt.Errorf("`IPNamedLocation` model not found") + } + model.ParentModelName = pointer.To("NamedLocation") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.ipNamedLocation") + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index 1b232d9e8f1..5ab9f154fe2 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -12,29 +12,24 @@ import ( var workarounds = []workaround{ workaroundApplication{}, + workaroundNamedLocation{}, + workaroundIPRange{}, } type workaround interface { - // IsApplicable determines whether this workaround is applicable for this AzureApiDefinition - IsApplicable(string, string, *parser.Model) bool - // Name returns the Service Name and associated Pull Request number Name() string // Process takes the apiDefinition and applies the Workaround to this AzureApiDefinition - Process(*parser.Model) error + Process(string, parser.Models, parser.Constants) error } -func ApplyWorkarounds(apiVersion string, models parser.Models) error { +func ApplyWorkarounds(apiVersion string, models parser.Models, constants parser.Constants) error { logging.Tracef("Processing Data Workarounds..") - for modelName, model := range models { - for _, fix := range workarounds { - if fix.IsApplicable(apiVersion, modelName, model) { - logging.Tracef("Applying Data Workaround %q to Model %q", fix.Name(), modelName) - if err := fix.Process(model); err != nil { - return fmt.Errorf("applying Data Workaround %q to Model %q: %v", fix.Name(), modelName, err) - } - } + for _, fix := range workarounds { + logging.Tracef("Applying Data Workaround %q to Model %q", fix.Name()) + if err := fix.Process(apiVersion, models, constants); err != nil { + return fmt.Errorf("applying Data Workaround %q: %v", fix.Name(), err) } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go index 214017e296d..14dcf54e6cc 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/importer.go @@ -63,13 +63,13 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha return err } - logging.Infof("Cleaning up models...") - if err = p.cleanupModels(); err != nil { + logging.Infof("Applying workarounds for invalid model definitions..") + if err = workarounds.ApplyWorkarounds(p.apiVersion, p.models, p.constants); err != nil { return err } - logging.Infof("Applying workarounds for invalid model definitions..") - if err = workarounds.ApplyWorkarounds(p.apiVersion, p.models); err != nil { + logging.Infof("Cleaning up models...") + if err = p.cleanupModels(); err != nil { return err } From 585af0c99d594814322e7dfc3c315c82804f83fe Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 01:51:00 +0100 Subject: [PATCH 050/117] importer-msgraph-metadata: manual casing fixes for `Ip` => `IP` and `Cidr` => `CIDR` --- .../components/normalize/normalize.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/normalize.go index 86d2effb0d6..c6cd1876091 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize.go @@ -87,6 +87,12 @@ func CleanName(name string) string { // Innererror should be InnerError name = regexp.MustCompile("^Innererror").ReplaceAllString(name, "InnerError") + // Ip[A-Zv] should be IP[A-Zv] + name = regexp.MustCompile("Ip([A-Zv])").ReplaceAllString(name, "IP${1}") + + // Cidr should be CIDR + name = regexp.MustCompile("Cidr").ReplaceAllString(name, "CIDR") + // Oauth should be OAuth name = regexp.MustCompile("^Oauth").ReplaceAllString(name, "OAuth") From c3a1f0268708802ff82aaa70e68477b87f91fd30 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 01:54:27 +0100 Subject: [PATCH 051/117] generator-go-sdk: export the `Unmarshal{Name}Implementation` function since it may reside in a different package to the client/methods --- .../internal/generator/templater_methods.go | 8 ++++---- .../internal/generator/templater_models.go | 10 +++++----- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index c4c8ea2d467..9e5e6f865d3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -645,9 +645,9 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin } if c.operation.FieldContainingPaginationDetails != nil { - unmarshaler := fmt.Sprintf("unmarshal%sImplementation", modelName) + unmarshaler := fmt.Sprintf("Unmarshal%sImplementation", modelName) if modelPackage != "" { - unmarshaler = fmt.Sprintf("%s.unmarshal%sImplementation", modelPackage, modelName) + unmarshaler = fmt.Sprintf("%s.Unmarshal%sImplementation", modelPackage, modelName) } output = fmt.Sprintf(` @@ -691,9 +691,9 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin // when this is a Discriminated Type (either the Parent or the Implementation, call the `unmarshal` func // for the relevant Parent if discriminatedTypeParentName != "" { - unmarshaler := fmt.Sprintf("unmarshal%sImplementation", discriminatedTypeParentName) + unmarshaler := fmt.Sprintf("Unmarshal%sImplementation", discriminatedTypeParentName) if modelPackage != "" { - unmarshaler = fmt.Sprintf("%s.unmarshal%sImplementation", modelPackage, discriminatedTypeParentName) + unmarshaler = fmt.Sprintf("%s.Unmarshal%sImplementation", modelPackage, discriminatedTypeParentName) } output = fmt.Sprintf(` diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 217785ca349..cb86f240c84 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -420,7 +420,7 @@ func (c modelsTemplater) codeForUnmarshalFunctions(data GeneratorData) (*string, } func (c modelsTemplater) codeForUnmarshalParentFunction(data GeneratorData) (*string, error) { - // if this is a Discriminated Type (e.g. Parent) then we need to generate a unmarshal{Name}Implementations + // if this is a Discriminated Type (e.g. Parent) then we need to generate an Unmarshal{Name}Implementation // function which can be used in any usages lines := make([]string, 0) if c.model.IsDiscriminatedParentType() { @@ -450,7 +450,7 @@ func (c modelsTemplater) codeForUnmarshalParentFunction(data GeneratorData) (*st // NOTE: unmarshaling null returns an empty map, which'll mean the `ok` fails // the 'type' field being omitted will also mean that `ok` is false lines = append(lines, fmt.Sprintf(` -func unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { +func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { if input == nil { return nil, nil } @@ -630,7 +630,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) output := make(map[string]%[2]s) for key, val := range dictionaryTemp { - impl, err := unmarshal%[2]sImplementation(val) + impl, err := Unmarshal%[2]sImplementation(val) if err != nil { return fmt.Errorf("unmarshaling key %%q field '%[1]s' for '%[3]s': %%+v", key, err) } @@ -665,7 +665,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) output := make([]%[2]s, 0) for i, val := range listTemp { - impl, err := unmarshal%[2]sImplementation(val) + impl, err := Unmarshal%[2]sImplementation(val) if err != nil { return fmt.Errorf("unmarshaling index %%d field '%[1]s' for '%[3]s': %%+v", i, err) } @@ -678,7 +678,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) if fieldDetails.ObjectDefinition.Type == models.ReferenceSDKObjectDefinitionType { lines = append(lines, fmt.Sprintf(` if v, ok := temp[%[4]q]; ok { - impl, err := unmarshal%[2]sImplementation(v) + impl, err := Unmarshal%[2]sImplementation(v) if err != nil { return fmt.Errorf("unmarshaling field '%[1]s' for '%[3]s': %%+v", err) } From 7f663b61e91cd638eed1547acec287869e661468 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 02:19:43 +0100 Subject: [PATCH 052/117] importer-msgraph-metadata: fix bug that generated constant fields without first checking for array --- .../components/parser/types.go | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index dda87e52d74..17cedb1f0ca 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -171,14 +171,6 @@ type ModelField struct { } func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObjectDefinition, error) { - if f.ConstantName != nil { - return &sdkModels.SDKObjectDefinition{ - Nullable: f.Nullable, - ReferenceName: f.ConstantName, - Type: sdkModels.ReferenceSDKObjectDefinitionType, - }, nil - } - if f.Type == nil { return nil, fmt.Errorf("field %q has no Type", f.Title) } @@ -251,6 +243,14 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj return nil, nil } + if f.ConstantName != nil { + return &sdkModels.SDKObjectDefinition{ + Nullable: f.Nullable, + ReferenceName: f.ConstantName, + Type: sdkModels.ReferenceSDKObjectDefinitionType, + }, nil + } + return &sdkModels.SDKObjectDefinition{ Nullable: f.Nullable, Type: f.Type.DataApiSdkObjectDefinitionType(), From 3d03964d653b4c3e338db3e18dacaf57c1369408 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 14 Aug 2024 02:27:00 +0100 Subject: [PATCH 053/117] importer-msgraph-metadata: singularize constant names --- tools/importer-msgraph-metadata/components/parser/types.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 17cedb1f0ca..4aa3ce8d79f 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -766,7 +766,7 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta if ((field.Type != nil && *field.Type == DataTypeString) || (field.ItemType != nil && *field.ItemType == DataTypeString)) && len(enum) > 0 { // Despite being "fully qualified", type names are not unique in MS Graph, so we prefix them with the field name to provide some namespacing. // This leads to some excessively long constant names, it is what it is. - field.ConstantName = pointer.To(name + field.Title) + field.ConstantName = pointer.To(normalize.Singularize(name + field.Title)) // Add the enumeration as a constant constants[*field.ConstantName] = &Constant{ From d66faaca977ab42c9421db82b5a96c4b5553e775 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 00:21:29 +0100 Subject: [PATCH 054/117] generator-go-sdk: support Required + Nullable model fields, and generating models with field descriptions as comments --- .../internal/transforms/sdk_field.go | 23 ++++--- .../generator-go-sdk/internal/cmd/generate.go | 3 +- .../internal/featureflags/flags.go | 4 +- .../internal/generator/data.go | 69 ++++++++++--------- .../internal/generator/settings.go | 1 + .../internal/generator/templater_models.go | 23 ++++++- 6 files changed, 77 insertions(+), 46 deletions(-) diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_field.go b/tools/data-api-repository/repository/internal/transforms/sdk_field.go index 787b79fe5c5..8365a6980c3 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_field.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_field.go @@ -49,19 +49,22 @@ func mapSDKFieldToRepository(fieldName string, input sdkModels.SDKField, isTypeH return nil, fmt.Errorf("mapping the ObjectDefinition for field %q: %+v", fieldName, err) } + var description *string + if input.Description != "" { + description = pointer.To(input.Description) + } + output := repositoryModels.ModelField{ ContainsDiscriminatedTypeValue: isTypeHint, DateFormat: nil, - Description: nil, - // TODO this can be uncommented when #3325 has been fixed - // Description: input.Description, - JsonName: input.JsonName, - Name: fieldName, - ObjectDefinition: *objectDefinition, - Optional: input.Optional, - ReadOnly: input.ReadOnly, - Required: input.Required, - Sensitive: input.Sensitive, + Description: description, + JsonName: input.JsonName, + Name: fieldName, + ObjectDefinition: *objectDefinition, + Optional: input.Optional, + ReadOnly: input.ReadOnly, + Required: input.Required, + Sensitive: input.Sensitive, } if input.DateFormat != nil { dateFormat, err := mapSDKDateFormatToRepository(*input.DateFormat) diff --git a/tools/generator-go-sdk/internal/cmd/generate.go b/tools/generator-go-sdk/internal/cmd/generate.go index 8c46041d351..3fa2de78bec 100644 --- a/tools/generator-go-sdk/internal/cmd/generate.go +++ b/tools/generator-go-sdk/internal/cmd/generate.go @@ -59,6 +59,7 @@ func (g GenerateCommand) Run(args []string) int { } if g.sourceDataType == models.MicrosoftGraphSourceDataType { + input.settings.GenerateDescriptionsForModels = true input.settings.CanonicalApiVersions = map[string]string{ "stable": "v1.0", } @@ -81,7 +82,7 @@ func (g GenerateCommand) Run(args []string) int { // is sent in the Request to update or remove a Key Vault Access Policy - and using other casings mean the update // or removal fails - which is tracked in https://github.com/hashicorp/pandora/issues/3229. // - // After testing it appears that `2023-07-01` doesn't suffer from this problem - as such we're going to leave + // After testing, it appears that `2023-07-01` doesn't suffer from this problem - as such we're going to leave // `2023-02-01` on the older base layer and use the newer API Version as a divide to give us a clear migration path. "KeyVault@2023-02-01", ) diff --git a/tools/generator-go-sdk/internal/featureflags/flags.go b/tools/generator-go-sdk/internal/featureflags/flags.go index ca2ffdb1e23..80035b2d71a 100644 --- a/tools/generator-go-sdk/internal/featureflags/flags.go +++ b/tools/generator-go-sdk/internal/featureflags/flags.go @@ -17,8 +17,8 @@ const OptionalDiscriminatorsShouldBeOutputWithoutOmitEmpty = false // functions should be generated. const GenerateCaseInsensitiveFunctions = true -// SkipDiscriminatedParentTypes returns whether or not the feature for skipping generation of -// discriminated parent types is enabled. +// SkipDiscriminatedParentTypes returns whether the feature is enabled for skipping generation of +// discriminated parent types. func SkipDiscriminatedParentTypes() bool { value := os.Getenv("PANDORA_SKIP_DISCRIMINATED_PARENT_TYPE") if value == "" { diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index 789362bb6d0..ddf14847d47 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -74,6 +74,9 @@ type GeneratorData struct { // the package name for the API version versionPackageName string + // whether descriptions should be generated for model fields etc. + generateDescriptionsForModels bool + // whether this is a data plane SDK (omits certain Resource Manager specific features, currently used in ID parsers) isDataPlane bool @@ -104,24 +107,25 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { } return GeneratorData{ - apiVersion: i.VersionName, - canonicalApiVersion: settings.CanonicalApiVersion(i.VersionName), - baseClientPackage: baseClientPackageForSdk(i.Type), - commonTypes: i.CommonTypes, - commonTypesIncludePath: commonTypesIncludePath, - commonTypesPackageName: commonTypesPackageName, - constants: i.ResourceDetails.Constants, - isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), - models: i.ResourceDetails.Models, - operations: i.ResourceDetails.Operations, - packageName: resourcePackageName, - resourceIds: i.ResourceDetails.ResourceIDs, - resourceOutputPath: resourceOutputPath, - serviceClientName: fmt.Sprintf("%sClient", strings.Title(i.ResourceName)), - servicePackageName: strings.ToLower(i.ServiceName), - source: i.Source, - sourceType: i.Type, - useNewBaseLayer: useNewBaseLayer, + apiVersion: i.VersionName, + canonicalApiVersion: settings.CanonicalApiVersion(i.VersionName), + baseClientPackage: baseClientPackageForSdk(i.Type), + commonTypes: i.CommonTypes, + commonTypesIncludePath: commonTypesIncludePath, + commonTypesPackageName: commonTypesPackageName, + constants: i.ResourceDetails.Constants, + generateDescriptionsForModels: settings.GenerateDescriptionsForModels, + isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), + models: i.ResourceDetails.Models, + operations: i.ResourceDetails.Operations, + packageName: resourcePackageName, + resourceIds: i.ResourceDetails.ResourceIDs, + resourceOutputPath: resourceOutputPath, + serviceClientName: fmt.Sprintf("%sClient", strings.Title(i.ResourceName)), + servicePackageName: strings.ToLower(i.ServiceName), + source: i.Source, + sourceType: i.Type, + useNewBaseLayer: useNewBaseLayer, } } @@ -156,20 +160,21 @@ func (i VersionGeneratorInput) generatorData(settings Settings) VersionGenerator return VersionGeneratorData{ GeneratorData: GeneratorData{ - apiVersion: i.VersionName, - canonicalApiVersion: settings.CanonicalApiVersion(i.VersionName), - baseClientPackage: baseClientPackageForSdk(i.Type), - commonTypes: i.CommonTypes, - constants: i.CommonTypes.Constants, - isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), - models: i.CommonTypes.Models, - packageName: versionPackageName, - servicePackageName: strings.ToLower(i.ServiceName), - source: i.Source, - sourceType: i.Type, - useNewBaseLayer: useNewBaseLayer, - versionDirectoryName: versionDirectoryName, - versionPackageName: versionPackageName, + apiVersion: i.VersionName, + canonicalApiVersion: settings.CanonicalApiVersion(i.VersionName), + baseClientPackage: baseClientPackageForSdk(i.Type), + commonTypes: i.CommonTypes, + constants: i.CommonTypes.Constants, + generateDescriptionsForModels: settings.GenerateDescriptionsForModels, + isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), + models: i.CommonTypes.Models, + packageName: versionPackageName, + servicePackageName: strings.ToLower(i.ServiceName), + source: i.Source, + sourceType: i.Type, + useNewBaseLayer: useNewBaseLayer, + versionDirectoryName: versionDirectoryName, + versionPackageName: versionPackageName, }, commonTypesOutputPath: commonTypesOutputPath, resources: i.Resources, diff --git a/tools/generator-go-sdk/internal/generator/settings.go b/tools/generator-go-sdk/internal/generator/settings.go index ee4d720e606..86fefd6bde4 100644 --- a/tools/generator-go-sdk/internal/generator/settings.go +++ b/tools/generator-go-sdk/internal/generator/settings.go @@ -12,6 +12,7 @@ import ( type Settings struct { CanonicalApiVersions map[string]string CommonTypesPackageName string + GenerateDescriptionsForModels bool VersionsToGenerateCommonTypes map[string]models.SourceDataOrigin servicesUsingOldBaseLayer map[string]struct{} } diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index cb86f240c84..e81b3537cba 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -167,12 +167,25 @@ type Raw%[1]sImpl struct { } } + formattedStructLines := make([]string, 0) + for i, v := range structLines { + if strings.Contains(v, "//") { + if i > 0 && !strings.HasSuffix(formattedStructLines[i-1], "\n") { + v = "\n" + v + } + if i < len(structLines)-1 { + v += "\n" + } + } + formattedStructLines = append(formattedStructLines, v) + } + out := fmt.Sprintf(` %[3]s type %[1]s struct { %[2]s } -`, c.name, strings.Join(structLines, "\n"), parentAssignmentInfo) +`, c.name, strings.Join(formattedStructLines, "\n"), parentAssignmentInfo) return &out, nil } @@ -229,9 +242,17 @@ func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDe fieldType = fmt.Sprintf("*%s", fieldType) } jsonDetails += ",omitempty" + } else if fieldDetails.ObjectDefinition.Nullable && !strings.HasPrefix(fieldType, "nullable.") { + fieldType = fmt.Sprintf("*%s", fieldType) } line := fmt.Sprintf("\t%s %s `json:\"%s\"`", fieldName, fieldType, jsonDetails) + + if data.generateDescriptionsForModels && fieldDetails.Description != "" { + description := fmt.Sprintf("// %s", fieldDetails.Description) + line = fmt.Sprintf("%s\n%s", description, line) + } + return &line, nil } From 7b608bd6399eb1c297d5699d909c66c6b9ef038a Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 00:22:54 +0100 Subject: [PATCH 055/117] importer-msgraph-metadata: improve name deuplication helper function, add tests --- .../components/normalize/duplicates.go | 77 +++++++++++++------ .../components/normalize/duplicates_test.go | 37 +++++++++ 2 files changed, 89 insertions(+), 25 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/normalize/duplicates_test.go diff --git a/tools/importer-msgraph-metadata/components/normalize/duplicates.go b/tools/importer-msgraph-metadata/components/normalize/duplicates.go index a51e429affa..bdf216f7e3d 100644 --- a/tools/importer-msgraph-metadata/components/normalize/duplicates.go +++ b/tools/importer-msgraph-metadata/components/normalize/duplicates.go @@ -8,38 +8,65 @@ import ( "strings" ) -func DeDuplicate(name string) string { +// DeDuplicateName reduces duplication in long form constant/model names. Since we typically prefix these to provide +// some namespacing, this can lead to excessive duplication which we try to cut down on here. +// Rules: +// 1. Immediately duplicated words should be removed, e.g. "GroupGroupMember" +// 2. Chains of successive words that match should be removed, e.g. "GroupMemberGroupMemberRef" +// 3. Single duplicate words that are not adjacent should not be removed, e.g. "GroupMemberGroupOwner" +// 4. Duplicate chains of words that reach the end of the string should not be removed, e.g. "SynchronizationSecretKeySynchronizationSecret" +// 5. Words should be compared in their singular form, but retained in their existing form, whether singular or plural +func DeDuplicateName(name string) string { nameSpaced := regexp.MustCompile("([A-Z])").ReplaceAllString(name, " $1") nameParts := strings.Split(strings.TrimSpace(nameSpaced), " ") + + singularParts := make([]string, len(nameParts)) + for i, s := range nameParts { + singularParts[i] = Singularize(s) + } + newParts := make([]string, 0) + var buffer []string + var matchStart, offset int + for i := 0; i < len(nameParts); i++ { - if i > 0 && strings.EqualFold(nameParts[i], nameParts[i-1]) { - continue + if i > 0 { + // preceding word is identical, omit this word + if strings.EqualFold(singularParts[i], singularParts[i-1]) { + offset++ + continue + } + + if matchStart < 0 { + // look for a matching word to begin a chain + for matchStart = offset; matchStart < i; matchStart++ { + if strings.EqualFold(singularParts[i], singularParts[matchStart]) { + buffer = append(buffer, nameParts[i]) + break + } + } + if len(buffer) > 0 { + continue + } + } else if matchStart+len(buffer) < i && strings.EqualFold(singularParts[i], singularParts[matchStart+len(buffer)]) { + // continue matching if a chain was started + buffer = append(buffer, nameParts[i]) + continue + } else if len(buffer) == 1 { + // if a chain was only a single word, that doesn't count + newParts = append(newParts, buffer[0]) + } } + + buffer = make([]string, 0) + matchStart = -1 newParts = append(newParts, nameParts[i]) } - return strings.Join(newParts, "") -} -func DeDuplicateName(name string) string { - words := make([]string, 0) - i := 0 - for j, c := range name { - char := string(c) - if j > 0 && strings.ToUpper(char) == char { - i++ - } - if len(words) <= i { - words = append(words, "") - } - words[i] = words[i] + char + // retain the final segment if it was specified and trimmed + if len(buffer) > 0 { + newParts = append(newParts, buffer...) } - out := "" - for k, word := range words { - if k > 0 && strings.EqualFold(words[k-1], word) { - continue - } - out = out + word - } - return out + + return strings.Join(newParts, "") } diff --git a/tools/importer-msgraph-metadata/components/normalize/duplicates_test.go b/tools/importer-msgraph-metadata/components/normalize/duplicates_test.go new file mode 100644 index 00000000000..50731b7eb7f --- /dev/null +++ b/tools/importer-msgraph-metadata/components/normalize/duplicates_test.go @@ -0,0 +1,37 @@ +package normalize + +import ( + "testing" +) + +func TestDeDuplicate(t *testing.T) { + testCases := map[string]string{ + // simple duplicate preceding word + "ApplicationApplicationOwner": "ApplicationOwner", + + // duplicate words from beginning of string + "ConditionalAccessConditionalAccessDevicePlatform": "ConditionalAccessDevicePlatform", + + // duplicate preceding word + duplicate words from beginning + "GroupGroupMemberGroupMemberRef": "GroupMemberRef", + + // single duplicate word from beginning should not be removed + "ServicePrincipalServiceProfile": "ServicePrincipalServiceProfile", + + // non-duplicate - not preceding and not from beginning + "UserAccountPasswordAccountPolicy": "UserAccountPasswordAccountPolicy", + + // duplicate from beginning, but at end of name, so must be retained + "SynchronizationSecretKeyStringValuePairSynchronizationSecret": "SynchronizationSecretKeyStringValuePairSynchronizationSecret", + + // duplicate chain after singularization + "ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType": "ConditionalAccessGuestsOrExternalUsersType", + } + + for input, expected := range testCases { + result := DeDuplicateName(input) + if result != expected { + t.Errorf("DeDuplicateName(%q): got %q, expected %q", input, result, expected) + } + } +} From 77de1bea8d89793dd7dfad608b9eea72d6bd8bc6 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 00:23:28 +0100 Subject: [PATCH 056/117] importer-msgraph-metadata: use `github.com/gertd/go-pluralize` for pluralization/singularization, add tests --- .../components/normalize/normalize.go | 77 ++++++++----------- .../components/normalize/normalize_test.go | 73 ++++++++++++++++++ tools/importer-msgraph-metadata/go.mod | 1 + tools/importer-msgraph-metadata/go.sum | 2 + 4 files changed, 109 insertions(+), 44 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/normalize/normalize_test.go diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/normalize.go index c6cd1876091..38936e74094 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize.go @@ -8,62 +8,35 @@ import ( "regexp" "strings" + "github.com/gertd/go-pluralize" "golang.org/x/text/cases" "golang.org/x/text/language" ) func Singularize(name string) string { - // "access" is already singular - if len(name) >= 5 && name[len(name)-5:] == "ccess" { - return name - } - - // "properties", "entities" etc - if len(name) >= 3 && name[len(name)-3:] == "ies" { - return fmt.Sprintf("%sy", name[:len(name)-3]) - } - - // "premises" etc - if len(name) >= 3 && name[len(name)-3:] == "ses" { - return name[:len(name)-2] + for _, v := range pluralExceptions { + if name == v.plural { + return v.singular + } } - // all other words, remove the trailing "s" - if len(name) >= 1 && name[len(name)-1:] == "s" { - return name[:len(name)-1] - } + client := pluralize.NewClient() + output := client.Singular(name) - // else just return the original name - return name + return output } func Pluralize(name string) string { - if name == "" { - return "" - } - ret := fmt.Sprintf("%ss", name) - if strings.EqualFold(name, "me") { - return name - } - if len(name) == 0 { - return ret - } - if strings.EqualFold(name[len(name)-2:], "ay") || strings.EqualFold(name[len(name)-2:], "ey") { - return fmt.Sprintf("%ss", name) - } - if strings.EqualFold(name[len(name)-1:], "y") { - return fmt.Sprintf("%sies", name[:len(name)-1]) - } - if strings.EqualFold(name[len(name)-1:], "s") { - return name - } - if len(name) < 2 { - return ret - } - if strings.EqualFold(name[len(name)-2:], "Of") { - return name + for _, v := range pluralExceptions { + if name == v.singular { + return v.plural + } } - return ret + + client := pluralize.NewClient() + output := client.Plural(name) + + return output } func CleanName(name string) string { @@ -149,3 +122,19 @@ var Verbs = operationVerbs{ "Update", "Validate", } + +type plural struct { + singular string + plural string +} + +var pluralExceptions = []plural{ + {"By", "By"}, + {"Compatibility", "Compatibility"}, + {"Cache", "Caches"}, + {"Data", "Data"}, + {"Metadata", "Metadata"}, + {"Orderby", "Orderby"}, + {"Premise", "Premises"}, + {"Sortby", "Sortby"}, +} diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize_test.go b/tools/importer-msgraph-metadata/components/normalize/normalize_test.go new file mode 100644 index 00000000000..388f7d4f75c --- /dev/null +++ b/tools/importer-msgraph-metadata/components/normalize/normalize_test.go @@ -0,0 +1,73 @@ +package normalize + +import "testing" + +func TestSingularize(t *testing.T) { + testCases := map[string]string{ + "Access": "Access", + "Apps": "App", + "Applications": "Application", + "By": "By", + "Caches": "Cache", + "Data": "Data", + "Details": "Detail", + "Compatibility": "Compatibility", + "Entities": "Entity", + "Kinds": "Kind", + "Metadata": "Metadata", + "Modalities": "Modality", + "Options": "Option", + "Orderby": "Orderby", + "Policies": "Policy", + "Premises": "Premise", + "Principals": "Principal", + "Properties": "Property", + "Services": "Service", + "Severities": "Severity", + "Sortby": "Sortby", + "Success": "Success", + "Successes": "Success", + } + + for input, expected := range testCases { + result := Singularize(input) + if result != expected { + t.Errorf("Singularize(%q): got %q, expected %q", input, result, expected) + } + } +} + +func TestPluralize(t *testing.T) { + testCases := map[string]string{ + "Access": "Accesses", + "App": "Apps", + "Application": "Applications", + "By": "By", + "Compatibility": "Compatibility", + "Cache": "Caches", + "Data": "Data", + "Detail": "Details", + "Entity": "Entities", + "Kind": "Kinds", + "Metadata": "Metadata", + "Modality": "Modalities", + "Option": "Options", + "Orderby": "Orderby", + "Policy": "Policies", + "Premise": "Premises", + "Principal": "Principals", + "Property": "Properties", + "Service": "Services", + "Severity": "Severities", + "Sortby": "Sortby", + "Success": "Successes", + "Successes": "Successes", + } + + for input, expected := range testCases { + result := Pluralize(input) + if result != expected { + t.Errorf("Pluralize(%q): got %q, expected %q", input, result, expected) + } + } +} diff --git a/tools/importer-msgraph-metadata/go.mod b/tools/importer-msgraph-metadata/go.mod index 8fea7689338..4d0ee4fbbcb 100644 --- a/tools/importer-msgraph-metadata/go.mod +++ b/tools/importer-msgraph-metadata/go.mod @@ -32,6 +32,7 @@ require ( github.com/cyphar/filepath-securejoin v0.2.4 // indirect github.com/emirpasic/gods v1.18.1 // indirect github.com/fatih/color v1.13.0 // indirect + github.com/gertd/go-pluralize v0.2.1 // indirect github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376 // indirect github.com/go-git/go-billy/v5 v5.5.0 // indirect github.com/go-openapi/jsonpointer v0.19.5 // indirect diff --git a/tools/importer-msgraph-metadata/go.sum b/tools/importer-msgraph-metadata/go.sum index cdb318e800c..6786b54ad84 100644 --- a/tools/importer-msgraph-metadata/go.sum +++ b/tools/importer-msgraph-metadata/go.sum @@ -40,6 +40,8 @@ github.com/emirpasic/gods v1.18.1/go.mod h1:8tpGGwCnJ5H4r6BWwaV6OrWmMoPhUl5jm/FM github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w= github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= +github.com/gertd/go-pluralize v0.2.1 h1:M3uASbVjMnTsPb0PNqg+E/24Vwigyo/tvyMTtAlLgiA= +github.com/gertd/go-pluralize v0.2.1/go.mod h1:rbYaKDbsXxmRfr8uygAEKhOWsjyrrqrkHVpZvoOp8zk= github.com/getkin/kin-openapi v0.117.0 h1:QT2DyGujAL09F4NrKDHJGsUoIprlIcFVHWDVDcUFE8A= github.com/getkin/kin-openapi v0.117.0/go.mod h1:l5e9PaFUo9fyLJCPGQeXI2ML8c3P8BHOEV2VaAVf/pc= github.com/gliderlabs/ssh v0.3.5 h1:OcaySEmAQJgyYcArR+gGGTHCyE7nvhEMTlYY+Dp8CpY= From b56cb174ed5c142cdc7196632555fca57c2e2ce5 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 00:26:05 +0100 Subject: [PATCH 057/117] importer-msgraph-metadata: improve constant naming --- .../components/parser/types.go | 187 ++++++++++-------- 1 file changed, 106 insertions(+), 81 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 4aa3ce8d79f..76d57db3e30 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -126,6 +126,13 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { continue } + optional := true + required := false + if field.Required { + optional = false + required = true + } + sdkFields[fieldName] = sdkModels.SDKField{ ContainsDiscriminatedValue: field.DiscriminatedValue, DateFormat: nil, @@ -133,11 +140,9 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { JsonName: field.JsonField, ObjectDefinition: *objectDefinition, - ReadOnly: field.ReadOnly, - - // TODO work these out - Optional: true, - Required: false, + Optional: optional, + ReadOnly: field.ReadOnly, + Required: required, Sensitive: false, } } @@ -159,6 +164,7 @@ type ModelField struct { Type *DataType Description string Default interface{} + Required bool ReadOnly bool WriteOnly bool Nullable bool @@ -265,6 +271,7 @@ const ( DataTypeBase64 DataTypeBinary DataTypeBool + DataTypeCsv DataTypeDate DataTypeDateTime DataTypeDuration @@ -295,6 +302,8 @@ func (ft DataType) DataApiSdkObjectDefinitionType() sdkModels.SDKObjectDefinitio return sdkModels.FloatSDKObjectDefinitionType case DataTypeBool: return sdkModels.BooleanSDKObjectDefinitionType + case DataTypeCsv: + return sdkModels.CSVSDKObjectDefinitionType case DataTypeDate, DataTypeDateTime, DataTypeTime: return sdkModels.DateTimeSDKObjectDefinitionType case DataTypeBinary: @@ -316,6 +325,8 @@ func (ft DataType) DataApiSdkOperationOptionObjectDefinitionType() sdkModels.SDK return sdkModels.FloatSDKOperationOptionObjectDefinitionType case DataTypeBool: return sdkModels.BooleanSDKOperationOptionObjectDefinitionType + case DataTypeCsv: + return sdkModels.CSVSDKOperationOptionObjectDefinitionType case DataTypeDate, DataTypeDateTime, DataTypeTime: return sdkModels.StringSDKOperationOptionObjectDefinitionType case DataTypeBinary: @@ -701,108 +712,122 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta for jsonField, schemaRef := range input.Schemas { // maintainer note: this is a good place to add a breakpoint for inspecting model fields and constants as they are processed // example breakpoint condition: name=="AdministrativeUnit" - if schema := schemaRef.Value; schema != nil { - field := ModelField{ - Title: normalize.CleanName(jsonField), - Description: schema.Description, - Default: schema.Default, - ReadOnly: schemaRef.Value.ReadOnly, - WriteOnly: schemaRef.Value.WriteOnly, - Nullable: schemaRef.Value.Nullable, - AllowEmptyValue: schemaRef.Value.AllowEmptyValue, - JsonField: jsonField, - } + schema := schemaRef.Value + if schema == nil { + continue + } - if field.Title == "" { - continue + field := ModelField{ + Title: normalize.CleanName(jsonField), + Description: schema.Description, + Default: schema.Default, + ReadOnly: schemaRef.Value.ReadOnly, + WriteOnly: schemaRef.Value.WriteOnly, + Nullable: schemaRef.Value.Nullable, + AllowEmptyValue: schemaRef.Value.AllowEmptyValue, + JsonField: jsonField, + } + + if field.Title == "" { + continue + } + + result, _ := FlattenSchemaRef(schemaRef, nil) + + // Determine any enumeration for the field + enum := parseEnum(schema.Enum) + if result != nil && len(result.Enum) > 0 && len(enum) == 0 { + enum = parseEnum(result.Enum) + } + + // Find the corresponding model when the field refers to it, and set the model name for the field + if result != nil && result.Title != "" && result.Schemas != nil { + if _, ok := models[result.Title]; !ok { + models, constants = Schemas(*result, result.Title, models, constants, common) } + field.ModelName = &result.Title + } - result, _ := FlattenSchemaRef(schemaRef, nil) + // Handle items for collections + if schema.Items != nil && schema.Items.Value != nil { + itemsResult, _ := FlattenSchemaRef(schema.Items, nil) - // Determine any enumeration for the field - enum := parseEnum(schema.Enum) - if result != nil && len(result.Enum) > 0 && len(enum) == 0 { - enum = parseEnum(result.Enum) + // Determine any type for the items + if schema.Items.Value.Type != "" || schema.Items.Value.Format != "" { + field.ItemType = FieldType(schema.Items.Value.Type, schema.Items.Value.Format, field.ModelName != nil) } - // Find the corresponding model when the field refers to it, and set the model name for the field - if result != nil && result.Title != "" && result.Schemas != nil { + // Find the corresponding model when the field (items) refers to it, and set the model name for the field items + if itemsResult != nil && itemsResult.Title != "" && itemsResult.Type == "object" { if _, ok := models[result.Title]; !ok { models, constants = Schemas(*result, result.Title, models, constants, common) } field.ModelName = &result.Title } + } - // Handle items for collections - if schema.Items != nil && schema.Items.Value != nil { - itemsResult, _ := FlattenSchemaRef(schema.Items, nil) - - // Determine any type for the items - if schema.Items.Value.Type != "" || schema.Items.Value.Format != "" { - field.ItemType = FieldType(schema.Items.Value.Type, schema.Items.Value.Format, field.ModelName != nil) - } + // Match the field type to the referenced object, or set a basic type + if result != nil && schema.Type == "" && schema.Format == "" && (result.Type != "" || result.Format != "") { + field.Type = FieldType(result.Type, result.Format, field.ModelName != nil) + } else { + field.Type = FieldType(schema.Type, schema.Format, field.ModelName != nil) + } - // Find the corresponding model when the field (items) refers to it, and set the model name for the field items - if itemsResult != nil && itemsResult.Title != "" && itemsResult.Type == "object" { - if _, ok := models[result.Title]; !ok { - models, constants = Schemas(*result, result.Title, models, constants, common) - } - field.ModelName = &result.Title - } - } + // Set the field type for enums; typically strings for constants, but could be any valid type + if result != nil && field.Type != nil && *field.Type == DataTypeArray && len(enum) > 0 && (result.Type != "" || result.Format != "") { + field.ItemType = FieldType(result.Type, result.Format, field.ModelName != nil) + } - // Match the field type to the referenced object, or set a basic type - if result != nil && schema.Type == "" && schema.Format == "" && (result.Type != "" || result.Format != "") { - field.Type = FieldType(result.Type, result.Format, field.ModelName != nil) + if ((field.Type != nil && *field.Type == DataTypeString) || (field.ItemType != nil && *field.ItemType == DataTypeString)) && len(enum) > 0 { + // Despite being "fully qualified", type names are not unique in MS Graph, so we prefix them with the model name to provide some namespacing. + // Though we attempt to de-duplicate, this does lead to some excessively long constant names, it is what it is. + constantName := name + if result.Title != "" { + // use provided ref name if present + constantName += result.Title } else { - field.Type = FieldType(schema.Type, schema.Format, field.ModelName != nil) + // otherwise use the field name + constantName += field.Title } + constantName = normalize.DeDuplicateName(constantName) + constantName = normalize.Singularize(constantName) - // Set the field type for enums; typically strings for constants, but could be any valid type - if result != nil && field.Type != nil && *field.Type == DataTypeArray && len(enum) > 0 && (result.Type != "" || result.Format != "") { - field.ItemType = FieldType(result.Type, result.Format, field.ModelName != nil) - } - - if ((field.Type != nil && *field.Type == DataTypeString) || (field.ItemType != nil && *field.ItemType == DataTypeString)) && len(enum) > 0 { - // Despite being "fully qualified", type names are not unique in MS Graph, so we prefix them with the field name to provide some namespacing. - // This leads to some excessively long constant names, it is what it is. - field.ConstantName = pointer.To(normalize.Singularize(name + field.Title)) - - // Add the enumeration as a constant - constants[*field.ConstantName] = &Constant{ - Enum: enum, - Type: field.Type, - } - } + field.ConstantName = pointer.To(constantName) - if field.Type == nil { - logging.Warnf("Skipping field %q in model %q because Type is nil", name, field.Title) - continue + // Add the enumeration as a constant + constants[constantName] = &Constant{ + Enum: enum, + Type: field.Type, } + } - // Insert an "@odata.bind" field where a field or collection refers to a DirectoryObject. The MS Graph - // OpenAPI spec unfortunately does not document relationships between entities. - if field.ModelName != nil && strings.EqualFold(*field.ModelName, "DirectoryObject") { - bindFieldName := fmt.Sprintf("%s_ODataBind", normalize.CleanName(jsonField)) + if field.Type == nil { + logging.Warnf("Skipping field %q in model %q because Type is nil", name, field.Title) + continue + } - var fieldType, itemType *DataType + // Insert an "@odata.bind" field where a field or collection refers to a DirectoryObject. The MS Graph + // OpenAPI spec unfortunately does not document relationships between entities. + if field.ModelName != nil && strings.EqualFold(*field.ModelName, "DirectoryObject") { + bindFieldName := fmt.Sprintf("%s_ODataBind", normalize.CleanName(jsonField)) - fieldType = pointer.To(DataTypeString) - if *field.Type == DataTypeArray { - fieldType = pointer.To(DataTypeArray) - itemType = pointer.To(DataTypeString) - } + var fieldType, itemType *DataType - model.Fields[bindFieldName] = &ModelField{ - Title: bindFieldName, - Type: fieldType, - ItemType: itemType, - JsonField: fmt.Sprintf("%s@odata.bind", jsonField), - } + fieldType = pointer.To(DataTypeString) + if *field.Type == DataTypeArray { + fieldType = pointer.To(DataTypeArray) + itemType = pointer.To(DataTypeString) } - model.Fields[normalize.CleanName(jsonField)] = &field + model.Fields[bindFieldName] = &ModelField{ + Title: bindFieldName, + Type: fieldType, + ItemType: itemType, + JsonField: fmt.Sprintf("%s@odata.bind", jsonField), + } } + + model.Fields[normalize.CleanName(jsonField)] = &field } return models, constants From e197f77909753c264dc9f0d594298af8deae37ba Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 00:26:20 +0100 Subject: [PATCH 058/117] importer-msgraph-metadata: workarounds for conditional access --- .../workarounds/workaround_application.go | 2 - .../workaround_conditionalaccesspolicy.go | 102 ++++++++++++++++++ .../components/workarounds/workarounds.go | 1 + 3 files changed, 103 insertions(+), 2 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go index d27aa820d9b..51933488f50 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go @@ -45,8 +45,6 @@ func (workaroundApplication) Process(apiVersion string, models parser.Models, co // Add the `applicationTemplateId` field if missing if _, ok := model.Fields["ApplicationTemplateId"]; !ok { - fmt.Println("fo") - model.Fields["ApplicationTemplateId"] = &parser.ModelField{ Title: "ApplicationTemplateId", Description: "Unique identifier of the applicationTemplate. Supports $filter (eq, not, ne). Read-only. null if the app wasn't created from an application template.", diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go new file mode 100644 index 00000000000..6a0bf477637 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -0,0 +1,102 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ workaround = workaroundConditionalAccessPolicy{} + +// workaroundConditionalAccessPolicy adds missing fields and fixes some field types. +type workaroundConditionalAccessPolicy struct{} + +func (workaroundConditionalAccessPolicy) Name() string { + return "Conditional Access Policy / fixing missing fields and types" +} + +func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parser.Models, constants parser.Constants) error { + model, ok := models["ConditionalAccessPolicy"] + if !ok { + return fmt.Errorf("`ConditionalAccessPolicy` model not found") + } + + // grantControls and sessionControls must be null to unset them, so make them nullable + required + if _, ok = model.Fields["GrantControls"]; !ok { + return fmt.Errorf("`GrantControls` field not found") + } + model.Fields["GrantControls"].Nullable = true + model.Fields["GrantControls"].Required = true + if _, ok = model.Fields["SessionControls"]; !ok { + return fmt.Errorf("`SessionControls` field not found") + } + model.Fields["SessionControls"].Nullable = true + model.Fields["SessionControls"].Required = true + + model, ok = models["ConditionalAccessConditionSet"] + if !ok { + return fmt.Errorf("`ConditionalAccessConditionSet` model not found") + } + + // devices, locations, platforms must each be null to unset them, so make them nullable + required + if _, ok = model.Fields["Devices"]; !ok { + return fmt.Errorf("`Devices` field not found") + } + model.Fields["Devices"].Nullable = true + model.Fields["Devices"].Required = true + if _, ok = model.Fields["Locations"]; !ok { + return fmt.Errorf("`Locations` field not found") + } + model.Fields["Locations"].Nullable = true + model.Fields["Locations"].Required = true + if _, ok = model.Fields["Platforms"]; !ok { + return fmt.Errorf("`Platforms` field not found") + } + model.Fields["Platforms"].Nullable = true + model.Fields["Platforms"].Required = true + + model, ok = models["ConditionalAccessExternalTenants"] + if !ok { + return fmt.Errorf("`ConditionalAccessExternalTenants` model not found") + } + + // Add the `members` field if missing + if _, ok = model.Fields["Members"]; !ok { + model.Fields["Members"] = &parser.ModelField{ + Title: "Members", + Type: pointer.To(parser.DataTypeArray), + ItemType: pointer.To(parser.DataTypeString), + JsonField: "members", + } + } + + // Set CSV type for field + model, ok = models["ConditionalAccessGuestsOrExternalUsers"] + if !ok { + return fmt.Errorf("`ConditionalAccessGuestsOrExternalUsers` model not found") + } + if _, ok = model.Fields["GuestOrExternalUserTypes"]; !ok { + return fmt.Errorf("`GuestOrExternalUserTypes` field not found") + } + model.Fields["GuestOrExternalUserTypes"].Type = pointer.To(parser.DataTypeCsv) + + // Rename this constant + if v, ok := constants["ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType"]; ok { + constants["ConditionalAccessGuestOrExternalUserType"] = v + + for _, model = range models { + for fieldName := range model.Fields { + if model.Fields[fieldName].ConstantName != nil && *model.Fields[fieldName].ConstantName == "ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType" { + model.Fields[fieldName].ConstantName = pointer.To("ConditionalAccessGuestOrExternalUserType") + } + } + } + + delete(constants, "ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType") + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index 5ab9f154fe2..6de4b99f24d 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -12,6 +12,7 @@ import ( var workarounds = []workaround{ workaroundApplication{}, + workaroundConditionalAccessPolicy{}, workaroundNamedLocation{}, workaroundIPRange{}, } From 15ddd29f630cda0b981d9e58d1855bf15116f031 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 01:18:21 +0100 Subject: [PATCH 059/117] importer-msgraph-metadata: more workaround for conditional access --- .../workaround_conditionalaccesspolicy.go | 29 +++++++++++++++++++ 1 file changed, 29 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index 6a0bf477637..c6fdb6132d1 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -98,5 +98,34 @@ func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parse delete(constants, "ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType") } + model, ok = models["ConditionalAccessSessionControls"] + if !ok { + return fmt.Errorf("`ConditionalAccessSessionControls` model not found") + } + + // cloudAppSecurityPolicy must be null to unset it, so make it nullable + required + if _, ok = model.Fields["CloudAppSecurity"]; !ok { + return fmt.Errorf("`CloudAppSecurity` field not found") + } + model.Fields["CloudAppSecurity"].Nullable = true + model.Fields["CloudAppSecurity"].Required = true + + model, ok = models["ConditionalAccessUsers"] + if !ok { + return fmt.Errorf("`ConditionalAccessUsers` model not found") + } + + // cloudAppSecurityPolicy must be null to unset it, so make it nullable + required + if _, ok = model.Fields["ExcludeGuestsOrExternalUsers"]; !ok { + return fmt.Errorf("`ExcludeGuestsOrExternalUsers` field not found") + } + model.Fields["ExcludeGuestsOrExternalUsers"].Nullable = true + model.Fields["ExcludeGuestsOrExternalUsers"].Required = true + if _, ok = model.Fields["IncludeGuestsOrExternalUsers"]; !ok { + return fmt.Errorf("`IncludeGuestsOrExternalUsers` field not found") + } + model.Fields["IncludeGuestsOrExternalUsers"].Nullable = true + model.Fields["IncludeGuestsOrExternalUsers"].Required = true + return nil } From 7432097663b5a6fe7b8f988f6420d9d5bd276dfe Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 13:25:33 +0100 Subject: [PATCH 060/117] generator-go-sdk: test fixes --- .../templater_models_constant_test.go | 19 ++++++++++++++++++- .../templater_models_discriminators_test.go | 4 ++++ .../generator/templater_models_test.go | 3 +++ .../internal/generator/templater_readme.go | 2 +- .../generator/templater_version_test.go | 2 +- 5 files changed, 27 insertions(+), 3 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go b/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go index 3ba3d203d08..d4ff3f8b9c9 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_constant_test.go @@ -67,6 +67,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -134,7 +135,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" - + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -203,6 +204,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -270,6 +272,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -337,6 +340,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -405,6 +409,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -484,6 +489,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -557,6 +563,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -631,6 +638,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -704,6 +712,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -777,6 +786,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -854,6 +864,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -927,6 +938,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -1003,6 +1015,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -1077,6 +1090,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -1150,6 +1164,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -1226,6 +1241,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -1300,6 +1316,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index ebc9c0a5692..5b51b079320 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -70,6 +70,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -206,6 +207,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -328,6 +330,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -429,6 +432,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder diff --git a/tools/generator-go-sdk/internal/generator/templater_models_test.go b/tools/generator-go-sdk/internal/generator/templater_models_test.go index 562d64905e0..6828320d382 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_test.go @@ -71,6 +71,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -145,6 +146,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder @@ -258,6 +260,7 @@ import ( "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" ) // acctests licence placeholder diff --git a/tools/generator-go-sdk/internal/generator/templater_readme.go b/tools/generator-go-sdk/internal/generator/templater_readme.go index 809a9afff26..4e060b98e85 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme.go @@ -62,7 +62,7 @@ This readme covers example usages, but further information on [using this SDK ca '''go %[5]s ''' -`, data.sourceType, data.servicePackageName, data.versionDirectoryName, data.packageName, strings.Join(importLines, "\n")) +`, data.sourceType, data.servicePackageName, data.apiVersion, data.packageName, strings.Join(importLines, "\n")) } func (r readmeTemplater) clientInitialization(packageName, clientName string) string { diff --git a/tools/generator-go-sdk/internal/generator/templater_version_test.go b/tools/generator-go-sdk/internal/generator/templater_version_test.go index 666a9b04f20..1f545e78421 100644 --- a/tools/generator-go-sdk/internal/generator/templater_version_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_version_test.go @@ -28,7 +28,7 @@ import "fmt" const defaultApiVersion = "2022-02-01" func userAgent() string { - return fmt.Sprintf("hashicorp/go-azure-sdk/somepackage/%s", defaultApiVersion) + return "hashicorp/go-azure-sdk/somepackage/2022-02-01" }` assertTemplatedCodeMatches(t, expected, *actual) } From 4a639592968496d451cfd91c3d2c2a6085dc3d92 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 14:35:27 +0100 Subject: [PATCH 061/117] importer-msgraph-metadata: add support for DirectoryRoles --- config/microsoft-graph.hcl | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/config/microsoft-graph.hcl b/config/microsoft-graph.hcl index 29ddecbc534..12ae783b59d 100644 --- a/config/microsoft-graph.hcl +++ b/config/microsoft-graph.hcl @@ -41,6 +41,11 @@ service "directoryObjects" { available = ["stable", "beta"] } +service "directoryRoles" { + name = "DirectoryRoles" + available = ["stable", "beta"] +} + service "domains" { name = "Domains" available = ["stable", "beta"] From b6b96b3edda14199bd92c0f6637e71b97887275c Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 15 Aug 2024 14:37:59 +0100 Subject: [PATCH 062/117] importer-msgraph-metadata: add support for DirectoryRoleTemplates --- config/microsoft-graph.hcl | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/config/microsoft-graph.hcl b/config/microsoft-graph.hcl index 12ae783b59d..5e9ffbb51d6 100644 --- a/config/microsoft-graph.hcl +++ b/config/microsoft-graph.hcl @@ -46,6 +46,11 @@ service "directoryRoles" { available = ["stable", "beta"] } +service "directoryRoleTemplates" { + name = "DirectoryRoleTemplates" + available = ["stable", "beta"] +} + service "domains" { name = "Domains" available = ["stable", "beta"] From e884bf697189f6eff9da72b9d2874f72a3e0fe9c Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 16 Aug 2024 18:07:51 +0100 Subject: [PATCH 063/117] generator-go-sdk: improvements to discriminated model types * Output the discriminated value field instead of hiding it. this is useful because the field may have additional semantics and consumers may wish to know its value. consumers may attempt to set it, but it will have no effect due to the custom marshalling function. * Define a base type for each discriminated model parent, rather than omitting it. * Unmarshal the base model and add a method to discriminator interfaces that returns it. Implement this method for all discriminated child models, including the `Raw{Type}Impl` struct for undefined types. --- .../internal/generator/helpers.go | 4 + .../internal/generator/helpers_test.go | 11 +- .../internal/generator/templater_models.go | 281 ++++++++++++------ .../templater_models_discriminators_test.go | 48 ++- 4 files changed, 242 insertions(+), 102 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/helpers.go b/tools/generator-go-sdk/internal/generator/helpers.go index 6841b458219..c76a85e3841 100644 --- a/tools/generator-go-sdk/internal/generator/helpers.go +++ b/tools/generator-go-sdk/internal/generator/helpers.go @@ -40,6 +40,10 @@ func baseClientPackageForSdk(input models.SourceDataType) string { return "client" } +func camelCase(input string) string { + return strings.ToLower(input[0:1]) + input[1:] +} + func capitalizeFirstLetter(input string) string { return strings.ToUpper(input[0:1]) + strings.ToLower(input[1:]) } diff --git a/tools/generator-go-sdk/internal/generator/helpers_test.go b/tools/generator-go-sdk/internal/generator/helpers_test.go index 0d2c13fe8a0..d9b43f4bd9c 100644 --- a/tools/generator-go-sdk/internal/generator/helpers_test.go +++ b/tools/generator-go-sdk/internal/generator/helpers_test.go @@ -4,6 +4,7 @@ package generator import ( + "fmt" "strings" "testing" @@ -19,8 +20,14 @@ func assertTemplatedCodeMatches(t *testing.T, expected string, actual string) { actualLines := splitLines(actual) expectedLines := splitLines(expected) - normalizedActualValue := strings.Join(actualLines, "\n") - normalizedExpectedValue := strings.Join(expectedLines, "\n") + normalizedActualValue := "" + for i, v := range actualLines { + normalizedActualValue += fmt.Sprintf("%d: %s\n", i+1, v) + } + normalizedExpectedValue := "" + for i, v := range expectedLines { + normalizedExpectedValue += fmt.Sprintf("%d: %s\n", i+1, v) + } if len(actualLines) != len(expectedLines) { t.Fatalf(`Expected %d lines but got %d lines. diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index e81b3537cba..a057fef2548 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -66,22 +66,12 @@ import ( } func (c modelsTemplater) structCode(data GeneratorData) (*string, error) { - // if this is an Abstract/Type Hint, we output an Interface with a manual unmarshal func that gets called wherever it's used - if c.model.FieldNameContainingDiscriminatedValue != nil && c.model.ParentTypeName == nil { - out := fmt.Sprintf(` -type %[1]s interface { -} + out := "" + structName := c.name -// Raw%[1]sImpl is returned when the Discriminated Value -// doesn't match any of the defined types -// NOTE: this should only be used when a type isn't defined for this type of Object (as a workaround) -// and is used only for Deserialization (e.g. this cannot be used as a Request Payload). -type Raw%[1]sImpl struct { - Type string - Values map[string]interface{} -} -`, c.name) - return &out, nil + // parent models get a {model}Base struct + if c.model.IsDiscriminatedParentType() { + structName = fmt.Sprintf("%sBase", c.name) } fields := make([]string, 0) @@ -105,11 +95,6 @@ type Raw%[1]sImpl struct { return nil, err } - if c.model.FieldNameContainingDiscriminatedValue != nil && *c.model.FieldNameContainingDiscriminatedValue == fieldName { - // this isn't user configurable (and is hard-coded) so there's no point outputting this - continue - } - structLines = append(structLines, *structLine) } @@ -127,7 +112,7 @@ type Raw%[1]sImpl struct { } else { parentTypeName = *c.model.ParentTypeName } - parentAssignmentInfo = fmt.Sprintf("var _ %[1]s = %[2]s{}", parentTypeName, c.name) + parentAssignmentInfo = fmt.Sprintf("var _ %[1]s = %[2]s{}", parentTypeName, structName) parent, ok := data.models[*c.model.ParentTypeName] if !ok { @@ -156,12 +141,6 @@ type Raw%[1]sImpl struct { return nil, err } - // check this field isn't used as the discriminated value - if c.model.FieldNameContainingDiscriminatedValue != nil && *c.model.FieldNameContainingDiscriminatedValue == fieldName { - // this isn't user configurable (and is hard-coded) so there's no point outputting this - continue - } - structLines = append(structLines, *structLine) } } @@ -180,12 +159,40 @@ type Raw%[1]sImpl struct { formattedStructLines = append(formattedStructLines, v) } - out := fmt.Sprintf(` + out += fmt.Sprintf(` %[3]s type %[1]s struct { %[2]s } -`, c.name, strings.Join(formattedStructLines, "\n"), parentAssignmentInfo) +`, structName, strings.Join(formattedStructLines, "\n"), parentAssignmentInfo) + + // if this is an Abstract/Type Hint, we output an Interface with a manual unmarshal func that gets called wherever it's used + if c.model.IsDiscriminatedParentType() { + out += fmt.Sprintf(` +type %[1]s interface { + %[1]s() %[1]sBase +} + +// Raw%[1]sImpl is returned when the Discriminated Value +// doesn't match any of the defined types +// NOTE: this should only be used when a type isn't defined for this type of Object (as a workaround) +// and is used only for Deserialization (e.g. this cannot be used as a Request Payload). +type Raw%[1]sImpl struct { + %[2]s %[1]sBase + Type string + Values map[string]interface{} +} + +`, c.name, camelCase(c.name)) + + out += fmt.Sprintf(` +func (s Raw%[1]sImpl) %[1]s() %[1]sBase { + return s.%[2]s +} + +`, c.name, camelCase(c.name)) + } + return &out, nil } @@ -198,6 +205,12 @@ func (c modelsTemplater) methods(data GeneratorData) (*string, error) { } code = append(code, *dateFunctions) + parentModelFunctions, err := c.codeForParentStructFunctions(data) + if err != nil { + return nil, fmt.Errorf("generating parent model functions: %+v", err) + } + code = append(code, *parentModelFunctions) + marshalFunctions, err := c.codeForMarshalFunctions(data) if err != nil { return nil, fmt.Errorf("generating marshal functions: %+v", err) @@ -362,37 +375,97 @@ func (c modelsTemplater) dateFunctionForField(fieldName string, fieldDetails mod return &out, nil } -func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, error) { - output := "" +func (c modelsTemplater) codeForParentStructFunctions(data GeneratorData) (*string, error) { + out := "" - if c.model.DiscriminatedValue != nil { - if c.model.FieldNameContainingDiscriminatedValue == nil { - return nil, fmt.Errorf("model %q must contain a TypeHintIn when a TypeHintValue is present", c.name) - } - if c.model.ParentTypeName == nil { - return nil, fmt.Errorf("model %q must contain a ParentTypeName when a TypeHintValue is present", c.name) + if c.model.ParentTypeName == nil { + return &out, nil + } + + parentTypeName := "" + structFields := make([]string, 0) + + if c.model.FieldNameContainingDiscriminatedValue != nil { + _, foundParentTypeName, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + if err != nil { + return nil, err } + parentTypeName = *foundParentTypeName - parentModel, ok := data.models[*c.model.ParentTypeName] - if !ok { - return nil, fmt.Errorf("the parent model %q for model %q was not found", *c.model.ParentTypeName, c.name) + } else { + parentTypeName = *c.model.ParentTypeName + } + + parent, ok := data.models[*c.model.ParentTypeName] + if !ok { + return nil, fmt.Errorf("couldn't find Parent Model %q for Model %q", *c.model.ParentTypeName, c.name) + } + + parentFields := make([]string, 0) + for fieldName := range parent.Fields { + parentFields = append(parentFields, fieldName) + } + sort.Strings(parentFields) + + if len(parentFields) > 0 { + for _, fieldName := range parentFields { + structFields = append(structFields, fmt.Sprintf(`%[1]s: s.%[1]s,`, fieldName)) } + } - // the TypeHintIn field comes from the parent and so won't be output on the inherited items - field, ok := parentModel.Fields[*c.model.FieldNameContainingDiscriminatedValue] - if !ok { - if parentModel.ParentTypeName != nil { - parentField, _, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) - if err != nil { - return nil, err - } - field = *parentField - } else { - return nil, fmt.Errorf("the field %q was not found on the parent model %q for model %q", *c.model.FieldNameContainingDiscriminatedValue, *c.model.ParentTypeName, c.name) + out += fmt.Sprintf(` +func (s %[1]s) %[2]s() %[2]sBase { + return %[2]sBase{ + %[3]s + } +} + +`, c.name, parentTypeName, strings.Join(structFields, "\n")) + + return &out, nil +} + +func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, error) { + output := "" + + if c.model.DiscriminatedValue == nil { + return &output, nil + } + + if c.model.FieldNameContainingDiscriminatedValue == nil { + return nil, fmt.Errorf("model %q must contain a TypeHintIn when a TypeHintValue is present", c.name) + } + if c.model.ParentTypeName == nil { + return nil, fmt.Errorf("model %q must contain a ParentTypeName when a TypeHintValue is present", c.name) + } + + structName := c.name + + // parent models get a {model}Parent struct + if c.model.IsDiscriminatedParentType() { + structName = fmt.Sprintf("%sBase", c.name) + } + + parentModel, ok := data.models[*c.model.ParentTypeName] + if !ok { + return nil, fmt.Errorf("the parent model %q for model %q was not found", *c.model.ParentTypeName, c.name) + } + + // the TypeHintIn field comes from the parent and so won't be output on the inherited items + field, ok := parentModel.Fields[*c.model.FieldNameContainingDiscriminatedValue] + if !ok { + if parentModel.ParentTypeName != nil { + parentField, _, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + if err != nil { + return nil, err } + field = *parentField + } else { + return nil, fmt.Errorf("the field %q was not found on the parent model %q for model %q", *c.model.FieldNameContainingDiscriminatedValue, *c.model.ParentTypeName, c.name) } + } - output = fmt.Sprintf(` + output += fmt.Sprintf(` var _ json.Marshaler = %[1]s{} func (s %[1]s) MarshalJSON() ([]byte, error) { @@ -416,8 +489,7 @@ func (s %[1]s) MarshalJSON() ([]byte, error) { return encoded, nil } -`, c.name, field.JsonName, *c.model.DiscriminatedValue) - } +`, structName, field.JsonName, *c.model.DiscriminatedValue) return &output, nil } @@ -441,36 +513,41 @@ func (c modelsTemplater) codeForUnmarshalFunctions(data GeneratorData) (*string, } func (c modelsTemplater) codeForUnmarshalParentFunction(data GeneratorData) (*string, error) { + output := "" + + if !c.model.IsDiscriminatedParentType() { + return &output, nil + } + // if this is a Discriminated Type (e.g. Parent) then we need to generate an Unmarshal{Name}Implementation // function which can be used in any usages lines := make([]string, 0) - if c.model.IsDiscriminatedParentType() { - modelsImplementingThisClass := make([]string, 0) - for modelName, model := range data.models { - if model.ParentTypeName == nil || model.FieldNameContainingDiscriminatedValue == nil || model.DiscriminatedValue == nil || modelName == c.name { - continue - } - - // sanity-checking - if *model.ParentTypeName != c.name { - continue - } - - if *model.FieldNameContainingDiscriminatedValue != *c.model.FieldNameContainingDiscriminatedValue { - return nil, fmt.Errorf("implementation %q uses a different discriminated field (%q) than parent %q (%q)", modelName, *model.FieldNameContainingDiscriminatedValue, c.name, *c.model.FieldNameContainingDiscriminatedValue) - } - - modelsImplementingThisClass = append(modelsImplementingThisClass, modelName) + modelsImplementingThisClass := make([]string, 0) + for modelName, model := range data.models { + if model.ParentTypeName == nil || model.FieldNameContainingDiscriminatedValue == nil || model.DiscriminatedValue == nil || modelName == c.name { + continue } // sanity-checking - if len(modelsImplementingThisClass) == 0 && featureflags.SkipDiscriminatedParentTypes() == false { - return nil, fmt.Errorf("model %q is a discriminated parent type with no implementations", c.name) + if *model.ParentTypeName != c.name { + continue } - jsonFieldName := c.model.Fields[*c.model.FieldNameContainingDiscriminatedValue].JsonName - // NOTE: unmarshaling null returns an empty map, which'll mean the `ok` fails - // the 'type' field being omitted will also mean that `ok` is false - lines = append(lines, fmt.Sprintf(` + + if *model.FieldNameContainingDiscriminatedValue != *c.model.FieldNameContainingDiscriminatedValue { + return nil, fmt.Errorf("implementation %q uses a different discriminated field (%q) than parent %q (%q)", modelName, *model.FieldNameContainingDiscriminatedValue, c.name, *c.model.FieldNameContainingDiscriminatedValue) + } + + modelsImplementingThisClass = append(modelsImplementingThisClass, modelName) + } + + // sanity-checking + if len(modelsImplementingThisClass) == 0 && featureflags.SkipDiscriminatedParentTypes() == false { + return nil, fmt.Errorf("model %q is a discriminated parent type with no implementations", c.name) + } + jsonFieldName := c.model.Fields[*c.model.FieldNameContainingDiscriminatedValue].JsonName + // NOTE: unmarshaling null returns an empty map, which'll mean the `ok` fails + // the 'type' field being omitted will also mean that `ok` is false + lines = append(lines, fmt.Sprintf(` func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { if input == nil { return nil, nil @@ -487,11 +564,11 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { } `, c.name, jsonFieldName)) - sort.Strings(modelsImplementingThisClass) - for _, implementationName := range modelsImplementingThisClass { - model := data.models[implementationName] + sort.Strings(modelsImplementingThisClass) + for _, implementationName := range modelsImplementingThisClass { + model := data.models[implementationName] - lines = append(lines, fmt.Sprintf(` + lines = append(lines, fmt.Sprintf(` if strings.EqualFold(value, %[1]q) { var out %[2]s if err := json.Unmarshal(input, &out); err != nil { @@ -500,30 +577,36 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { return out, nil } `, *model.DiscriminatedValue, implementationName)) - } + } + + // if it doesn't match - we generate and deserialize into a 'Raw{Name}Impl' type - named intentionally + // so that we don't conflict with a generated 'Raw{Name}' type which exists in a handful of Swaggers + lines = append(lines, fmt.Sprintf(` + var parent %[1]sBase + if err := json.Unmarshal(input, &parent); err != nil { + return nil, fmt.Errorf("unmarshaling into %[1]sBase: %+v", err) + } - // if it doesn't match - we generate and deserialize into a 'Raw{Name}Impl' type - named intentionally - // so that we don't conflict with a generated 'Raw{Name}' type which exists in a handful of Swaggers - lines = append(lines, fmt.Sprintf(` out := Raw%[1]sImpl{ - Type: value, + %[2]s: parent, + Type: value, Values: temp, } return out, nil -`, c.name)) +`, c.name, camelCase(c.name))) - lines = append(lines, "}") - } + lines = append(lines, "}") - output := strings.Join(lines, "\n") + output += strings.Join(lines, "\n") return &output, nil } func (c modelsTemplater) codeForUnmarshalStructFunction(data GeneratorData) (*string, error) { - // this is a parent, therefore there'll be no struct fields to check here + structName := c.name + + // parent models get a {model}Parent struct if c.model.IsDiscriminatedParentType() { - out := "" - return &out, nil + structName = fmt.Sprintf("%sBase", c.name) } lines := make([]string, 0) @@ -576,7 +659,7 @@ func (c modelsTemplater) codeForUnmarshalStructFunction(data GeneratorData) (*st lines = append(lines, fmt.Sprintf(` var _ json.Unmarshaler = &%[1]s{} -func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) +func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, structName)) // first for each regular field, decode & assign that if len(fieldsRequiringAssignment) > 0 { @@ -585,7 +668,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) if err := json.Unmarshal(bytes, &decoded); err != nil { return fmt.Errorf("unmarshaling into %[1]s: %%+v", err) } -`, c.name)) +`, structName)) sort.Strings(fieldsRequiringAssignment) for _, fieldName := range fieldsRequiringAssignment { @@ -598,7 +681,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) if err := json.Unmarshal(bytes, &temp); err != nil { return fmt.Errorf("unmarshaling %[1]s into map[string]json.RawMessage: %%+v", err) } -`, c.name)) +`, structName)) sort.Strings(fieldsRequiringUnmarshalling) for _, fieldName := range fieldsRequiringUnmarshalling { @@ -658,7 +741,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) output[key] = impl } s.%[1]s = %[4]soutput - }`, fieldName, *topLevelObjectDef.ReferenceName, c.name, assignmentPrefix, fieldDetails.JsonName)) + }`, fieldName, *topLevelObjectDef.ReferenceName, structName, assignmentPrefix, fieldDetails.JsonName)) } if fieldDetails.ObjectDefinition.Type == models.ListSDKObjectDefinitionType { @@ -693,7 +776,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) output = append(output, impl) } s.%[1]s = %[4]soutput - }`, fieldName, *topLevelObjectDef.ReferenceName, c.name, assignmentPrefix, fieldDetails.JsonName)) + }`, fieldName, *topLevelObjectDef.ReferenceName, structName, assignmentPrefix, fieldDetails.JsonName)) } if fieldDetails.ObjectDefinition.Type == models.ReferenceSDKObjectDefinitionType { @@ -704,7 +787,7 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, c.name)) return fmt.Errorf("unmarshaling field '%[1]s' for '%[3]s': %%+v", err) } s.%[1]s = impl - }`, fieldName, *topLevelObjectDef.ReferenceName, c.name, fieldDetails.JsonName)) + }`, fieldName, *topLevelObjectDef.ReferenceName, structName, fieldDetails.JsonName)) } } diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index 5b51b079320..e18fa7f11ce 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -75,7 +75,12 @@ import ( // acctests licence placeholder +type ModeOfTransitBase struct { + Type string ''json:"type"'' +} + type ModeOfTransit interface { + ModeOfTransit() ModeOfTransitBase } // RawModeOfTransitImpl is returned when the Discriminated Value @@ -83,10 +88,15 @@ type ModeOfTransit interface { // NOTE: this should only be used when a type isn't defined for this type of Object (as a workaround) // and is used only for Deserialization (e.g. this cannot be used as a Request Payload). type RawModeOfTransitImpl struct { + modeOfTransit ModeOfTransitBase Type string Values map[string]interface{} } +func (s RawModeOfTransitImpl) ModeOfTransit() ModeOfTransitBase { + return s.modeOfTransit +} + func unmarshalModeOfTransitImplementation(input []byte) (ModeOfTransit, error) { if input == nil { return nil, nil @@ -118,8 +128,14 @@ func unmarshalModeOfTransitImplementation(input []byte) (ModeOfTransit, error) { return out, nil } + var parent ModeOfTransitBase + if err := json.Unmarshal(input, &parent); err != nil { + return nil, fmt.Errorf("unmarshaling into ModeOfTransitBase: modeOfTransit", err) + } + out := RawModeOfTransitImpl{ - Type: value, + modeOfTransit: parent, + Type: value, Values: temp, } return out, nil @@ -158,6 +174,13 @@ func TestTemplaterModelsImplementation(t *testing.T) { "ModeOfTransit": { FieldNameContainingDiscriminatedValue: stringPointer("Type"), Fields: map[string]models.SDKField{ + "Name": { + Required: true, + JsonName: "name", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + }, "Type": { ContainsDiscriminatedValue: true, JsonName: "type", @@ -219,6 +242,15 @@ type Train struct { Operator string ''json:"operator"'' // Fields inherited from ModeOfTransit + Name string ''json:"name"'' + Type string ''json:"type"'' +} + +func (s Train) ModeOfTransit() ModeOfTransitBase { + return ModeOfTransitBase{ + Name: s.Name, + Type: s.Type, + } } var _ json.Marshaler = Train{} @@ -269,6 +301,13 @@ func TestTemplaterModelsFieldImplementation(t *testing.T) { "ModeOfTransit": { FieldNameContainingDiscriminatedValue: stringPointer("Type"), Fields: map[string]models.SDKField{ + "Name": { + Required: true, + JsonName: "name", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + }, "Type": { ContainsDiscriminatedValue: true, JsonName: "type", @@ -443,6 +482,13 @@ type FirstImplementation struct { Serialization Serialization ''json:"serialization"'' // Fields inherited from First + Type string ''json:"type"'' +} + +func (s FirstImplementation) First() FirstBase { + return FirstBase{ + Type: s.Type, + } } var _ json.Marshaler = FirstImplementation{} From d79b5b9bbea5c6eab8a4f42a6905fcd87dad6f0c Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sat, 17 Aug 2024 12:26:48 +0100 Subject: [PATCH 064/117] importer-msgraph-metadata: output unknown array member types as `[]interface{}` --- .../components/parser/types.go | 48 +++++++++++-------- 1 file changed, 29 insertions(+), 19 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 76d57db3e30..b699af5bfb0 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -214,11 +214,11 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - Nullable: f.Nullable, ReferenceName: f.ModelName, ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), Type: sdkModels.ReferenceSDKObjectDefinitionType, }, + Nullable: f.Nullable, ReferenceName: nil, Type: sdkModels.ListSDKObjectDefinitionType, }, nil @@ -228,25 +228,32 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj // TODO validate constant exists return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - Nullable: f.Nullable, ReferenceName: f.ConstantName, Type: sdkModels.ReferenceSDKObjectDefinitionType, }, - Type: sdkModels.ListSDKObjectDefinitionType, + Nullable: f.Nullable, + Type: sdkModels.ListSDKObjectDefinitionType, }, nil } if f.ItemType != nil { return &sdkModels.SDKObjectDefinition{ NestedItem: &sdkModels.SDKObjectDefinition{ - Nullable: f.Nullable, - Type: f.ItemType.DataApiSdkObjectDefinitionType(), + Type: f.ItemType.DataApiSdkObjectDefinitionType(), }, - Type: sdkModels.ListSDKObjectDefinitionType, + Nullable: f.Nullable, + Type: sdkModels.ListSDKObjectDefinitionType, }, nil } - return nil, nil + // Unknown types should be []interface{} + return &sdkModels.SDKObjectDefinition{ + NestedItem: &sdkModels.SDKObjectDefinition{ + Type: sdkModels.RawObjectSDKObjectDefinitionType, + }, + Nullable: f.Nullable, + Type: sdkModels.ListSDKObjectDefinitionType, + }, nil } if f.ConstantName != nil { @@ -690,16 +697,18 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta model := Model{ Fields: map[string]*ModelField{ "ODataId": { - Title: "ODataId", - Type: pointer.To(DataTypeString), - Default: "", - JsonField: "@odata.id", + Title: "ODataId", + Description: "The OData ID of this entity", + Type: pointer.To(DataTypeString), + Default: "", + JsonField: "@odata.id", }, "ODataType": { - Title: "ODataType", - Type: pointer.To(DataTypeString), - Default: "", - JsonField: "@odata.type", + Title: "ODataType", + Description: "The OData Type of this entity", + Type: pointer.To(DataTypeString), + Default: "", + JsonField: "@odata.type", }, }, Common: common, @@ -820,10 +829,11 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta } model.Fields[bindFieldName] = &ModelField{ - Title: bindFieldName, - Type: fieldType, - ItemType: itemType, - JsonField: fmt.Sprintf("%s@odata.bind", jsonField), + Title: bindFieldName, + Description: fmt.Sprintf("List of OData IDs for `%s` to bind to this entity", normalize.CleanName(jsonField)), + Type: fieldType, + ItemType: itemType, + JsonField: fmt.Sprintf("%s@odata.bind", jsonField), } } From 70694a34938ff7779a5eb8053636d5fbe22047ec Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sat, 17 Aug 2024 12:27:25 +0100 Subject: [PATCH 065/117] importer-msgraph-metadata: remove prefix from actions in URL segments --- .../components/parser/resourceids.go | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index ba7e444387c..72616083445 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -381,9 +381,19 @@ func NewResourceId(path string, tags []string) (id ResourceId) { } } else if strings.HasPrefix(strings.ToLower(s), "microsoft.graph.") || strings.HasPrefix(strings.ToLower(s), "graph.") { if tagSuffix(".actions") { + value := s + if strings.HasPrefix(strings.ToLower(value), "microsoft.graph.") { + value = value[16:] + } + if strings.HasPrefix(strings.ToLower(value), "graph.") { + value = value[6:] + } + if value == "" { + value = s + } segment = ResourceIdSegment{ Type: SegmentAction, - Value: s, + Value: value, Field: nil, } } else { From 23b48991275b5594a4ff5024758419238ff8d694 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sat, 17 Aug 2024 15:21:16 +0100 Subject: [PATCH 066/117] generator-go-sdk: handle ReadOnly fields by omitting them when marshalling --- .../internal/generator/helpers_test.go | 2 +- .../internal/generator/templater_models.go | 92 ++++++++------ .../generator/templater_models_test.go | 114 ++++++++++++++++++ 3 files changed, 173 insertions(+), 35 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/helpers_test.go b/tools/generator-go-sdk/internal/generator/helpers_test.go index d9b43f4bd9c..3e9241afeea 100644 --- a/tools/generator-go-sdk/internal/generator/helpers_test.go +++ b/tools/generator-go-sdk/internal/generator/helpers_test.go @@ -61,7 +61,7 @@ Actual Value: --- %s --- -`, i, expectedLine, actualLine, normalizedExpectedValue, normalizedActualValue) +`, i+1, expectedLine, actualLine, normalizedExpectedValue, normalizedActualValue) } } } diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index a057fef2548..ee1b67326f3 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -11,8 +11,6 @@ import ( "github.com/hashicorp/pandora/tools/generator-go-sdk/internal/featureflags" ) -// TODO: add unit tests covering this - var _ templaterForResource = modelsTemplater{} type modelsTemplater struct { @@ -249,7 +247,6 @@ func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDe } } } - // TODO: proper support for ReadOnly fields, which is likely to necessitate a custom marshal func if isOptional || fieldDetails.ReadOnly { if !strings.HasPrefix(fieldType, "nullable.") { fieldType = fmt.Sprintf("*%s", fieldType) @@ -347,7 +344,7 @@ func (c modelsTemplater) dateFunctionForField(fieldName string, fieldDetails mod } // Get{Name}AsTime method for getting *time.Time from a string - if fieldDetails.Optional || fieldDetails.ReadOnly { // TODO: work out how to handle ReadOnly fields + if fieldDetails.Optional || fieldDetails.ReadOnly { linesForField = append(linesForField, fmt.Sprintf("\t\tif o.%s == nil {", fieldName)) linesForField = append(linesForField, fmt.Sprintf("\t\t\treturn nil, nil")) linesForField = append(linesForField, fmt.Sprintf("\t\t}")) @@ -428,15 +425,25 @@ func (s %[1]s) %[2]s() %[2]sBase { func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, error) { output := "" - if c.model.DiscriminatedValue == nil { - return &output, nil + readOnlyFields := make([]string, 0) + for fieldName, field := range c.model.Fields { + if field.ReadOnly { + readOnlyFields = append(readOnlyFields, fieldName) + } } - if c.model.FieldNameContainingDiscriminatedValue == nil { - return nil, fmt.Errorf("model %q must contain a TypeHintIn when a TypeHintValue is present", c.name) + // Only output a Marshal function when there are discriminated value or read-only fields to customize + if c.model.DiscriminatedValue == nil && len(readOnlyFields) == 0 { + return &output, nil } - if c.model.ParentTypeName == nil { - return nil, fmt.Errorf("model %q must contain a ParentTypeName when a TypeHintValue is present", c.name) + + if c.model.DiscriminatedValue != nil { + if c.model.FieldNameContainingDiscriminatedValue == nil { + return nil, fmt.Errorf("model %q must contain a TypeHintIn when a TypeHintValue is present", c.name) + } + if c.model.ParentTypeName == nil { + return nil, fmt.Errorf("model %q must contain a ParentTypeName when a TypeHintValue is present", c.name) + } } structName := c.name @@ -446,25 +453,6 @@ func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, e structName = fmt.Sprintf("%sBase", c.name) } - parentModel, ok := data.models[*c.model.ParentTypeName] - if !ok { - return nil, fmt.Errorf("the parent model %q for model %q was not found", *c.model.ParentTypeName, c.name) - } - - // the TypeHintIn field comes from the parent and so won't be output on the inherited items - field, ok := parentModel.Fields[*c.model.FieldNameContainingDiscriminatedValue] - if !ok { - if parentModel.ParentTypeName != nil { - parentField, _, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) - if err != nil { - return nil, err - } - field = *parentField - } else { - return nil, fmt.Errorf("the field %q was not found on the parent model %q for model %q", *c.model.FieldNameContainingDiscriminatedValue, *c.model.ParentTypeName, c.name) - } - } - output += fmt.Sprintf(` var _ json.Marshaler = %[1]s{} @@ -480,7 +468,37 @@ func (s %[1]s) MarshalJSON() ([]byte, error) { if err := json.Unmarshal(encoded, &decoded); err != nil { return nil, fmt.Errorf("unmarshaling %[1]s: %%+v", err) } - decoded[%[2]q] = %[3]q + +`, structName) + + for _, fieldName := range readOnlyFields { + output += fmt.Sprintf(" delete(decoded, %[1]q)\n", fieldName) + } + + if c.model.DiscriminatedValue != nil { + parentModel, ok := data.models[*c.model.ParentTypeName] + if !ok { + return nil, fmt.Errorf("the parent model %q for model %q was not found", *c.model.ParentTypeName, c.name) + } + + // the TypeHintIn field comes from the parent and so won't be output on the inherited items + field, ok := parentModel.Fields[*c.model.FieldNameContainingDiscriminatedValue] + if !ok { + if parentModel.ParentTypeName != nil { + parentField, _, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + if err != nil { + return nil, err + } + field = *parentField + } else { + return nil, fmt.Errorf("the field %q was not found on the parent model %q for model %q", *c.model.FieldNameContainingDiscriminatedValue, *c.model.ParentTypeName, c.name) + } + } + + output += fmt.Sprintf(" decoded[%[1]q] = %[2]q\n", field.JsonName, *c.model.DiscriminatedValue) + } + + output += fmt.Sprintf(` encoded, err = json.Marshal(decoded) if err != nil { @@ -489,7 +507,7 @@ func (s %[1]s) MarshalJSON() ([]byte, error) { return encoded, nil } -`, structName, field.JsonName, *c.model.DiscriminatedValue) +`, structName) return &output, nil } @@ -779,15 +797,21 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, structName)) }`, fieldName, *topLevelObjectDef.ReferenceName, structName, assignmentPrefix, fieldDetails.JsonName)) } + // if the field is read-only, we need to assign the pointer value + assignmentPrefix := "" + if fieldDetails.ReadOnly { + assignmentPrefix = "&" + } + if fieldDetails.ObjectDefinition.Type == models.ReferenceSDKObjectDefinitionType { lines = append(lines, fmt.Sprintf(` - if v, ok := temp[%[4]q]; ok { + if v, ok := temp[%[5]q]; ok { impl, err := Unmarshal%[2]sImplementation(v) if err != nil { return fmt.Errorf("unmarshaling field '%[1]s' for '%[3]s': %%+v", err) } - s.%[1]s = impl - }`, fieldName, *topLevelObjectDef.ReferenceName, structName, fieldDetails.JsonName)) + s.%[1]s = %[4]simpl + }`, fieldName, *topLevelObjectDef.ReferenceName, structName, assignmentPrefix, fieldDetails.JsonName)) } } diff --git a/tools/generator-go-sdk/internal/generator/templater_models_test.go b/tools/generator-go-sdk/internal/generator/templater_models_test.go index 6828320d382..763ca656fba 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_test.go @@ -273,3 +273,117 @@ type Basic struct { `, "''", "`") assertTemplatedCodeMatches(t, expected, *actual) } + +func TestModelTemplaterWithReadOnlyField(t *testing.T) { + actual, err := modelsTemplater{ + name: "Example", + model: models.SDKModel{ + Fields: map[string]models.SDKField{ + "Name": { + JsonName: "name", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + Required: true, + }, + "Description": { + JsonName: "description", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + Optional: true, + }, + "Type": { + JsonName: "type", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + ReadOnly: true, + }, + }, + }, + }.template(GeneratorData{ + packageName: "somepackage", + models: map[string]models.SDKModel{ + "Example": { + Fields: map[string]models.SDKField{ + "Name": { + JsonName: "name", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + Required: true, + }, + "Description": { + JsonName: "description", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + Optional: true, + }, + "Type": { + JsonName: "type", + ObjectDefinition: models.SDKObjectDefinition{ + Type: models.StringSDKObjectDefinitionType, + }, + ReadOnly: true, + }, + }, + }, + }, + source: AccTestLicenceType, + }) + if err != nil { + t.Fatal(err.Error()) + } + expected := strings.ReplaceAll(`package somepackage + +import ( + "encoding/json" + "fmt" + "strings" + "time" + "github.com/hashicorp/go-azure-helpers/lang/dates" + "github.com/hashicorp/go-azure-helpers/resourcemanager/edgezones" + "github.com/hashicorp/go-azure-helpers/resourcemanager/identity" + "github.com/hashicorp/go-azure-helpers/resourcemanager/systemdata" + "github.com/hashicorp/go-azure-helpers/resourcemanager/zones" + "github.com/hashicorp/go-azure-sdk/sdk/nullable" +) + +// acctests licence placeholder + +type Example struct { + Description *string ''json:"description,omitempty"'' + Name string ''json:"name"'' + Type *string ''json:"type,omitempty"'' +} + +var _ json.Marshaler = Example{} + +func (s Example) MarshalJSON() ([]byte, error) { + type wrapper Example + wrapped := wrapper(s) + encoded, err := json.Marshal(wrapped) + if err != nil { + return nil, fmt.Errorf("marshaling Example: %+v", err) + } + + var decoded map[string]interface{} + if err := json.Unmarshal(encoded, &decoded); err != nil { + return nil, fmt.Errorf("unmarshaling Example: %+v", err) + } + + delete(decoded, "Type") + encoded, err = json.Marshal(decoded) + + if err != nil { + return nil, fmt.Errorf("re-marshaling Example: %+v", err) + } + + return encoded, nil +} + +`, "''", "`") + assertTemplatedCodeMatches(t, expected, *actual) +} From f55f7ad9075ff47633a50ff0a9dfb6283f40bb04 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sat, 17 Aug 2024 15:23:42 +0100 Subject: [PATCH 067/117] importer-msgraph-metadata: workaround for common read-only fields --- .../workaround_conditionalaccesspolicy.go | 1 + .../workarounds/workaround_readonlyfields.go | 28 +++++++++++++++++++ .../components/workarounds/workarounds.go | 1 + 3 files changed, 30 insertions(+) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index c6fdb6132d1..86ca59486ca 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -5,6 +5,7 @@ package workarounds import ( "fmt" + "github.com/hashicorp/go-azure-helpers/lang/pointer" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" ) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go new file mode 100644 index 00000000000..937aa07566c --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go @@ -0,0 +1,28 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ workaround = workaroundReadOnlyFields{} + +// workaroundReadOnlyFields ensures all common readonly fields are marked as such - unfortunately the spec +// does not mark these as ReadOnly. +type workaroundReadOnlyFields struct{} + +func (workaroundReadOnlyFields) Name() string { + return "Read-Only Fields / override common readonly fields" +} + +func (workaroundReadOnlyFields) Process(apiVersion string, models parser.Models, constants parser.Constants) error { + for _, model := range models { + if _, ok := model.Fields["CreatedOnBehalfOf"]; ok { + model.Fields["CreatedOnBehalfOf"].ReadOnly = true + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index 6de4b99f24d..5dad0dd3734 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -11,6 +11,7 @@ import ( ) var workarounds = []workaround{ + workaroundReadOnlyFields{}, workaroundApplication{}, workaroundConditionalAccessPolicy{}, workaroundNamedLocation{}, From 48e59a17835c99c29fd9cab48d858689e5aa1a8b Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sat, 17 Aug 2024 15:24:18 +0100 Subject: [PATCH 068/117] importer-msgraph-metadata: workaround to implement discrimination for DirectoryObjects --- .../workarounds/workaround_directoryobject.go | 114 ++++++++++++++++++ .../components/workarounds/workarounds.go | 1 + 2 files changed, 115 insertions(+) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go new file mode 100644 index 00000000000..d8c8d71b916 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go @@ -0,0 +1,114 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ workaround = workaroundDirectoryObject{} + +// workaroundDirectoryObject implements discrimination +type workaroundDirectoryObject struct{} + +func (workaroundDirectoryObject) Name() string { + return "Directory Object / discrimination" +} + +func (workaroundDirectoryObject) Process(apiVersion string, models parser.Models, constants parser.Constants) error { + model, ok := models["DirectoryObject"] + if !ok { + return fmt.Errorf("`DirectoryObject` model not found") + } + + if _, ok = model.Fields["ODataType"]; !ok { + return fmt.Errorf("`ODataType` field not found in `DireectoryObject` model") + } + + model.Fields["ODataType"].ConstantName = pointer.To("DirectoryObjectType") + model.Fields["ODataType"].DiscriminatedValue = true + model.TypeField = pointer.To("ODataType") + + // Add constant values for the discriminated type value + constants["DirectoryObjectType"] = &parser.Constant{ + Enum: []string{ + "#microsoft.graph.administrativeUnit", + "#microsoft.graph.application", + "#microsoft.graph.device", + "#microsoft.graph.group", + "#microsoft.graph.orgContact", + "#microsoft.graph.servicePrincipal", + "#microsoft.graph.user", + }, + Type: pointer.To(parser.DataTypeString), + } + + // Set the parent model and discriminated type value for AdministrativeUnit + model, ok = models["AdministrativeUnit"] + if !ok { + return fmt.Errorf("`AdministrativeUnit` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.administrativeUnit") + + // Set the parent model and discriminated type value for Application + model, ok = models["Application"] + if !ok { + return fmt.Errorf("`Application` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.application") + + // Set the parent model and discriminated type value for Device + model, ok = models["Device"] + if !ok { + return fmt.Errorf("`Device` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.device") + + // Set the parent model and discriminated type value for Group + model, ok = models["Group"] + if !ok { + return fmt.Errorf("`Group` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.group") + + // Set the parent model and discriminated type value for OrgContact + model, ok = models["OrgContact"] + if !ok { + return fmt.Errorf("`OrgContact` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.orgContact") + + // Set the parent model and discriminated type value for ServicePrincipal + model, ok = models["ServicePrincipal"] + if !ok { + return fmt.Errorf("`ServicePrincipal` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.servicePrincipal") + + // Set the parent model and discriminated type value for User + model, ok = models["User"] + if !ok { + return fmt.Errorf("`User` model not found") + } + model.ParentModelName = pointer.To("DirectoryObject") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#microsoft.graph.user") + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index 5dad0dd3734..c2c851a1ec0 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -14,6 +14,7 @@ var workarounds = []workaround{ workaroundReadOnlyFields{}, workaroundApplication{}, workaroundConditionalAccessPolicy{}, + workaroundDirectoryObject{}, workaroundNamedLocation{}, workaroundIPRange{}, } From d559a5977404a1a8059a96d7793c9c1195e95309 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sat, 17 Aug 2024 15:25:16 +0100 Subject: [PATCH 069/117] importer-msgraph-metadata: clean up field description for OData Bind fields --- .../importer-msgraph-metadata/components/parser/types.go | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index b699af5bfb0..c37d5b04ffd 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -820,6 +820,13 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta if field.ModelName != nil && strings.EqualFold(*field.ModelName, "DirectoryObject") { bindFieldName := fmt.Sprintf("%s_ODataBind", normalize.CleanName(jsonField)) + description := "" + if *field.Type == DataTypeArray { + description = fmt.Sprintf("List of OData IDs for `%s` to bind to this entity", normalize.CleanName(jsonField)) + } else { + description = fmt.Sprintf("OData ID for `%s` to bind to this entity", normalize.CleanName(jsonField)) + } + var fieldType, itemType *DataType fieldType = pointer.To(DataTypeString) @@ -830,7 +837,7 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta model.Fields[bindFieldName] = &ModelField{ Title: bindFieldName, - Description: fmt.Sprintf("List of OData IDs for `%s` to bind to this entity", normalize.CleanName(jsonField)), + Description: description, Type: fieldType, ItemType: itemType, JsonField: fmt.Sprintf("%s@odata.bind", jsonField), From 91b5eb09432b0e552abff115f188cd6aa4481b3e Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 03:56:30 +0100 Subject: [PATCH 070/117] data-api: support descriptions for operations --- .../repository/internal/models/operations.go | 3 +++ .../internal/transforms/sdk_operation.go | 2 ++ .../sdk_operation_object_definition.go | 16 +++++++++++++--- .../golang_type_for_sdk_object_definition.go | 1 + tools/data-api-sdk/v1/models/sdk_operation.go | 3 +++ 5 files changed, 22 insertions(+), 3 deletions(-) diff --git a/tools/data-api-repository/repository/internal/models/operations.go b/tools/data-api-repository/repository/internal/models/operations.go index 7114e73a658..bbd0e31114c 100644 --- a/tools/data-api-repository/repository/internal/models/operations.go +++ b/tools/data-api-repository/repository/internal/models/operations.go @@ -11,6 +11,9 @@ type Operation struct { // ContentType specifies the format of the information being sent with the Operation (e.g. `application/json; charset=utf-8`) ContentType string `json:"contentType"` + // Description is used to write a comment for the operation method + Description string `json:"description"` + // ExpectedStatusCodes specifies is a list of Status Codes which are expected to be returned (e.g. 200, 201) ExpectedStatusCodes []int `json:"expectedStatusCodes"` diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_operation.go b/tools/data-api-repository/repository/internal/transforms/sdk_operation.go index e82e3e70390..71209fcfc87 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_operation.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_operation.go @@ -28,6 +28,7 @@ func MapSDKOperationFromRepository(input repositoryModels.Operation, knownData h } output := sdkModels.SDKOperation{ ContentType: input.ContentType, + Description: input.Description, ExpectedStatusCodes: input.ExpectedStatusCodes, FieldContainingPaginationDetails: input.FieldContainingPaginationDetails, LongRunning: input.LongRunning, @@ -74,6 +75,7 @@ func MapSDKOperationToRepository(operationName string, input sdkModels.SDKOperat output := repositoryModels.Operation{ Name: operationName, ContentType: contentType, + Description: input.Description, ExpectedStatusCodes: input.ExpectedStatusCodes, FieldContainingPaginationDetails: input.FieldContainingPaginationDetails, LongRunning: input.LongRunning, diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go b/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go index 88e1d1cf707..61fdc38e133 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_operation_object_definition.go @@ -12,6 +12,8 @@ import ( sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) +var specialPackageNames = []string{"odata"} + func mapSDKOperationOptionObjectDefinitionFromRepository(input repositoryModels.OptionObjectDefinition, knownData helpers.KnownData) (*sdkModels.SDKOperationOptionObjectDefinition, error) { typeVal, ok := sdkOperationOptionsFromRepository[input.Type] if !ok { @@ -25,7 +27,13 @@ func mapSDKOperationOptionObjectDefinitionFromRepository(input repositoryModels. } if input.ReferenceName != nil { - if !strings.HasPrefix(*input.ReferenceName, "odata.") { + referencedObjectInSpecialPackage := false + for _, packageName := range specialPackageNames { + if strings.HasPrefix(*input.ReferenceName, fmt.Sprintf("%s.", packageName)) { + referencedObjectInSpecialPackage = true + } + } + if !referencedObjectInSpecialPackage { isConstant := knownData.ConstantExists(*input.ReferenceName) isModel := knownData.ModelExists(*input.ReferenceName) if !isConstant && !isModel { @@ -96,8 +104,10 @@ func validateSDKOperationOptionObjectDefinition(input repositoryModels.OptionObj return fmt.Errorf("a Reference must be specified for a %q type but didn't get one", string(input.Type)) } - if strings.HasPrefix(*input.ReferenceName, "odata.") { - return nil + for _, packageName := range specialPackageNames { + if strings.HasPrefix(*input.ReferenceName, fmt.Sprintf("%s.", packageName)) { + return nil + } } isConstant := knownData.ConstantExists(*input.ReferenceName) diff --git a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go index bac359bbbda..90175f6b341 100644 --- a/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go +++ b/tools/data-api-sdk/v1/helpers/golang_type_for_sdk_object_definition.go @@ -57,6 +57,7 @@ func GolangTypeForSDKObjectDefinition(input models.SDKObjectDefinition, golangPa return nil, fmt.Errorf("missing Reference for a Reference ObjectDefinition") } + // Prepend a containing package name to the type where necessary out := *input.ReferenceName if golangPackageName != nil { out = fmt.Sprintf("%s.%s", *golangPackageName, out) diff --git a/tools/data-api-sdk/v1/models/sdk_operation.go b/tools/data-api-sdk/v1/models/sdk_operation.go index bd1b022ea4f..f17e04d7da3 100644 --- a/tools/data-api-sdk/v1/models/sdk_operation.go +++ b/tools/data-api-sdk/v1/models/sdk_operation.go @@ -12,6 +12,9 @@ type SDKOperation struct { // performing the Request for this Operation. ContentType string `json:"contentType"` + // Description is used to write a comment for the operation method + Description string `json:"description"` + // ExpectedStatusCodes specifies the list of Status Codes which are expected to be // returned by this Operation. ExpectedStatusCodes []int `json:"expectedStatusCodes"` From 90ccc1019a7e063e579166bb0ac109ed38e7e291 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 03:56:55 +0100 Subject: [PATCH 071/117] generator-go-sdk: use operation descriptions to output method comments --- .../internal/generator/helpers.go | 22 ++++++++++++ .../internal/generator/templater_methods.go | 36 +++++++++++++++---- 2 files changed, 52 insertions(+), 6 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/helpers.go b/tools/generator-go-sdk/internal/generator/helpers.go index c76a85e3841..b349074bab6 100644 --- a/tools/generator-go-sdk/internal/generator/helpers.go +++ b/tools/generator-go-sdk/internal/generator/helpers.go @@ -160,3 +160,25 @@ func wordifyString(input string) string { return strings.TrimPrefix(output, " ") } + +// wrapOnWordBoundary attempts to wrap a string on whitespace to the specified maximum length, optionally +// prefixing each line with the provided prefix (useful for comments). +func wrapOnWordBoundary(in string, maxLength int, prefix string) string { + words := strings.Fields(in) + out := make([]string, 0) + currentLine := prefix + + for _, word := range words { + if len(currentLine)+len(word) > maxLength { + out = append(out, currentLine) + currentLine = prefix + } + currentLine = fmt.Sprintf("%s %s", currentLine, word) + } + + if currentLine != prefix { + out = append(out, currentLine) + } + + return strings.Join(out, "\n") +} diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 9e5e6f865d3..a55c14c7086 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -163,12 +163,20 @@ func (c methodsPandoraTemplater) immediateOperationTemplate(data GeneratorData) return nil, fmt.Errorf("building options struct: %+v", err) } + comment := c.operationName + if c.operation.Description != "" { + comment += fmt.Sprintf(": %s", c.operation.Description) + } else { + comment += " ..." + } + comment = wrapOnWordBoundary(comment, 120, "//") + templated := fmt.Sprintf(` %[7]s %[8]s %[9]s -// %[2]s ... +%[10]s func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, err error) { opts := %[4]s @@ -194,7 +202,7 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, return } -`, data.serviceClientName, c.operationName, *methodArguments, *requestOptions, *marshalerCode, *unmarshalerCode, *responseStruct, *optionsStruct, requestOptionStruct) +`, data.serviceClientName, c.operationName, *methodArguments, *requestOptions, *marshalerCode, *unmarshalerCode, *responseStruct, *optionsStruct, requestOptionStruct, comment) return &templated, nil } @@ -226,12 +234,20 @@ func (c methodsPandoraTemplater) longRunningOperationTemplate(data GeneratorData return nil, fmt.Errorf("building options struct: %+v", err) } + comment := c.operationName + if c.operation.Description != "" { + comment += fmt.Sprintf(": %s", c.operation.Description) + } else { + comment += " ..." + } + comment = wrapOnWordBoundary(comment, 120, "//") + templated := fmt.Sprintf(` %[9]s %[10]s %[11]s -// %[3]s ... +%[12]s func (c %[1]s) %[3]s(ctx context.Context %[4]s) (result %[3]sOperationResponse, err error) { opts := %[5]s @@ -275,7 +291,7 @@ func (c %[1]s) %[3]sThenPoll(ctx context.Context %[4]s) error { return nil } -`, data.serviceClientName, data.baseClientPackage, c.operationName, *methodArguments, *requestOptions, *marshalerCode, *unmarshalerCode, argumentsCode, *responseStruct, *optionsStruct, requestOptionStruct) +`, data.serviceClientName, data.baseClientPackage, c.operationName, *methodArguments, *requestOptions, *marshalerCode, *unmarshalerCode, argumentsCode, *responseStruct, *optionsStruct, requestOptionStruct, comment) return &templated, nil } @@ -313,12 +329,20 @@ func (c methodsPandoraTemplater) listOperationTemplate(data GeneratorData) (*str predicateName = fmt.Sprintf("%s%s", *typeName, predicateName) } + comment := c.operationName + if c.operation.Description != "" { + comment += fmt.Sprintf(": %s", c.operation.Description) + } else { + comment += " ..." + } + comment = wrapOnWordBoundary(comment, 120, "//") + templated := fmt.Sprintf(` %[6]s %[7]s %[8]s -// %[2]s ... +%[9]s func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, err error) { opts := %[4]s @@ -341,7 +365,7 @@ func (c %[1]s) %[2]s(ctx context.Context %[3]s) (result %[2]sOperationResponse, return } -`, data.serviceClientName, c.operationName, *methodArguments, *requestOptions, *unmarshalerCode, *responseStruct, *optionsStruct, requestOptionStruct) +`, data.serviceClientName, c.operationName, *methodArguments, *requestOptions, *unmarshalerCode, *responseStruct, *optionsStruct, requestOptionStruct, comment) // Only output predicate functions for models and not for base types like string, int etc. if c.operation.ResponseObject.Type == models.ReferenceSDKObjectDefinitionType || c.operation.ResponseObject.Type == models.ListSDKObjectDefinitionType { From 5311455afb95e0c0809c6cd93d90fa774b1fcfb6 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 03:57:25 +0100 Subject: [PATCH 072/117] importer-msgraph-metadata: set descriptions for operations --- .../components/parser/resources.go | 1 + .../internal/pipeline/task_parse_resources.go | 9 +++++++++ .../internal/pipeline/task_translate_service.go | 1 + 3 files changed, 11 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/parser/resources.go b/tools/importer-msgraph-metadata/components/parser/resources.go index 755ea85ff10..11dc30a6138 100644 --- a/tools/importer-msgraph-metadata/components/parser/resources.go +++ b/tools/importer-msgraph-metadata/components/parser/resources.go @@ -21,6 +21,7 @@ type Resource struct { type Operation struct { Name string + Description string Type OperationType Method string ResourceId *ResourceId diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index afc40d4335d..3bb63ee2bd3 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -346,8 +346,17 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + operationDescription := make([]string, 0) + if operation.Summary != "" { + operationDescription = append(operationDescription, operation.Summary) + } + if operation.Description != "" { + operationDescription = append(operationDescription, operation.Description) + } + resources[resourceName].Operations = append(resources[resourceName].Operations, parser.Operation{ Name: operationName, + Description: strings.Join(operationDescription, ". "), Type: operationType, Method: method, ResourceId: resourceId, diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index e1d1c61dfe8..e632e89fb61 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -296,6 +296,7 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv sdkService.APIVersions[resource.Version].Resources[resource.Category].Operations[operation.Name] = sdkModels.SDKOperation{ ContentType: contentType, + Description: operation.Description, ExpectedStatusCodes: expectedStatusCodes, FieldContainingPaginationDetails: operation.PaginationField, LongRunning: false, From a23cef6693628ee5f619f29ac55b1344609a1cba Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 15:55:03 +0100 Subject: [PATCH 073/117] importer-msgraph-metadata: clean up operation descriptions --- .../internal/pipeline/task_parse_resources.go | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 3bb63ee2bd3..f1379c1e3f2 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -346,17 +346,22 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } - operationDescription := make([]string, 0) + descriptionChunks := make([]string, 0) if operation.Summary != "" { - operationDescription = append(operationDescription, operation.Summary) + descriptionChunks = append(descriptionChunks, operation.Summary) } if operation.Description != "" { - operationDescription = append(operationDescription, operation.Description) + descriptionChunks = append(descriptionChunks, operation.Description) } + // Trim a trailing colon/semicolon from descriptions because they are confusing + operationDescription := strings.Join(descriptionChunks, ". ") + operationDescription = strings.TrimPrefix(operationDescription, ":") + operationDescription = strings.TrimPrefix(operationDescription, ";") + resources[resourceName].Operations = append(resources[resourceName].Operations, parser.Operation{ Name: operationName, - Description: strings.Join(operationDescription, ". "), + Description: operationDescription, Type: operationType, Method: method, ResourceId: resourceId, From de9120adc4ebc11e76d406c58102dc7a02e0cd1b Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 19:42:00 +0100 Subject: [PATCH 074/117] generator-go-sdk: sort read-only fields for consistency --- tools/generator-go-sdk/internal/generator/templater_models.go | 1 + 1 file changed, 1 insertion(+) diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index ee1b67326f3..c982ca2573a 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -431,6 +431,7 @@ func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, e readOnlyFields = append(readOnlyFields, fieldName) } } + sort.Strings(readOnlyFields) // Only output a Marshal function when there are discriminated value or read-only fields to customize if c.model.DiscriminatedValue == nil && len(readOnlyFields) == 0 { From e0f4e9a30ec33f3f4b042d7ea0667e4a7821a4ef Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 19:49:17 +0100 Subject: [PATCH 075/117] generator-go-sdk: should use the actual json field name when working with a decoded map --- tools/generator-go-sdk/internal/generator/templater_models.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index c982ca2573a..b2d9712f32e 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -426,9 +426,9 @@ func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, e output := "" readOnlyFields := make([]string, 0) - for fieldName, field := range c.model.Fields { + for _, field := range c.model.Fields { if field.ReadOnly { - readOnlyFields = append(readOnlyFields, fieldName) + readOnlyFields = append(readOnlyFields, field.JsonName) } } sort.Strings(readOnlyFields) From 1b7c65e1243a0a04cf8780bcee6f563905ce32d3 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 19:50:02 +0100 Subject: [PATCH 076/117] importer-msgraph-metadata: use `interface{}` not `string` for unknown field types --- tools/importer-msgraph-metadata/components/parser/types.go | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index c37d5b04ffd..295a9935ef9 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -121,7 +121,6 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { } if objectDefinition == nil { - //return nil, fmt.Errorf("could not determine SDKObjectDefinition for field: %s", fieldName) logging.Warnf("Could not determine SDKObjectDefinition for field %q, skipping", fieldName) continue } @@ -317,8 +316,8 @@ func (ft DataType) DataApiSdkObjectDefinitionType() sdkModels.SDKObjectDefinitio return sdkModels.RawFileSDKObjectDefinitionType } - // Fall back to string where the type is not known - return sdkModels.StringSDKObjectDefinitionType + // Fall back to `interface{}` where the type is not known + return sdkModels.RawObjectSDKObjectDefinitionType } func (ft DataType) DataApiSdkOperationOptionObjectDefinitionType() sdkModels.SDKOperationOptionObjectDefinitionType { From f082d47d5ab927f61dea96bdb588dd09443419c9 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 19:50:45 +0100 Subject: [PATCH 077/117] importer-msgraph-metadata: detect nullable, read-only and required fields from the field description. remove manual workaround. --- .../components/parser/types.go | 11 ++++++++ .../workaround_conditionalaccesspolicy.go | 2 +- .../workarounds/workaround_directoryobject.go | 3 +- .../workarounds/workaround_iprange.go | 2 +- .../workarounds/workaround_namedlocation.go | 2 +- .../workarounds/workaround_readonlyfields.go | 28 ------------------- .../components/workarounds/workarounds.go | 1 - 7 files changed, 16 insertions(+), 33 deletions(-) delete mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 295a9935ef9..20a7faff251 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -736,6 +736,17 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta JsonField: jsonField, } + // Detect nullable, read-only and requried fields from the description + if strings.Contains(schema.Description, "Nullable.") { + field.Nullable = true + } + if strings.Contains(schema.Description, "Read-only.") { + field.ReadOnly = true + } + if strings.Contains(schema.Description, "Required.") { + field.Required = true + } + if field.Title == "" { continue } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index 86ca59486ca..45e9d556814 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -116,7 +116,7 @@ func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parse return fmt.Errorf("`ConditionalAccessUsers` model not found") } - // cloudAppSecurityPolicy must be null to unset it, so make it nullable + required + // excludeGuestsOrExternalUsers / includeGuestsOrExternalUsers must be null to unset them, so make them nullable + required if _, ok = model.Fields["ExcludeGuestsOrExternalUsers"]; !ok { return fmt.Errorf("`ExcludeGuestsOrExternalUsers` field not found") } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go index d8c8d71b916..7537117688b 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go @@ -12,7 +12,8 @@ import ( var _ workaround = workaroundDirectoryObject{} -// workaroundDirectoryObject implements discrimination +// workaroundDirectoryObject implements discrimination for the most common implementations of DirectoryObject. More +// can be added as necessary over time; unfortunately there doesn't seem to be any way of auto-detecting these. type workaroundDirectoryObject struct{} func (workaroundDirectoryObject) Name() string { diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go index f66abd30778..73bd11dd025 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go @@ -12,7 +12,7 @@ import ( var _ workaround = workaroundIPRange{} -// workaroundIPRange implements discrimination +// workaroundIPRange implements discrimination for iPv4CidrRange, iPv4Range, iPv6CidrRange, and iPv6Range type workaroundIPRange struct{} func (workaroundIPRange) Name() string { diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go index 8a66c1994a8..540101b3941 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go @@ -12,7 +12,7 @@ import ( var _ workaround = workaroundNamedLocation{} -// workaroundNamedLocation implements discrimination +// workaroundNamedLocation implements discrimination for countryNamedLocation and ipNamedLocation type workaroundNamedLocation struct{} func (workaroundNamedLocation) Name() string { diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go deleted file mode 100644 index 937aa07566c..00000000000 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_readonlyfields.go +++ /dev/null @@ -1,28 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package workarounds - -import ( - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -var _ workaround = workaroundReadOnlyFields{} - -// workaroundReadOnlyFields ensures all common readonly fields are marked as such - unfortunately the spec -// does not mark these as ReadOnly. -type workaroundReadOnlyFields struct{} - -func (workaroundReadOnlyFields) Name() string { - return "Read-Only Fields / override common readonly fields" -} - -func (workaroundReadOnlyFields) Process(apiVersion string, models parser.Models, constants parser.Constants) error { - for _, model := range models { - if _, ok := model.Fields["CreatedOnBehalfOf"]; ok { - model.Fields["CreatedOnBehalfOf"].ReadOnly = true - } - } - - return nil -} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index c2c851a1ec0..f78f8c627a1 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -11,7 +11,6 @@ import ( ) var workarounds = []workaround{ - workaroundReadOnlyFields{}, workaroundApplication{}, workaroundConditionalAccessPolicy{}, workaroundDirectoryObject{}, From a06fc9a992489382c62af0d2ac91555450bcd079 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 19:55:37 +0100 Subject: [PATCH 078/117] importer-msgraph-metadata: extra safety when assumung nullable fields from description --- tools/importer-msgraph-metadata/components/parser/types.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 20a7faff251..05cf5c6e30e 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -737,13 +737,13 @@ func Schemas(input flattenedSchema, name string, models Models, constants Consta } // Detect nullable, read-only and requried fields from the description - if strings.Contains(schema.Description, "Nullable.") { + if (strings.HasPrefix(schema.Description, "Nullable.") || strings.Contains(schema.Description, " Nullable.")) && !strings.Contains(strings.ToLower(schema.Description), "not nullable.") { field.Nullable = true } - if strings.Contains(schema.Description, "Read-only.") { + if strings.HasPrefix(schema.Description, "Read-only.") || strings.Contains(schema.Description, " Read-only.") { field.ReadOnly = true } - if strings.Contains(schema.Description, "Required.") { + if strings.HasPrefix(schema.Description, "Required.") || strings.Contains(schema.Description, " Required.") { field.Required = true } From 85d039856bc040a40d9c961e162ce41907a64bb9 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 23:32:45 +0100 Subject: [PATCH 079/117] generator-go-sdk: bugfix - escape formatting directive --- tools/generator-go-sdk/internal/generator/templater_models.go | 2 +- .../internal/generator/templater_models_discriminators_test.go | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index b2d9712f32e..bb5a3e85d1e 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -603,7 +603,7 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { lines = append(lines, fmt.Sprintf(` var parent %[1]sBase if err := json.Unmarshal(input, &parent); err != nil { - return nil, fmt.Errorf("unmarshaling into %[1]sBase: %+v", err) + return nil, fmt.Errorf("unmarshaling into %[1]sBase: %%+v", err) } out := Raw%[1]sImpl{ diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index e18fa7f11ce..825153f7542 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -130,7 +130,7 @@ func unmarshalModeOfTransitImplementation(input []byte) (ModeOfTransit, error) { var parent ModeOfTransitBase if err := json.Unmarshal(input, &parent); err != nil { - return nil, fmt.Errorf("unmarshaling into ModeOfTransitBase: modeOfTransit", err) + return nil, fmt.Errorf("unmarshaling into ModeOfTransitBase: %+v", err) } out := RawModeOfTransitImpl{ From 789b1b37d64755b86a3451c32baf470071a97e8d Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Sun, 18 Aug 2024 23:50:10 +0100 Subject: [PATCH 080/117] generator-go-sdk: wrap comment lines for model fields --- tools/generator-go-sdk/internal/generator/helpers.go | 6 +++--- .../generator-go-sdk/internal/generator/templater_models.go | 6 +++--- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/helpers.go b/tools/generator-go-sdk/internal/generator/helpers.go index b349074bab6..4a7d71f1cb5 100644 --- a/tools/generator-go-sdk/internal/generator/helpers.go +++ b/tools/generator-go-sdk/internal/generator/helpers.go @@ -169,15 +169,15 @@ func wrapOnWordBoundary(in string, maxLength int, prefix string) string { currentLine := prefix for _, word := range words { - if len(currentLine)+len(word) > maxLength { - out = append(out, currentLine) + if len(currentLine)+len(word) >= maxLength { + out = append(out, strings.TrimSpace(currentLine)) currentLine = prefix } currentLine = fmt.Sprintf("%s %s", currentLine, word) } if currentLine != prefix { - out = append(out, currentLine) + out = append(out, strings.TrimSpace(currentLine)) } return strings.Join(out, "\n") diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index bb5a3e85d1e..7ed1c799e71 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -146,7 +146,7 @@ func (c modelsTemplater) structCode(data GeneratorData) (*string, error) { formattedStructLines := make([]string, 0) for i, v := range structLines { - if strings.Contains(v, "//") { + if strings.HasPrefix(strings.TrimSpace(v), "//") { if i > 0 && !strings.HasSuffix(formattedStructLines[i-1], "\n") { v = "\n" + v } @@ -259,8 +259,8 @@ func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDe line := fmt.Sprintf("\t%s %s `json:\"%s\"`", fieldName, fieldType, jsonDetails) if data.generateDescriptionsForModels && fieldDetails.Description != "" { - description := fmt.Sprintf("// %s", fieldDetails.Description) - line = fmt.Sprintf("%s\n%s", description, line) + comment := wrapOnWordBoundary(fieldDetails.Description, 120, "//") + line = fmt.Sprintf("%s\n%s", comment, line) } return &line, nil From 906cf849f1d3aa7415060eae25836eb2265a647c Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 19 Aug 2024 00:35:01 +0100 Subject: [PATCH 081/117] tooling: split end-to-end tests by SDK for easier analysis --- .../unit-test-end-to-end-microsoft-graph.yaml | 35 +++++++++++++++++++ ...unit-test-end-to-end-resource-manager.yaml | 2 +- scripts/automation-generate-go-sdk.sh | 31 +++++++++------- 3 files changed, 55 insertions(+), 13 deletions(-) create mode 100644 .github/workflows/unit-test-end-to-end-microsoft-graph.yaml diff --git a/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml b/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml new file mode 100644 index 00000000000..cb4bf62c49e --- /dev/null +++ b/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml @@ -0,0 +1,35 @@ +--- +name: Verify All Microsoft Graph Services can be Imported and Generated +on: + pull_request: + types: ['opened', 'synchronize'] + paths: + - '.github/workflows/**' + - 'config/**' + +jobs: + test: + runs-on: custom-linux-xl + strategy: + fail-fast: true + steps: + - uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7 + with: + submodules: recursive + + - uses: actions/setup-go@0a12ed9d6a96ab950c8f026ed9f722fe0da7ef32 # v5.0.2 + with: + go-version-file: ./.go-version + + - name: "Build and Run importer-msgraph-metadata" + id: import-data + run: | + cd ./tools/importer-msgraph-metadata + make tools + make build + make import + + - name: "Run the Go SDK Generator" + run: | + # go go gadget generator + ./scripts/automation-generate-go-sdk.sh microsoft-graph diff --git a/.github/workflows/unit-test-end-to-end-resource-manager.yaml b/.github/workflows/unit-test-end-to-end-resource-manager.yaml index f9a08ef1dbb..a3feb831fe5 100644 --- a/.github/workflows/unit-test-end-to-end-resource-manager.yaml +++ b/.github/workflows/unit-test-end-to-end-resource-manager.yaml @@ -32,4 +32,4 @@ jobs: - name: "Run the Go SDK Generator" run: | # go go gadget generator - ./scripts/automation-generate-go-sdk.sh + ./scripts/automation-generate-go-sdk.sh resource-manager diff --git a/scripts/automation-generate-go-sdk.sh b/scripts/automation-generate-go-sdk.sh index 034d352e46c..f5e5d81b9fc 100755 --- a/scripts/automation-generate-go-sdk.sh +++ b/scripts/automation-generate-go-sdk.sh @@ -7,6 +7,19 @@ set -e DIR="$(cd "$(dirname "$0")" && pwd)/.." +sdkToGenerate="${1}" +if [[ "${sdkToGenerate}" == "" ]]; then + echo "must specify SDK to generate!" >&2 + echo "" >&2 + echo "supported values are:" >&2 + echo " microsoft-graph" >&2 + echo " resource-manager" >&2 + echo "" >&2 + echo "example usage:" >&2 + echo "${0} resource-manager" >&2 + exit 1 +fi + function buildAndInstallDependencies { echo "Outputting Go Version.." go version @@ -31,15 +44,9 @@ function runWrapper { local apiDefinitionsDirectory=$1 local outputDirectory=$2 - echo "Running Wrapper for Resource Manager.." - cd "${DIR}/tools/wrapper-automation" - ./wrapper-automation resource-manager go-sdk \ - --api-definitions-dir="../../$apiDefinitionsDirectory"\ - --output-dir="../../$outputDirectory" - - echo "Running Wrapper for Microsoft Graph.." + echo "Running Wrapper for ${sdkToGenerate}.." cd "${DIR}/tools/wrapper-automation" - ./wrapper-automation microsoft-graph go-sdk \ + ./wrapper-automation "${sdkToGenerate}" go-sdk \ --api-definitions-dir="../../$apiDefinitionsDirectory"\ --output-dir="../../$outputDirectory" @@ -61,11 +68,11 @@ function runWrapper { function runGoSDKUnitTests { local outputDirectory=$1 - cd "${DIR}" + echo "Running unit tests within the SDK codebase.." + cd "${outputDirectory}/${sdkToGenerate}" + go test -v ./... - echo "Running 'make test' within the SDK codebase.." - cd "${outputDirectory}" - make test + cd "${DIR}" } function prepareGoSdk { From 6f3f1a52fda068fd442e4d698574bb3ed9d1fc80 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 19 Aug 2024 02:22:37 +0100 Subject: [PATCH 082/117] generator-go-sdk: populate inherited model struct fields from all parents, only set ancestor-specific fields when implementing its interface --- .../generator-go-sdk/internal/cmd/generate.go | 2 +- .../internal/generator/templater_models.go | 38 ++++++++++++++----- 2 files changed, 29 insertions(+), 11 deletions(-) diff --git a/tools/generator-go-sdk/internal/cmd/generate.go b/tools/generator-go-sdk/internal/cmd/generate.go index 3fa2de78bec..c610f45a74d 100644 --- a/tools/generator-go-sdk/internal/cmd/generate.go +++ b/tools/generator-go-sdk/internal/cmd/generate.go @@ -93,7 +93,7 @@ func (g GenerateCommand) Run(args []string) int { f := flag.NewFlagSet("generator-go-sdk", flag.ExitOnError) f.StringVar(&input.apiServerEndpoint, "data-api", "http://localhost:8080", "-data-api=http://localhost:8080") f.StringVar(&input.outputDirectory, "output-dir", "", "-output-dir=../generated-sdk-dev") - f.StringVar(&serviceNames, "services", "", "A list of comma separated Service named from the Data API to import") + f.StringVar(&serviceNames, "services", "", "A list of comma separated Service named from the Data API to generate") if err := f.Parse(args); err != nil { log.Fatalf("parsing arguments: %+v", err) } diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 7ed1c799e71..66c89cdf5d5 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -112,21 +112,38 @@ func (c modelsTemplater) structCode(data GeneratorData) (*string, error) { } parentAssignmentInfo = fmt.Sprintf("var _ %[1]s = %[2]s{}", parentTypeName, structName) + parentFields := make(map[string]models.SDKField) + parent, ok := data.models[*c.model.ParentTypeName] if !ok { return nil, fmt.Errorf("couldn't find Parent Model %q for Model %q", *c.model.ParentTypeName, c.name) } + for fieldName, fieldDetails := range parent.Fields { + parentFields[fieldName] = fieldDetails + } + + // Also include fields from the grandparent model + // Related to: https://github.com/hashicorp/pandora/issues/1235 + if parentTypeName != *c.model.ParentTypeName { + grandParent, ok := data.models[parentTypeName] + if !ok { + return nil, fmt.Errorf("couldn't find [Grand]Parent Model %q for Model %q", parentTypeName, c.name) + } + for fieldName, fieldDetails := range grandParent.Fields { + parentFields[fieldName] = fieldDetails + } + } - parentFields := make([]string, 0) - for fieldName := range parent.Fields { - parentFields = append(parentFields, fieldName) + parentFieldNames := make([]string, 0, len(parentFields)) + for fieldName := range parentFields { + parentFieldNames = append(parentFieldNames, fieldName) } - sort.Strings(parentFields) + sort.Strings(parentFieldNames) - if len(parentFields) > 0 { + if len(parentFieldNames) > 0 { structLines = append(structLines, fmt.Sprintf("\n// Fields inherited from %s", *c.model.ParentTypeName)) - for _, fieldName := range parentFields { - fieldDetails := parent.Fields[fieldName] + for _, fieldName := range parentFieldNames { + fieldDetails := parentFields[fieldName] fieldTypeName := "FIXME" fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, data.commonTypesPackageName) if err != nil { @@ -393,7 +410,8 @@ func (c modelsTemplater) codeForParentStructFunctions(data GeneratorData) (*stri parentTypeName = *c.model.ParentTypeName } - parent, ok := data.models[*c.model.ParentTypeName] + // Intentionally only setting fields from the outermost parent model + parent, ok := data.models[parentTypeName] if !ok { return nil, fmt.Errorf("couldn't find Parent Model %q for Model %q", *c.model.ParentTypeName, c.name) } @@ -449,7 +467,7 @@ func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, e structName := c.name - // parent models get a {model}Parent struct + // parent models get a {model}Base struct if c.model.IsDiscriminatedParentType() { structName = fmt.Sprintf("%sBase", c.name) } @@ -623,7 +641,7 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { func (c modelsTemplater) codeForUnmarshalStructFunction(data GeneratorData) (*string, error) { structName := c.name - // parent models get a {model}Parent struct + // parent models get a {model}Base struct if c.model.IsDiscriminatedParentType() { structName = fmt.Sprintf("%sBase", c.name) } From 70799a56ed7554beb8dd40b3b3e17b6fb2ce7be2 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 19 Aug 2024 02:47:28 +0100 Subject: [PATCH 083/117] tooling: adjust workflow triggers to only run when necessary --- .../workflows/automation-msgraph-metadata-importer.yaml | 1 - .github/workflows/automation-regenerate-terraform.yaml | 3 ++- .github/workflows/automation-rest-api-specs-importer.yaml | 4 ++-- .github/workflows/automation-version-bumper.yaml | 4 ++-- .github/workflows/unit-test-data-api-differ.yaml | 2 +- .github/workflows/unit-test-data-api-v2.yaml | 2 +- .../workflows/unit-test-end-to-end-microsoft-graph.yaml | 6 ++++-- .../workflows/unit-test-end-to-end-resource-manager.yaml | 7 +++++-- .github/workflows/unit-test-generator-go-sdk.yaml | 2 +- .github/workflows/unit-test-generator-terraform.yaml | 2 +- .github/workflows/unit-test-rest-api-specs-importer.yaml | 5 +++-- 11 files changed, 22 insertions(+), 16 deletions(-) diff --git a/.github/workflows/automation-msgraph-metadata-importer.yaml b/.github/workflows/automation-msgraph-metadata-importer.yaml index db5d7d9556c..6a7bb38e658 100644 --- a/.github/workflows/automation-msgraph-metadata-importer.yaml +++ b/.github/workflows/automation-msgraph-metadata-importer.yaml @@ -5,7 +5,6 @@ on: branches: - fake_name_to_disable_trigger paths: - - '.github/workflows/**' - 'config/microsoft-graph.hcl' - 'submodules/msgraph-metadata' - 'tools/importer-msgraph/**' diff --git a/.github/workflows/automation-regenerate-terraform.yaml b/.github/workflows/automation-regenerate-terraform.yaml index 0366cc537a7..f99bc163cc4 100644 --- a/.github/workflows/automation-regenerate-terraform.yaml +++ b/.github/workflows/automation-regenerate-terraform.yaml @@ -5,7 +5,8 @@ on: branches: - main paths: - - 'api-definitions/**' + - 'api-definitions/handwritten-resource-manager/**' + - 'api-definitions/resource-manager/**' - 'tools/generator-terraform/**' workflow_dispatch: # for manual invocations diff --git a/.github/workflows/automation-rest-api-specs-importer.yaml b/.github/workflows/automation-rest-api-specs-importer.yaml index d06974640fc..46fbae977d8 100644 --- a/.github/workflows/automation-rest-api-specs-importer.yaml +++ b/.github/workflows/automation-rest-api-specs-importer.yaml @@ -4,8 +4,8 @@ on: branches: - main paths: - - '.github/workflows/**' - - 'config/**' + - 'config/resource-manager.hcl' + - 'config/resources/**' - 'submodules/rest-api-specs' - 'tools/importer-rest-api-specs/**' workflow_dispatch: # for manual invocations diff --git a/.github/workflows/automation-version-bumper.yaml b/.github/workflows/automation-version-bumper.yaml index 51d38f00e76..345c3a869fc 100644 --- a/.github/workflows/automation-version-bumper.yaml +++ b/.github/workflows/automation-version-bumper.yaml @@ -4,8 +4,8 @@ on: branches: - main paths: - - '.github/workflows/**' - - 'config/**' + - '.github/workflows/automation-version-bumper.yaml' + - 'config/resource-manager.hcl' - 'submodules/msgraph-metadata' - 'submodules/rest-api-specs' - 'tools/version-bumper/**' diff --git a/.github/workflows/unit-test-data-api-differ.yaml b/.github/workflows/unit-test-data-api-differ.yaml index ce22c4712b7..c213168afeb 100644 --- a/.github/workflows/unit-test-data-api-differ.yaml +++ b/.github/workflows/unit-test-data-api-differ.yaml @@ -4,7 +4,7 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' + - '.github/workflows/unit-test-data-api-differ.yaml' - 'tools/data-api-differ/**' jobs: diff --git a/.github/workflows/unit-test-data-api-v2.yaml b/.github/workflows/unit-test-data-api-v2.yaml index a9d2a2cfe86..fcdefcbd5ef 100644 --- a/.github/workflows/unit-test-data-api-v2.yaml +++ b/.github/workflows/unit-test-data-api-v2.yaml @@ -4,7 +4,7 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' + - '.github/workflows/unit-test-data-api-v2.yaml' - 'api-definitions/**' - 'tools/data-api/**' diff --git a/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml b/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml index cb4bf62c49e..7c899430fbf 100644 --- a/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml +++ b/.github/workflows/unit-test-end-to-end-microsoft-graph.yaml @@ -4,8 +4,10 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' - - 'config/**' + - '.github/workflows/unit-test-end-to-end-microsoft-graph.yaml' + - 'config/microsoft-graph.hcl' + - 'tools/generator-go-sdk/**' + - 'tools/importer-msgraph-metadata/**' jobs: test: diff --git a/.github/workflows/unit-test-end-to-end-resource-manager.yaml b/.github/workflows/unit-test-end-to-end-resource-manager.yaml index a3feb831fe5..472abc29922 100644 --- a/.github/workflows/unit-test-end-to-end-resource-manager.yaml +++ b/.github/workflows/unit-test-end-to-end-resource-manager.yaml @@ -4,8 +4,11 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' - - 'config/**' + - '.github/workflows/unit-test-end-to-end-resource-manager.yaml' + - 'config/resource-manager.hcl' + - 'config/resources/**' + - 'tools/generator-go-sdk/**' + - 'tools/importer-rest-api-specs/**' jobs: test: diff --git a/.github/workflows/unit-test-generator-go-sdk.yaml b/.github/workflows/unit-test-generator-go-sdk.yaml index 71de572ab78..3e2026bc309 100644 --- a/.github/workflows/unit-test-generator-go-sdk.yaml +++ b/.github/workflows/unit-test-generator-go-sdk.yaml @@ -4,7 +4,7 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' + - '.github/workflows/unit-test-generator-go-sdk.yaml' - 'tools/generator-go-sdk/**' jobs: diff --git a/.github/workflows/unit-test-generator-terraform.yaml b/.github/workflows/unit-test-generator-terraform.yaml index f503d3a5101..c2ec94d3f3b 100644 --- a/.github/workflows/unit-test-generator-terraform.yaml +++ b/.github/workflows/unit-test-generator-terraform.yaml @@ -4,7 +4,7 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' + - '.github/workflows/unit-test-generator-terraform.yaml' - 'tools/generator-terraform/**' jobs: diff --git a/.github/workflows/unit-test-rest-api-specs-importer.yaml b/.github/workflows/unit-test-rest-api-specs-importer.yaml index 9ef598d82e9..af7b97fa43a 100644 --- a/.github/workflows/unit-test-rest-api-specs-importer.yaml +++ b/.github/workflows/unit-test-rest-api-specs-importer.yaml @@ -4,8 +4,9 @@ on: pull_request: types: ['opened', 'synchronize'] paths: - - '.github/workflows/**' - - 'config/**' + - '.github/workflows/unit-test-rest-api-specs-importer.yaml' + - 'config/resource-manager.hcl' + - 'config/resources/**' - 'submodules/rest-api-specs' - 'tools/importer-rest-api-specs/**' From b53beb0aca812b1c3b2a50d5b4fc1da8fdfce66f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 19 Aug 2024 02:57:41 +0100 Subject: [PATCH 084/117] tooling: run `go mod tidy` prior to running unit tests after generation --- scripts/automation-generate-go-sdk.sh | 1 + 1 file changed, 1 insertion(+) diff --git a/scripts/automation-generate-go-sdk.sh b/scripts/automation-generate-go-sdk.sh index f5e5d81b9fc..aab7b2b940c 100755 --- a/scripts/automation-generate-go-sdk.sh +++ b/scripts/automation-generate-go-sdk.sh @@ -70,6 +70,7 @@ function runGoSDKUnitTests { echo "Running unit tests within the SDK codebase.." cd "${outputDirectory}/${sdkToGenerate}" + go mod tidy go test -v ./... cd "${DIR}" From 1c71b9c21dc0d27869ec962e09592423c66cf13c Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 19 Aug 2024 16:00:20 +0100 Subject: [PATCH 085/117] importer-msgraph-metadata: support Resource IDs in data workarounds --- .../components/parser/resourceids.go | 6 ++--- .../workarounds/workaround_application.go | 2 +- .../workaround_conditionalaccesspolicy.go | 2 +- .../workarounds/workaround_directoryobject.go | 2 +- .../workarounds/workaround_iprange.go | 2 +- .../workarounds/workaround_namedlocation.go | 2 +- .../components/workarounds/workarounds.go | 7 +++--- .../internal/pipeline/importer.go | 23 ++++++++----------- ...dels.go => task_translate_common_types.go} | 0 9 files changed, 22 insertions(+), 24 deletions(-) rename tools/importer-msgraph-metadata/internal/pipeline/{task_translate_models.go => task_translate_common_types.go} (100%) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 72616083445..1da8aec926a 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -21,7 +21,7 @@ const ( ResourceIdSuffix = "Id" ) -type ResourceIds []*ResourceId +type ResourceIds map[string]*ResourceId type ResourceIdMatch struct { Id *ResourceId @@ -436,7 +436,7 @@ func NewResourceId(path string, tags []string) (id ResourceId) { } func ParseResourceIDs(paths openapi3.Paths, serviceName *string) (resourceIds ResourceIds, err error) { - resourceIds = make(ResourceIds, 0) + resourceIds = make(ResourceIds) for path, item := range paths { operations := item.Operations() operationTags := make([]string, 0) @@ -490,7 +490,7 @@ func ParseResourceIDs(paths openapi3.Paths, serviceName *string) (resourceIds Re id.Service = normalize.CleanName(*serviceName) } - resourceIds = append(resourceIds, &id) + resourceIds[resourceIdName] = &id } } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go index 51933488f50..fc7f224f577 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go @@ -23,7 +23,7 @@ func (workaroundApplication) Name() string { return "Application / missing fields in beta" } -func (workaroundApplication) Process(apiVersion string, models parser.Models, constants parser.Constants) error { +func (workaroundApplication) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { if apiVersion != versions.ApiVersionBeta { return nil } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index 45e9d556814..1f275279f5b 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -19,7 +19,7 @@ func (workaroundConditionalAccessPolicy) Name() string { return "Conditional Access Policy / fixing missing fields and types" } -func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parser.Models, constants parser.Constants) error { +func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { model, ok := models["ConditionalAccessPolicy"] if !ok { return fmt.Errorf("`ConditionalAccessPolicy` model not found") diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go index 7537117688b..89e4de2b7f8 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go @@ -20,7 +20,7 @@ func (workaroundDirectoryObject) Name() string { return "Directory Object / discrimination" } -func (workaroundDirectoryObject) Process(apiVersion string, models parser.Models, constants parser.Constants) error { +func (workaroundDirectoryObject) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { model, ok := models["DirectoryObject"] if !ok { return fmt.Errorf("`DirectoryObject` model not found") diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go index 73bd11dd025..fca8b6c2d43 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go @@ -19,7 +19,7 @@ func (workaroundIPRange) Name() string { return "IPRange / discrimination" } -func (workaroundIPRange) Process(apiVersion string, models parser.Models, constants parser.Constants) error { +func (workaroundIPRange) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { // microsoft.graph.ipRange model is empty and is omitted by the OpenAPI parser models["IPRange"] = &parser.Model{ Fields: map[string]*parser.ModelField{ diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go index 540101b3941..c9348a669e3 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go @@ -19,7 +19,7 @@ func (workaroundNamedLocation) Name() string { return "NamedLocation / discrimination" } -func (workaroundNamedLocation) Process(apiVersion string, models parser.Models, constants parser.Constants) error { +func (workaroundNamedLocation) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { model, ok := models["NamedLocation"] if !ok { return fmt.Errorf("`NamedLocation` model not found") diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index f78f8c627a1..05c7772c34f 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -23,14 +23,15 @@ type workaround interface { Name() string // Process takes the apiDefinition and applies the Workaround to this AzureApiDefinition - Process(string, parser.Models, parser.Constants) error + Process(string, parser.Models, parser.Constants, parser.ResourceIds) error } -func ApplyWorkarounds(apiVersion string, models parser.Models, constants parser.Constants) error { +// ApplyWorkarounds invokes the specified workarounds for models, constants and resource +func ApplyWorkarounds(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { logging.Tracef("Processing Data Workarounds..") for _, fix := range workarounds { logging.Tracef("Applying Data Workaround %q to Model %q", fix.Name()) - if err := fix.Process(apiVersion, models, constants); err != nil { + if err := fix.Process(apiVersion, models, constants, resourceIds); err != nil { return fmt.Errorf("applying Data Workaround %q: %v", fix.Name(), err) } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go index 14dcf54e6cc..e1180d7bc79 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/importer.go @@ -63,19 +63,19 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha return err } - logging.Infof("Applying workarounds for invalid model definitions..") - if err = workarounds.ApplyWorkarounds(p.apiVersion, p.models, p.constants); err != nil { + logging.Infof("Parsing resource IDs...") + p.resourceIds, err = parser.ParseResourceIDs(p.spec.Paths, nil) + if err != nil { return err } - logging.Infof("Cleaning up models...") - if err = p.cleanupModels(); err != nil { + logging.Infof("Applying workarounds for invalid type definitions..") + if err = workarounds.ApplyWorkarounds(p.apiVersion, p.models, p.constants, p.resourceIds); err != nil { return err } - logging.Infof("Parsing resource IDs...") - p.resourceIds, err = parser.ParseResourceIDs(p.spec.Paths, nil) - if err != nil { + logging.Infof("Cleaning up models...") + if err = p.cleanupModels(); err != nil { return err } @@ -126,20 +126,17 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } // Determine which resource IDs were actually used in resources - usedResourceIds := make(map[string]parser.ResourceId) + usedResourceIds := make(parser.ResourceIds) for _, resources := range p.resources { for _, resource := range resources { for _, operation := range resource.Operations { if operation.ResourceId != nil { - usedResourceIds[operation.ResourceId.Name] = *operation.ResourceId + usedResourceIds[operation.ResourceId.Name] = operation.ResourceId } } } } - p.resourceIds = make(parser.ResourceIds, 0, len(usedResourceIds)) - for _, resourceId := range usedResourceIds { - p.resourceIds = append(p.resourceIds, &resourceId) - } + p.resourceIds = usedResourceIds commonTypesForApiVersion, err := p.translateCommonTypesToDataApiSdkTypes() if err != nil { diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_models.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_common_types.go similarity index 100% rename from tools/importer-msgraph-metadata/internal/pipeline/task_translate_models.go rename to tools/importer-msgraph-metadata/internal/pipeline/task_translate_common_types.go From 32c9c9982442c228f4261d49a91040ccf5b218c5 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 19 Aug 2024 16:00:39 +0100 Subject: [PATCH 086/117] importer-msgraph-metadata: add workaround for invalid resource IDs now that these are common types --- ...rkaround_repeating_resource_id_segments.go | 73 +++++++++++++++++++ .../components/workarounds/workarounds.go | 1 + 2 files changed, 74 insertions(+) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go new file mode 100644 index 00000000000..c3d2aba5c8d --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go @@ -0,0 +1,73 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "strings" + + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ workaround = workaroundRepeatingResourceIdSegments{} + +// workaroundRepeatingResourceIdSegments removes incompatible resource IDs due to repeating segments which are not supported at this time. +type workaroundRepeatingResourceIdSegments struct{} + +func (workaroundRepeatingResourceIdSegments) Name() string { + return "Resource IDs with repeating segments / remove incompatible resource IDs" +} + +func (workaroundRepeatingResourceIdSegments) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { + for resourceIdName := range resourceIds { + var invalid bool + + // Repeating segments, which are not supported + if strings.Contains(resourceIdName, "SiteIdSiteId") { + invalid = true + } + + // GroupSiteTermStore resources have repeating ID segments which are not supported at this time + if strings.Contains(resourceIdName, "TermStore") { + invalid = true + } + + // Onenote resources have repeating ID segments which are not supported at this time + if strings.Contains(resourceIdName, "Onenote") { + invalid = true + } + + // These contain IDs with repeating segments, which are not supported at this time + prefixes := []string{ + "IdentityGovernanceEntitlementManagementAccessPackageIdResourceRoleScopeIdRoleResourceScopeIdResourceRoleId", + "IdentityGovernanceEntitlementManagementAccessPackageIdResourceRoleScopeIdScopeResourceRoleIdResourceScopeId", + "IdentityGovernanceEntitlementManagementCatalogIdResourceRoleIdResourceScopeIdResourceRoleId", + "IdentityGovernanceEntitlementManagementCatalogIdResourceScopeIdResourceRoleIdResourceScopeId", + "IdentityGovernanceEntitlementManagementResourceRequestIdCatalogResourceRoleIdResourceScopeIdResourceRoleId", + "IdentityGovernanceEntitlementManagementResourceRequestIdCatalogResourceScopeIdResourceRoleIdResourceScopeId", + "IdentityGovernanceEntitlementManagementResourceRequestIdResourceRoleIdResourceScopeId", + "IdentityGovernanceEntitlementManagementResourceRequestIdResourceScopeIdResourceRoleId", + "IdentityGovernanceEntitlementManagementResourceRoleScopeIdRoleResourceScopeIdResourceRoleId", + "IdentityGovernanceEntitlementManagementResourceRoleScopeIdScopeResourceRoleIdResourceScopeId", + "MePendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionId", + "MePendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionIdInsightId", + "MePendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionId", + "MePendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionIdInsightId", + "UserIdPendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionId", + "UserIdPendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionIdInsightId", + "UserIdPendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionId", + "UserIdPendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionIdInsightId", + } + for _, prefix := range prefixes { + if strings.HasPrefix(resourceIdName, prefix) { + invalid = true + } + } + + if invalid { + delete(resourceIds, resourceIdName) + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index 05c7772c34f..e44c3345aac 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -16,6 +16,7 @@ var workarounds = []workaround{ workaroundDirectoryObject{}, workaroundNamedLocation{}, workaroundIPRange{}, + workaroundRepeatingResourceIdSegments{}, } type workaround interface { From 127b0cba285dd000d18da752ae699d0fc5624e2d Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 22 Aug 2024 04:21:34 +0100 Subject: [PATCH 087/117] generator-go-sdk: linting for method comments --- .../internal/generator/templater_methods.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index a55c14c7086..0972c76a5ec 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -165,7 +165,7 @@ func (c methodsPandoraTemplater) immediateOperationTemplate(data GeneratorData) comment := c.operationName if c.operation.Description != "" { - comment += fmt.Sprintf(": %s", c.operation.Description) + comment += fmt.Sprintf(" - %s", c.operation.Description) } else { comment += " ..." } @@ -236,7 +236,7 @@ func (c methodsPandoraTemplater) longRunningOperationTemplate(data GeneratorData comment := c.operationName if c.operation.Description != "" { - comment += fmt.Sprintf(": %s", c.operation.Description) + comment += fmt.Sprintf(" - %s", c.operation.Description) } else { comment += " ..." } @@ -331,7 +331,7 @@ func (c methodsPandoraTemplater) listOperationTemplate(data GeneratorData) (*str comment := c.operationName if c.operation.Description != "" { - comment += fmt.Sprintf(": %s", c.operation.Description) + comment += fmt.Sprintf(" - %s", c.operation.Description) } else { comment += " ..." } From 0b2f7cd8e528f2975a52913915e80100ec768688 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 22 Aug 2024 04:23:01 +0100 Subject: [PATCH 088/117] importer-msgraph-metadata: refactor model parser, various improvements - Rewrite the model parsing functions to be more selective and opinionated - Don't flatten the schema as it loses all nuance - Drop the crutch of recursion to traverse the model tree - Fully support model inheritence with SDK compatibility in mind - Process constants as a first class construct - Use the model names from the API as much as possible - Support constant and model references in operations - Automatically infer all discriminated models - Automatically redact improper URLs that are not supported as resource IDs - Various bug fixes with type translation, particularly with arrays - Handle different content types in responses --- .../components/normalize/normalize.go | 6 +- .../components/parser/resources.go | 41 +- .../components/parser/types.go | 783 ++++++------------ .../workarounds/workaround_application.go | 12 +- .../workaround_conditionalaccesspolicy.go | 90 +- .../workarounds/workaround_directoryobject.go | 115 --- .../workarounds/workaround_iprange.go | 97 --- .../workarounds/workaround_namedlocation.go | 69 -- .../workarounds/workaround_odata_bind.go | 61 ++ ...rkaround_repeating_resource_id_segments.go | 48 +- .../components/workarounds/workarounds.go | 11 +- .../internal/pipeline/importer.go | 7 +- .../internal/pipeline/task_cleanup_models.go | 31 - .../internal/pipeline/task_parse_resources.go | 314 ++++--- .../pipeline/task_translate_common_types.go | 14 +- .../pipeline/task_translate_service.go | 105 +-- 16 files changed, 680 insertions(+), 1124 deletions(-) delete mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go delete mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go delete mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go delete mode 100644 tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/normalize.go index 38936e74094..b9f069115ff 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize.go @@ -40,11 +40,15 @@ func Pluralize(name string) string { } func CleanName(name string) string { + if name == "" { + return name + } + // Trim a "microsoft.graph" prefix from type names name = strings.TrimPrefix(name, "microsoft.graph") // Replace all periods with spaces to allow for title casing - name = strings.ReplaceAll(name, ".", " ") + name = strings.TrimSpace(strings.ReplaceAll(name, ".", " ")) // Convert name to title case name = cases.Title(language.AmericanEnglish, cases.NoLower).String(name) diff --git a/tools/importer-msgraph-metadata/components/parser/resources.go b/tools/importer-msgraph-metadata/components/parser/resources.go index 11dc30a6138..1cfe9da5863 100644 --- a/tools/importer-msgraph-metadata/components/parser/resources.go +++ b/tools/importer-msgraph-metadata/components/parser/resources.go @@ -20,19 +20,20 @@ type Resource struct { } type Operation struct { - Name string - Description string - Type OperationType - Method string - ResourceId *ResourceId - UriSuffix *string - RequestModel *string - RequestHeaders *Headers - RequestParams *Params - RequestType *DataType - Responses Responses - PaginationField *string - Tags []string + Name string + Description string + Type OperationType + Method string + ResourceId *ResourceId + UriSuffix *string + RequestContentType *string + RequestModel *string + RequestHeaders *Headers + RequestParams *Params + RequestType *DataType + Responses Responses + PaginationField *string + Tags []string } type Header struct { @@ -91,18 +92,20 @@ func (p Param) DataApiSdkObjectDefinition() (*sdkModels.SDKOperationOptionObject type Params []Param type Response struct { - Status int - ContentType *string - ModelName *string - Type *DataType + Status int + ContentType *string + ReferenceName *string + Type *DataType + Model *Model + Constant *Constant } type Responses []Response func (rs Responses) FindModelName() *string { for _, r := range rs { - if r.ModelName != nil { - return r.ModelName + if r.ReferenceName != nil { + return r.ReferenceName } } return nil diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 05cf5c6e30e..023d2d58f9a 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -21,7 +21,7 @@ import ( SchemaRef is a struct{Ref, Value} where Ref is a string, Value is a *Schema The Ref string (after trimming) indicates a Schemas map key to follow/inherit Schema has Properties which is a nested Schemas - Schema has AllOf and/or AnyOf which are SchemaRefs + Schema has AllOf, AnyOf or OneOf which are SchemaRefs SchemaRefs is a []*SchemaRef Schemas is a model SchemaRefs, SchemaRef lead to a Schema or other another SchemaRef @@ -30,98 +30,78 @@ import ( const RefPrefix = "#/components/schemas/" +func TrimRefPrefix(ref string) string { + if strings.HasPrefix(ref, RefPrefix) { + return ref[len(RefPrefix):] + } + return ref +} + type Constants map[string]*Constant type Models map[string]*Model -// Found returns true when the provided modelName was found in the Models map -func (m Models) Found(modelName string) bool { - // Safety check, don't allow an empty model name - if modelName == "" { +// Found returns true when the provided schemaName was found in the Constants map +func (c Constants) Found(schemaName string) bool { + // Safety check, don't allow an empty constant name + if schemaName == "" { return false } - if model, ok := m[modelName]; ok && model != nil { + if constant, ok := c[schemaName]; ok && constant != nil { return true } return false } -// MergeDependants inspects the named model in m, then traverses allModels and appends any dependant models to m, recursively -func (m Models) MergeDependants(allModels Models, modelName string, includeCommon bool) error { - if !allModels.Found(modelName) { - return fmt.Errorf("model not found: %q", modelName) - } - - if _, ok := m[modelName]; !ok { - if !includeCommon && allModels[modelName].Common { - return nil - } - m[modelName] = allModels[modelName] +// Found returns true when the provided schemaName was found in the Models map +func (m Models) Found(schemaName string) bool { + // Safety check, don't allow an empty model name + if schemaName == "" { + return false } - for _, field := range allModels[modelName].Fields { - if field.ModelName == nil { - continue - } - - if _, ok := m[*field.ModelName]; ok { - continue - } - - if !allModels.Found(*field.ModelName) { - return fmt.Errorf("dependant model not found: %q", modelName) - } - - if err := m.MergeDependants(allModels, *field.ModelName, includeCommon); err != nil { - return err - } + if model, ok := m[schemaName]; ok && model != nil { + return true } - return nil -} - -func (m Models) Merge(m2 Models) { - if m2 == nil { - return - } - for modelName, model := range m2 { - m[modelName] = model - } + return false } type Constant struct { - Enum []string - Type *DataType + Name string + Common bool + Enum []string + Type *DataType } type Model struct { - Fields map[string]*ModelField - Common bool - Prefix string - TypeField *string - TypeValue *string - ParentModelName *string + Name string + Fields map[string]*ModelField + Common bool + TypeField *string + TypeValue *string + ParentModel *string } -func (m *Model) IsValid() bool { - // Several constants are presented as models, these have no fields and are of no use - if m == nil || len(m.Fields) == 0 { - return false +func (m *Model) AppendDefaultFields() { + for jsonName, fieldDetails := range defaultModelFields() { + if _, ok := m.Fields[jsonName]; !ok { + m.Fields[jsonName] = fieldDetails + } } - return true } -func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { +func (m *Model) DataApiSdkModel(models Models, constants Constants) (*sdkModels.SDKModel, error) { sdkFields := make(map[string]sdkModels.SDKField) - for fieldName, field := range m.Fields { - objectDefinition, err := field.DataApiSdkObjectDefinition(models) + for jsonName, field := range m.Fields { + objectDefinition, err := field.DataApiSdkObjectDefinition(models, constants) if err != nil { return nil, err } if objectDefinition == nil { - logging.Warnf("Could not determine SDKObjectDefinition for field %q, skipping", fieldName) + logging.Warnf("Could not determine SDKObjectDefinition for field %q, skipping", jsonName) continue } @@ -132,12 +112,13 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { required = true } - sdkFields[fieldName] = sdkModels.SDKField{ + sdkFields[field.Name] = sdkModels.SDKField{ + DateFormat: nil, + Description: field.Description, + JsonName: jsonName, + ObjectDefinition: *objectDefinition, + ContainsDiscriminatedValue: field.DiscriminatedValue, - DateFormat: nil, - Description: field.Description, - JsonName: field.JsonField, - ObjectDefinition: *objectDefinition, Optional: optional, ReadOnly: field.ReadOnly, @@ -150,16 +131,42 @@ func (m *Model) DataApiSdkModel(models Models) (*sdkModels.SDKModel, error) { return nil, nil } + var parentTypeName *string + if m.ParentModel != nil { + parentTypeName = pointer.To(normalize.CleanName(*m.ParentModel)) + } + return &sdkModels.SDKModel{ + Fields: sdkFields, + DiscriminatedValue: m.TypeValue, FieldNameContainingDiscriminatedValue: m.TypeField, - Fields: sdkFields, - ParentTypeName: m.ParentModelName, + ParentTypeName: parentTypeName, }, nil } +func defaultModelFields() map[string]*ModelField { + // Add an explicit ODataId and ODataType field to each model, since it is inconsistently defined in the API specs. + // This won't be valid for every model, but it's impossible to tell which models support them, and it's effectively + // harmless to leave these in so long as they have the `omitempty` struct tag in the generated SDK. + return map[string]*ModelField{ + "@odata.id": { + Name: "ODataId", + Description: "The OData ID of this entity", + Type: pointer.To(DataTypeString), + Default: "", + }, + "@odata.type": { + Name: "ODataType", + Description: "The OData Type of this entity", + Type: pointer.To(DataTypeString), + Default: "", + }, + } +} + type ModelField struct { - Title string + Name string Type *DataType Description string Default interface{} @@ -170,69 +177,69 @@ type ModelField struct { AllowEmptyValue bool DiscriminatedValue bool ItemType *DataType - ConstantName *string - ModelName *string - JsonField string + ReferenceName *string } -func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObjectDefinition, error) { +func (f ModelField) DataApiSdkObjectDefinition(models Models, constants Constants) (*sdkModels.SDKObjectDefinition, error) { if f.Type == nil { - return nil, fmt.Errorf("field %q has no Type", f.Title) + return nil, fmt.Errorf("field %q has no Type", f.Name) } switch *f.Type { - case DataTypeModel: - if f.ModelName == nil { - return nil, fmt.Errorf("field type Model encountered without model name") + case DataTypeReference: + if f.ReferenceName == nil { + return nil, fmt.Errorf("field type Reference encountered without ReferenceName") } - if !models.Found(*f.ModelName) { - return nil, fmt.Errorf("field type Model encountered with unknown referenced model") + if models.Found(*f.ReferenceName) { + return &sdkModels.SDKObjectDefinition{ + Nullable: f.Nullable, + ReferenceName: pointer.To(normalize.CleanName(*f.ReferenceName)), + ReferenceNameIsCommonType: pointer.To(models[*f.ReferenceName].Common), + Type: sdkModels.ReferenceSDKObjectDefinitionType, + }, nil } - if !models[*f.ModelName].IsValid() { - logging.Warnf("Skipping field %q with type Model as the referenced model %q is invalid", f.Title, *f.ModelName) + if constants.Found(*f.ReferenceName) { + return &sdkModels.SDKObjectDefinition{ + Nullable: f.Nullable, + ReferenceName: pointer.To(normalize.CleanName(*f.ReferenceName)), + ReferenceNameIsCommonType: pointer.To(constants[*f.ReferenceName].Common), + Type: sdkModels.ReferenceSDKObjectDefinitionType, + }, nil } - return &sdkModels.SDKObjectDefinition{ - Nullable: f.Nullable, - ReferenceName: f.ModelName, - ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), - Type: sdkModels.ReferenceSDKObjectDefinitionType, - }, nil + return nil, fmt.Errorf("field type Reference encountered with unknown referenced model/constant") case DataTypeArray: - if f.ModelName != nil { - if !models.Found(*f.ModelName) { - return nil, fmt.Errorf("field type Array[Model] encountered with unknown referenced model") + if f.ReferenceName != nil { + if models.Found(*f.ReferenceName) { + return &sdkModels.SDKObjectDefinition{ + NestedItem: &sdkModels.SDKObjectDefinition{ + ReferenceName: pointer.To(normalize.CleanName(*f.ReferenceName)), + ReferenceNameIsCommonType: pointer.To(models[*f.ReferenceName].Common), + Type: sdkModels.ReferenceSDKObjectDefinitionType, + }, + Nullable: f.Nullable, + ReferenceName: nil, + Type: sdkModels.ListSDKObjectDefinitionType, + }, nil } - if !models[*f.ModelName].IsValid() { - logging.Warnf("Skipping field %q with type Array[Model] as the referenced model %q is invalid", f.Title, *f.ModelName) + if constants.Found(*f.ReferenceName) { + return &sdkModels.SDKObjectDefinition{ + NestedItem: &sdkModels.SDKObjectDefinition{ + ReferenceName: pointer.To(normalize.CleanName(*f.ReferenceName)), + ReferenceNameIsCommonType: pointer.To(constants[*f.ReferenceName].Common), + Type: sdkModels.ReferenceSDKObjectDefinitionType, + }, + Nullable: f.Nullable, + ReferenceName: nil, + Type: sdkModels.ListSDKObjectDefinitionType, + }, nil } - return &sdkModels.SDKObjectDefinition{ - NestedItem: &sdkModels.SDKObjectDefinition{ - ReferenceName: f.ModelName, - ReferenceNameIsCommonType: pointer.To(models[*f.ModelName].Common), - Type: sdkModels.ReferenceSDKObjectDefinitionType, - }, - Nullable: f.Nullable, - ReferenceName: nil, - Type: sdkModels.ListSDKObjectDefinitionType, - }, nil - } - - if f.ConstantName != nil { - // TODO validate constant exists - return &sdkModels.SDKObjectDefinition{ - NestedItem: &sdkModels.SDKObjectDefinition{ - ReferenceName: f.ConstantName, - Type: sdkModels.ReferenceSDKObjectDefinitionType, - }, - Nullable: f.Nullable, - Type: sdkModels.ListSDKObjectDefinitionType, - }, nil + return nil, fmt.Errorf("field type Array[Reference] encountered with unknown referenced model/constant") } if f.ItemType != nil { @@ -255,12 +262,8 @@ func (f ModelField) DataApiSdkObjectDefinition(models Models) (*sdkModels.SDKObj }, nil } - if f.ConstantName != nil { - return &sdkModels.SDKObjectDefinition{ - Nullable: f.Nullable, - ReferenceName: f.ConstantName, - Type: sdkModels.ReferenceSDKObjectDefinitionType, - }, nil + if f.ReferenceName != nil { + return nil, fmt.Errorf("field that is not a Reference or Array[Reference] encountered with ReferenceName") } return &sdkModels.SDKObjectDefinition{ @@ -291,7 +294,7 @@ const ( DataTypeIntegerUnsigned32 DataTypeIntegerUnsigned64 DataTypeIntegerUnsigned8 - DataTypeModel + DataTypeReference DataTypeString DataTypeTime DataTypeUuid @@ -344,9 +347,14 @@ func (ft DataType) DataApiSdkOperationOptionObjectDefinitionType() sdkModels.SDK } // FieldType parses the schemaType and schemaFormat from the OpenAPI spec for a given field, and returns the appropriate DataType -func FieldType(schemaType, schemaFormat string, hasModel bool) *DataType { +func FieldType(schemaType, schemaFormat string, hasReference bool) *DataType { var ret DataType + if hasReference { + ret = DataTypeReference + return &ret + } + switch strings.ToLower(schemaFormat) { case "int64": ret = DataTypeInteger64 @@ -403,461 +411,212 @@ func FieldType(schemaType, schemaFormat string, hasModel bool) *DataType { return &ret } - if hasModel { - ret = DataTypeModel - return &ret - } - return nil } -func Common(schemas openapi3.Schemas) (models Models, constants Constants, err error) { - models = make(Models) - constants = make(Constants) - for modelName, schemaRef := range schemas { - name := normalize.CleanName(modelName) - if schemaRef.Value != nil { - var f *flattenedSchema - if f, _ = FlattenSchemaRef(schemaRef, nil); f != nil { - models, constants = Schemas(*f, name, models, constants, true) - } - } - } +func ModelsAndConstants(schemas openapi3.Schemas) (Models, Constants, error) { + models := make(Models) + constants := make(Constants) - return -} - -type flattenedSchema struct { - Schemas openapi3.Schemas - Prefix string - Title string - Type string - Format string - Enum []interface{} -} - -// FlattenSchemaRef attempts to recursively parse and flatten the provided *openapi3.Schema and returns a flattenedSchema -// which is much more convenient to inspect for types. The returned map[string]bool is used when recursing to track -// Refs which have been observed in order to avoid infinite recursion, and is usually not interesting to the caller. -func FlattenSchemaRef(schemaRef *openapi3.SchemaRef, seenRefs map[string]bool) (*flattenedSchema, map[string]bool) { - if seenRefs == nil { - seenRefs = make(map[string]bool) - } - - if schemaRef.Value == nil { - return nil, seenRefs - } - - prefix := "" - title := "" - titleFromRef := false - if strings.HasPrefix(schemaRef.Ref, RefPrefix) { - ref := schemaRef.Ref[len(RefPrefix):] - if i := strings.LastIndex(ref, "."); i > 0 { - prefix = normalize.CleanName(ref[0:i]) + for schemaName, schemaRef := range schemas { + if models.Found(schemaName) { + return nil, nil, fmt.Errorf("model %q already encountered", schemaName) } - title = normalize.CleanName(ref) - titleFromRef = true - } - schema := schemaRef.Value - schemas := make(openapi3.Schemas, 0) - typ := "" - format := "" - enum := make([]interface{}, 0) - - if r := schema.Items; r != nil { - if r.Ref != "" { - for s := range seenRefs { - if s == r.Ref { - continue - } - } - seenRefs[r.Ref] = true - if title == "" && strings.HasPrefix(r.Ref, RefPrefix) { - ref := r.Ref[len(RefPrefix):] - if i := strings.LastIndex(ref, "."); i > 0 { - prefix = normalize.CleanName(ref[0:i]) - } - title = normalize.CleanName(ref) - } + if constants.Found(schemaName) { + return nil, nil, fmt.Errorf("constant %q already encountered", schemaName) } - if r.Value != nil { - var result *flattenedSchema - result, seenRefs = FlattenSchemaRef(r, seenRefs) - if title == "" && result.Title != "" { - title = normalize.CleanName(result.Title) - } - if result.Type != "" { - typ = result.Type - } - if result.Format != "" { - format = result.Format - } - if len(result.Enum) > 0 { - enum = result.Enum - } - for k, v := range result.Schemas { - schemas[k] = v - } - } - } else { - if schema.AllOf != nil { - for _, r := range schema.AllOf { - if r.Ref != "" { - for s := range seenRefs { - if s == r.Ref { - continue - } - } - seenRefs[r.Ref] = true - if !titleFromRef && strings.HasPrefix(r.Ref, RefPrefix) { - ref := r.Ref[len(RefPrefix):] - if i := strings.LastIndex(ref, "."); i > 0 { - prefix = normalize.CleanName(ref[0:i]) - } - title = normalize.CleanName(ref) - titleFromRef = true - } - } - - if r.Value != nil { - var result *flattenedSchema - result, seenRefs = FlattenSchemaRef(r, seenRefs) - if !titleFromRef && result.Title != "" { - title = normalize.CleanName(result.Title) - } - if typ == "" && result.Type != "" { - typ = result.Type - } - if format == "" && result.Format != "" { - format = result.Format - } - if len(result.Enum) > 0 { - enum = result.Enum - } - for k, v := range result.Schemas { - schemas[k] = v - } - } - } + model, constant, err := ModelOrConstant(schemaName, schemaRef, true) + if err != nil { + return nil, nil, err } - if schema.AnyOf != nil { - for _, r := range schema.AnyOf { - if r.Ref != "" { - for s := range seenRefs { - if s == r.Ref { - continue - } - } - seenRefs[r.Ref] = true - if !titleFromRef && strings.HasPrefix(r.Ref, RefPrefix) { - ref := r.Ref[len(RefPrefix):] - if i := strings.LastIndex(ref, "."); i > 0 { - prefix = normalize.CleanName(ref[0:i]) - } - title = normalize.CleanName(ref) - titleFromRef = true - } - } - - if r.Value != nil { - var result *flattenedSchema - result, seenRefs = FlattenSchemaRef(r, seenRefs) - if !titleFromRef && result.Title != "" { - title = normalize.CleanName(result.Title) - } - if typ == "" && result.Type != "" { - typ = result.Type - } - if format == "" && result.Format != "" { - format = result.Format - } - if len(result.Enum) > 0 { - enum = result.Enum - } - for k, v := range result.Schemas { - schemas[k] = v - } - } - } + if model != nil { + models[schemaName] = model + models[schemaName].AppendDefaultFields() } - if schema.OneOf != nil { - for _, r := range schema.OneOf { - if r.Ref != "" { - for s := range seenRefs { - if s == r.Ref { - continue - } - } - seenRefs[r.Ref] = true - if !titleFromRef && strings.HasPrefix(r.Ref, RefPrefix) { - ref := r.Ref[len(RefPrefix):] - if i := strings.LastIndex(ref, "."); i > 0 { - prefix = normalize.CleanName(ref[0:i]) - } - title = normalize.CleanName(ref) - titleFromRef = true - } - } - - if r.Value != nil { - var result *flattenedSchema - result, seenRefs = FlattenSchemaRef(r, seenRefs) - if !titleFromRef && result.Title != "" { - title = normalize.CleanName(result.Title) - } - if typ == "" && result.Type != "" { - typ = result.Type - } - if format == "" && result.Format != "" { - format = result.Format - } - if len(result.Enum) > 0 { - enum = result.Enum - } - for k, v := range result.Schemas { - schemas[k] = v - } - } - } + if constant != nil { + constants[schemaName] = constant } } - // TODO: may need to prefer innermost title - if schema.Title != "" { - title = normalize.CleanName(schema.Title) - } - - // prefer the innermost type - if typ == "" && schema.Type != "" { - typ = schema.Type - } - - // prefer the innermost format - if format == "" && schema.Format != "" { - format = schema.Format - } - - if len(schema.Enum) > 0 { - enum = schema.Enum - } + // Now iterate models and populate discriminated children + for schemaName, model := range models { + if model.ParentModel == nil { + continue + } - if schema.Properties != nil { - for k, v := range schema.Properties { - schemas[k] = v + parentModel, ok := models[*model.ParentModel] + if !ok { + return nil, nil, fmt.Errorf("parent model %q was not found for model %q", *model.ParentModel, schemaName) } - } - if len(schemas) == 0 { - schemas = nil + parentModel.Fields["@odata.type"].DiscriminatedValue = true + parentModel.TypeField = pointer.To("ODataType") + model.TypeField = pointer.To("ODataType") + model.TypeValue = pointer.To("#" + schemaName) } - return &flattenedSchema{ - Schemas: schemas, - Prefix: prefix, - Title: title, - Type: typ, - Format: format, - Enum: enum, - }, seenRefs + return models, constants, nil } -// Schemas inspects the provided flattenedSchema to parse out the fields for the provided modelName, optionally -// marking it as a common model. The provided Models (map[string]Model) is mutated to append the new model and its fields. -// Fields having the type of another model are parsed recursively to extract all known models that may not be directly -// referenced in the root schema. -func Schemas(input flattenedSchema, name string, models Models, constants Constants, common bool) (Models, Constants) { - if _, ok := models[name]; ok { - return models, constants +func ModelOrConstant(schemaName string, schemaRef *openapi3.SchemaRef, common bool) (*Model, *Constant, error) { + schema := schemaRef.Value + if schema == nil { + logging.Tracef("OpenAPI model %q has no schema, skipping", schemaName) + return nil, nil, nil } - // Check if this is a constant - if input.Schemas == nil && len(input.Enum) > 0 { + // First check if this is a constant + if schema.Enum != nil { constant := Constant{ - Enum: parseEnum(input.Enum), - Type: FieldType(input.Type, input.Format, false), + Name: normalize.CleanName(schemaName), + Common: common, + Enum: parseEnum(schema.Enum), + Type: FieldType(schema.Type, schema.Format, false), } - constants[name] = &constant - return models, constants - } - - // If there are no schemas, this is not a valid model and is likely a custom type which we don't currently use, e.g. ODataCountResponse - if input.Schemas == nil { - return models, constants + return nil, &constant, nil } - // Add an explicit ODataId and ODataType field to each model, since it is inconsistently defined in the API specs. - // This won't be valid for every model, but it's impossible to tell which models support them, and it's effectively - // harmless to leave these in so long as they have the `omitempty` struct tag in the generated SDK. + // Proceed to build a model model := Model{ - Fields: map[string]*ModelField{ - "ODataId": { - Title: "ODataId", - Description: "The OData ID of this entity", - Type: pointer.To(DataTypeString), - Default: "", - JsonField: "@odata.id", - }, - "ODataType": { - Title: "ODataType", - Description: "The OData Type of this entity", - Type: pointer.To(DataTypeString), - Default: "", - JsonField: "@odata.type", - }, - }, + Name: normalize.CleanName(schemaName), + Fields: make(map[string]*ModelField), Common: common, - Prefix: input.Prefix, } - // Add to models map before descending, to prevent recursion - models[name] = &model - - for jsonField, schemaRef := range input.Schemas { - // maintainer note: this is a good place to add a breakpoint for inspecting model fields and constants as they are processed - // example breakpoint condition: name=="AdministrativeUnit" - schema := schemaRef.Value - if schema == nil { - continue - } - - field := ModelField{ - Title: normalize.CleanName(jsonField), - Description: schema.Description, - Default: schema.Default, - ReadOnly: schemaRef.Value.ReadOnly, - WriteOnly: schemaRef.Value.WriteOnly, - Nullable: schemaRef.Value.Nullable, - AllowEmptyValue: schemaRef.Value.AllowEmptyValue, - JsonField: jsonField, - } + if schema.Properties != nil { + // Simple model with no base type + for jsonField, fieldDetails := range schema.Properties { + field, err := modelFieldFromSchemaRef(jsonField, fieldDetails) + if err != nil { + return nil, nil, fmt.Errorf("building field %q for model %q: %v", jsonField, schemaName, err) + } - // Detect nullable, read-only and requried fields from the description - if (strings.HasPrefix(schema.Description, "Nullable.") || strings.Contains(schema.Description, " Nullable.")) && !strings.Contains(strings.ToLower(schema.Description), "not nullable.") { - field.Nullable = true - } - if strings.HasPrefix(schema.Description, "Read-only.") || strings.Contains(schema.Description, " Read-only.") { - field.ReadOnly = true - } - if strings.HasPrefix(schema.Description, "Required.") || strings.Contains(schema.Description, " Required.") { - field.Required = true - } + if field == nil { + logging.Warnf("Skipping field %q in model %q", schemaName, jsonField) + continue + } - if field.Title == "" { - continue + model.Fields[jsonField] = field } - result, _ := FlattenSchemaRef(schemaRef, nil) - - // Determine any enumeration for the field - enum := parseEnum(schema.Enum) - if result != nil && len(result.Enum) > 0 && len(enum) == 0 { - enum = parseEnum(result.Enum) - } + } else if schema.AllOf != nil { + // Model with inheritance + for _, allOf := range schema.AllOf { + if referencedSchemaName := TrimRefPrefix(allOf.Ref); referencedSchemaName != "" { + if model.ParentModel != nil { + return nil, nil, fmt.Errorf("model %q already has a parent model %q, cannot set additional parent %q", schemaName, *model.ParentModel, referencedSchemaName) + } - // Find the corresponding model when the field refers to it, and set the model name for the field - if result != nil && result.Title != "" && result.Schemas != nil { - if _, ok := models[result.Title]; !ok { - models, constants = Schemas(*result, result.Title, models, constants, common) + model.ParentModel = &referencedSchemaName + continue } - field.ModelName = &result.Title - } - // Handle items for collections - if schema.Items != nil && schema.Items.Value != nil { - itemsResult, _ := FlattenSchemaRef(schema.Items, nil) + if allOf.Value != nil && allOf.Value.Properties != nil { + for jsonField, fieldDetails := range allOf.Value.Properties { + field, err := modelFieldFromSchemaRef(jsonField, fieldDetails) + if err != nil { + return nil, nil, fmt.Errorf("building field %q for model %q: %v", jsonField, schemaName, err) + } - // Determine any type for the items - if schema.Items.Value.Type != "" || schema.Items.Value.Format != "" { - field.ItemType = FieldType(schema.Items.Value.Type, schema.Items.Value.Format, field.ModelName != nil) - } + if field == nil { + logging.Warnf("Skipping field %q in model %q", schemaName, jsonField) + continue + } - // Find the corresponding model when the field (items) refers to it, and set the model name for the field items - if itemsResult != nil && itemsResult.Title != "" && itemsResult.Type == "object" { - if _, ok := models[result.Title]; !ok { - models, constants = Schemas(*result, result.Title, models, constants, common) + model.Fields[jsonField] = field } - field.ModelName = &result.Title } } + } - // Match the field type to the referenced object, or set a basic type - if result != nil && schema.Type == "" && schema.Format == "" && (result.Type != "" || result.Format != "") { - field.Type = FieldType(result.Type, result.Format, field.ModelName != nil) - } else { - field.Type = FieldType(schema.Type, schema.Format, field.ModelName != nil) - } + return &model, nil, nil +} - // Set the field type for enums; typically strings for constants, but could be any valid type - if result != nil && field.Type != nil && *field.Type == DataTypeArray && len(enum) > 0 && (result.Type != "" || result.Format != "") { - field.ItemType = FieldType(result.Type, result.Format, field.ModelName != nil) - } +func modelFieldFromSchemaRef(jsonField string, fieldSchema *openapi3.SchemaRef) (*ModelField, error) { + if fieldSchema.Value == nil { + return nil, fmt.Errorf("field has no definition") + } - if ((field.Type != nil && *field.Type == DataTypeString) || (field.ItemType != nil && *field.ItemType == DataTypeString)) && len(enum) > 0 { - // Despite being "fully qualified", type names are not unique in MS Graph, so we prefix them with the model name to provide some namespacing. - // Though we attempt to de-duplicate, this does lead to some excessively long constant names, it is what it is. - constantName := name - if result.Title != "" { - // use provided ref name if present - constantName += result.Title - } else { - // otherwise use the field name - constantName += field.Title - } - constantName = normalize.DeDuplicateName(constantName) - constantName = normalize.Singularize(constantName) + field := ModelField{ + Name: normalize.CleanName(jsonField), + Description: fieldSchema.Value.Description, + Default: fieldSchema.Value.Default, + ReadOnly: fieldSchema.Value.ReadOnly, + WriteOnly: fieldSchema.Value.WriteOnly, + Nullable: fieldSchema.Value.Nullable, + AllowEmptyValue: fieldSchema.Value.AllowEmptyValue, + } - field.ConstantName = pointer.To(constantName) + if fieldSchema.Value.AnyOf != nil { + for _, fieldReference := range fieldSchema.Value.AnyOf { + if referencedSchemaName := TrimRefPrefix(fieldReference.Ref); referencedSchemaName != "" { + if field.ReferenceName != nil { + return nil, fmt.Errorf("reference %q already set, cannot set new reference %q", *field.ReferenceName, referencedSchemaName) + } - // Add the enumeration as a constant - constants[constantName] = &Constant{ - Enum: enum, - Type: field.Type, + field.ReferenceName = &referencedSchemaName } } + } - if field.Type == nil { - logging.Warnf("Skipping field %q in model %q because Type is nil", name, field.Title) - continue + if referencedSchemaName := TrimRefPrefix(fieldSchema.Ref); referencedSchemaName != "" { + if field.ReferenceName != nil { + return nil, fmt.Errorf("reference %q already set, cannot set new reference %q", *field.ReferenceName, referencedSchemaName) } - // Insert an "@odata.bind" field where a field or collection refers to a DirectoryObject. The MS Graph - // OpenAPI spec unfortunately does not document relationships between entities. - if field.ModelName != nil && strings.EqualFold(*field.ModelName, "DirectoryObject") { - bindFieldName := fmt.Sprintf("%s_ODataBind", normalize.CleanName(jsonField)) + field.ReferenceName = &referencedSchemaName + } - description := "" - if *field.Type == DataTypeArray { - description = fmt.Sprintf("List of OData IDs for `%s` to bind to this entity", normalize.CleanName(jsonField)) - } else { - description = fmt.Sprintf("OData ID for `%s` to bind to this entity", normalize.CleanName(jsonField)) - } + field.Type = FieldType(fieldSchema.Value.Type, fieldSchema.Value.Format, field.ReferenceName != nil) - var fieldType, itemType *DataType + if items := fieldSchema.Value.Items; items != nil { + if items.Value != nil && items.Value.AnyOf != nil { + for _, itemsReference := range items.Value.AnyOf { + if referencedSchemaName := TrimRefPrefix(itemsReference.Ref); referencedSchemaName != "" { + if field.ReferenceName != nil { + return nil, fmt.Errorf("item reference %q already set, cannot set new reference %q", *field.ReferenceName, referencedSchemaName) + } - fieldType = pointer.To(DataTypeString) - if *field.Type == DataTypeArray { - fieldType = pointer.To(DataTypeArray) - itemType = pointer.To(DataTypeString) + field.ReferenceName = &referencedSchemaName + } } + } - model.Fields[bindFieldName] = &ModelField{ - Title: bindFieldName, - Description: description, - Type: fieldType, - ItemType: itemType, - JsonField: fmt.Sprintf("%s@odata.bind", jsonField), + if referencedSchemaName := TrimRefPrefix(items.Ref); referencedSchemaName != "" { + if field.ReferenceName != nil { + return nil, fmt.Errorf("item reference %q already set, cannot set new reference %q", *field.ReferenceName, referencedSchemaName) } + + field.ReferenceName = &referencedSchemaName + } + + if field.ReferenceName == nil && items.Value == nil { + return nil, fmt.Errorf("item reference not found and items have no definition") } - model.Fields[normalize.CleanName(jsonField)] = &field + field.ItemType = FieldType(items.Value.Type, items.Value.Format, field.ReferenceName != nil) + } + + // Detect nullable, read-only and required fields from the description + if (strings.HasPrefix(fieldSchema.Value.Description, "Nullable.") || strings.Contains(fieldSchema.Value.Description, " Nullable.")) && !strings.Contains(strings.ToLower(fieldSchema.Value.Description), "not nullable.") { + field.Nullable = true + } + if strings.HasPrefix(fieldSchema.Value.Description, "Read-only.") || strings.Contains(fieldSchema.Value.Description, " Read-only.") { + field.ReadOnly = true + } + if strings.HasPrefix(fieldSchema.Value.Description, "Required.") || strings.Contains(fieldSchema.Value.Description, " Required.") { + field.Required = true + } + + if field.Type == nil { + return nil, nil } - return models, constants + return &field, nil } // parseEnum returns a slice of sanitized enum values (which are always strings) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go index fc7f224f577..4df7fed50c7 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go @@ -28,28 +28,26 @@ func (workaroundApplication) Process(apiVersion string, models parser.Models, co return nil } - model, ok := models["Application"] + model, ok := models["microsoft.graph.application"] if !ok { return fmt.Errorf("`Application` model not found") } // Add the `oauth2RequirePostResponse` field if missing if _, ok := model.Fields["OAuth2RequirePostResponse"]; !ok { - model.Fields["OAuth2RequirePostResponse"] = &parser.ModelField{ - Title: "OAuth2RequirePostResponse", + model.Fields["oauth2RequirePostResponse"] = &parser.ModelField{ + Name: "OAuth2RequirePostResponse", Description: "Specifies whether, as part of OAuth 2.0 token requests, Microsoft Entra ID allows POST requests, as opposed to GET requests. The default is false, which specifies that only GET requests are allowed.", Type: pointer.To(parser.DataTypeBool), - JsonField: "oauth2RequirePostResponse", } } // Add the `applicationTemplateId` field if missing if _, ok := model.Fields["ApplicationTemplateId"]; !ok { - model.Fields["ApplicationTemplateId"] = &parser.ModelField{ - Title: "ApplicationTemplateId", + model.Fields["applicationTemplateId"] = &parser.ModelField{ + Name: "ApplicationTemplateId", Description: "Unique identifier of the applicationTemplate. Supports $filter (eq, not, ne). Read-only. null if the app wasn't created from an application template.", Type: pointer.To(parser.DataTypeString), - JsonField: "applicationTemplateId", Nullable: true, } } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index 1f275279f5b..86c1b584267 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -20,113 +20,97 @@ func (workaroundConditionalAccessPolicy) Name() string { } func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { - model, ok := models["ConditionalAccessPolicy"] + model, ok := models["microsoft.graph.conditionalAccessPolicy"] if !ok { return fmt.Errorf("`ConditionalAccessPolicy` model not found") } // grantControls and sessionControls must be null to unset them, so make them nullable + required - if _, ok = model.Fields["GrantControls"]; !ok { + if _, ok = model.Fields["grantControls"]; !ok { return fmt.Errorf("`GrantControls` field not found") } - model.Fields["GrantControls"].Nullable = true - model.Fields["GrantControls"].Required = true - if _, ok = model.Fields["SessionControls"]; !ok { + model.Fields["grantControls"].Nullable = true + model.Fields["grantControls"].Required = true + if _, ok = model.Fields["sessionControls"]; !ok { return fmt.Errorf("`SessionControls` field not found") } - model.Fields["SessionControls"].Nullable = true - model.Fields["SessionControls"].Required = true + model.Fields["sessionControls"].Nullable = true + model.Fields["sessionControls"].Required = true - model, ok = models["ConditionalAccessConditionSet"] + model, ok = models["microsoft.graph.conditionalAccessConditionSet"] if !ok { return fmt.Errorf("`ConditionalAccessConditionSet` model not found") } // devices, locations, platforms must each be null to unset them, so make them nullable + required - if _, ok = model.Fields["Devices"]; !ok { + if _, ok = model.Fields["devices"]; !ok { return fmt.Errorf("`Devices` field not found") } - model.Fields["Devices"].Nullable = true - model.Fields["Devices"].Required = true - if _, ok = model.Fields["Locations"]; !ok { + model.Fields["devices"].Nullable = true + model.Fields["devices"].Required = true + if _, ok = model.Fields["locations"]; !ok { return fmt.Errorf("`Locations` field not found") } - model.Fields["Locations"].Nullable = true - model.Fields["Locations"].Required = true - if _, ok = model.Fields["Platforms"]; !ok { + model.Fields["locations"].Nullable = true + model.Fields["locations"].Required = true + if _, ok = model.Fields["platforms"]; !ok { return fmt.Errorf("`Platforms` field not found") } - model.Fields["Platforms"].Nullable = true - model.Fields["Platforms"].Required = true + model.Fields["platforms"].Nullable = true + model.Fields["platforms"].Required = true - model, ok = models["ConditionalAccessExternalTenants"] + model, ok = models["microsoft.graph.conditionalAccessExternalTenants"] if !ok { return fmt.Errorf("`ConditionalAccessExternalTenants` model not found") } // Add the `members` field if missing - if _, ok = model.Fields["Members"]; !ok { - model.Fields["Members"] = &parser.ModelField{ - Title: "Members", - Type: pointer.To(parser.DataTypeArray), - ItemType: pointer.To(parser.DataTypeString), - JsonField: "members", + if _, ok = model.Fields["members"]; !ok { + model.Fields["members"] = &parser.ModelField{ + Name: "Members", + Type: pointer.To(parser.DataTypeArray), + ItemType: pointer.To(parser.DataTypeString), } } // Set CSV type for field - model, ok = models["ConditionalAccessGuestsOrExternalUsers"] + model, ok = models["microsoft.graph.conditionalAccessGuestsOrExternalUsers"] if !ok { return fmt.Errorf("`ConditionalAccessGuestsOrExternalUsers` model not found") } - if _, ok = model.Fields["GuestOrExternalUserTypes"]; !ok { + if _, ok = model.Fields["guestOrExternalUserTypes"]; !ok { return fmt.Errorf("`GuestOrExternalUserTypes` field not found") } - model.Fields["GuestOrExternalUserTypes"].Type = pointer.To(parser.DataTypeCsv) + //model.Fields["guestOrExternalUserTypes"].Type = pointer.To(parser.DataTypeCsv) - // Rename this constant - if v, ok := constants["ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType"]; ok { - constants["ConditionalAccessGuestOrExternalUserType"] = v - - for _, model = range models { - for fieldName := range model.Fields { - if model.Fields[fieldName].ConstantName != nil && *model.Fields[fieldName].ConstantName == "ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType" { - model.Fields[fieldName].ConstantName = pointer.To("ConditionalAccessGuestOrExternalUserType") - } - } - } - - delete(constants, "ConditionalAccessGuestsOrExternalUsersGuestOrExternalUserType") - } - - model, ok = models["ConditionalAccessSessionControls"] + model, ok = models["microsoft.graph.conditionalAccessSessionControls"] if !ok { return fmt.Errorf("`ConditionalAccessSessionControls` model not found") } // cloudAppSecurityPolicy must be null to unset it, so make it nullable + required - if _, ok = model.Fields["CloudAppSecurity"]; !ok { + if _, ok = model.Fields["cloudAppSecurity"]; !ok { return fmt.Errorf("`CloudAppSecurity` field not found") } - model.Fields["CloudAppSecurity"].Nullable = true - model.Fields["CloudAppSecurity"].Required = true + model.Fields["cloudAppSecurity"].Nullable = true + model.Fields["cloudAppSecurity"].Required = true - model, ok = models["ConditionalAccessUsers"] + model, ok = models["microsoft.graph.conditionalAccessUsers"] if !ok { return fmt.Errorf("`ConditionalAccessUsers` model not found") } // excludeGuestsOrExternalUsers / includeGuestsOrExternalUsers must be null to unset them, so make them nullable + required - if _, ok = model.Fields["ExcludeGuestsOrExternalUsers"]; !ok { + if _, ok = model.Fields["excludeGuestsOrExternalUsers"]; !ok { return fmt.Errorf("`ExcludeGuestsOrExternalUsers` field not found") } - model.Fields["ExcludeGuestsOrExternalUsers"].Nullable = true - model.Fields["ExcludeGuestsOrExternalUsers"].Required = true - if _, ok = model.Fields["IncludeGuestsOrExternalUsers"]; !ok { + model.Fields["excludeGuestsOrExternalUsers"].Nullable = true + model.Fields["excludeGuestsOrExternalUsers"].Required = true + if _, ok = model.Fields["includeGuestsOrExternalUsers"]; !ok { return fmt.Errorf("`IncludeGuestsOrExternalUsers` field not found") } - model.Fields["IncludeGuestsOrExternalUsers"].Nullable = true - model.Fields["IncludeGuestsOrExternalUsers"].Required = true + model.Fields["includeGuestsOrExternalUsers"].Nullable = true + model.Fields["includeGuestsOrExternalUsers"].Required = true return nil } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go deleted file mode 100644 index 89e4de2b7f8..00000000000 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_directoryobject.go +++ /dev/null @@ -1,115 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package workarounds - -import ( - "fmt" - - "github.com/hashicorp/go-azure-helpers/lang/pointer" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -var _ workaround = workaroundDirectoryObject{} - -// workaroundDirectoryObject implements discrimination for the most common implementations of DirectoryObject. More -// can be added as necessary over time; unfortunately there doesn't seem to be any way of auto-detecting these. -type workaroundDirectoryObject struct{} - -func (workaroundDirectoryObject) Name() string { - return "Directory Object / discrimination" -} - -func (workaroundDirectoryObject) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { - model, ok := models["DirectoryObject"] - if !ok { - return fmt.Errorf("`DirectoryObject` model not found") - } - - if _, ok = model.Fields["ODataType"]; !ok { - return fmt.Errorf("`ODataType` field not found in `DireectoryObject` model") - } - - model.Fields["ODataType"].ConstantName = pointer.To("DirectoryObjectType") - model.Fields["ODataType"].DiscriminatedValue = true - model.TypeField = pointer.To("ODataType") - - // Add constant values for the discriminated type value - constants["DirectoryObjectType"] = &parser.Constant{ - Enum: []string{ - "#microsoft.graph.administrativeUnit", - "#microsoft.graph.application", - "#microsoft.graph.device", - "#microsoft.graph.group", - "#microsoft.graph.orgContact", - "#microsoft.graph.servicePrincipal", - "#microsoft.graph.user", - }, - Type: pointer.To(parser.DataTypeString), - } - - // Set the parent model and discriminated type value for AdministrativeUnit - model, ok = models["AdministrativeUnit"] - if !ok { - return fmt.Errorf("`AdministrativeUnit` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.administrativeUnit") - - // Set the parent model and discriminated type value for Application - model, ok = models["Application"] - if !ok { - return fmt.Errorf("`Application` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.application") - - // Set the parent model and discriminated type value for Device - model, ok = models["Device"] - if !ok { - return fmt.Errorf("`Device` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.device") - - // Set the parent model and discriminated type value for Group - model, ok = models["Group"] - if !ok { - return fmt.Errorf("`Group` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.group") - - // Set the parent model and discriminated type value for OrgContact - model, ok = models["OrgContact"] - if !ok { - return fmt.Errorf("`OrgContact` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.orgContact") - - // Set the parent model and discriminated type value for ServicePrincipal - model, ok = models["ServicePrincipal"] - if !ok { - return fmt.Errorf("`ServicePrincipal` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.servicePrincipal") - - // Set the parent model and discriminated type value for User - model, ok = models["User"] - if !ok { - return fmt.Errorf("`User` model not found") - } - model.ParentModelName = pointer.To("DirectoryObject") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.user") - - return nil -} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go deleted file mode 100644 index fca8b6c2d43..00000000000 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_iprange.go +++ /dev/null @@ -1,97 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package workarounds - -import ( - "fmt" - - "github.com/hashicorp/go-azure-helpers/lang/pointer" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -var _ workaround = workaroundIPRange{} - -// workaroundIPRange implements discrimination for iPv4CidrRange, iPv4Range, iPv6CidrRange, and iPv6Range -type workaroundIPRange struct{} - -func (workaroundIPRange) Name() string { - return "IPRange / discrimination" -} - -func (workaroundIPRange) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { - // microsoft.graph.ipRange model is empty and is omitted by the OpenAPI parser - models["IPRange"] = &parser.Model{ - Fields: map[string]*parser.ModelField{ - "ODataId": { - Title: "ODataId", - Type: pointer.To(parser.DataTypeString), - Default: "", - JsonField: "@odata.id", - }, - "ODataType": { - Title: "ODataType", - Type: pointer.To(parser.DataTypeString), - ConstantName: pointer.To("IPRangeType"), - Default: "", - DiscriminatedValue: true, - JsonField: "@odata.type", - }, - }, - Common: true, - TypeField: pointer.To("ODataType"), - } - - if _, ok := constants["IPRangeType"]; ok { - return fmt.Errorf("`IPRangeType` constant already defined") - } - - // Add constant values for the discriminated type value - constants["IPRangeType"] = &parser.Constant{ - Enum: []string{ - "#microsoft.graph.iPv4CidrRange", - "#microsoft.graph.iPv4Range", - "#microsoft.graph.iPv6CidrRange", - "#microsoft.graph.iPv6Range", - }, - Type: pointer.To(parser.DataTypeString), - } - - // Set the parent model and discriminated type value for IPv4CIDRRange - model, ok := models["IPv4CIDRRange"] - if !ok { - return fmt.Errorf("`IPv4CIDRRange` model not found") - } - model.ParentModelName = pointer.To("IPRange") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.iPv4CidrRange") - - // Set the parent model and discriminated type value for IPv4Range - model, ok = models["IPv4Range"] - if !ok { - return fmt.Errorf("`IPv4Range` model not found") - } - model.ParentModelName = pointer.To("IPRange") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.iPv4Range") - - // Set the parent model and discriminated type value for IPv6CIDRRange - model, ok = models["IPv6CIDRRange"] - if !ok { - return fmt.Errorf("`IPv6CIDRRange` model not found") - } - model.ParentModelName = pointer.To("IPRange") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.iPv6CidrRange") - - // Set the parent model and discriminated type value for IPv6Range - model, ok = models["IPv6Range"] - if !ok { - return fmt.Errorf("`IPv6Range` model not found") - } - model.ParentModelName = pointer.To("IPRange") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.iPv6Range") - - return nil -} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go deleted file mode 100644 index c9348a669e3..00000000000 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_namedlocation.go +++ /dev/null @@ -1,69 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package workarounds - -import ( - "fmt" - - "github.com/hashicorp/go-azure-helpers/lang/pointer" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -var _ workaround = workaroundNamedLocation{} - -// workaroundNamedLocation implements discrimination for countryNamedLocation and ipNamedLocation -type workaroundNamedLocation struct{} - -func (workaroundNamedLocation) Name() string { - return "NamedLocation / discrimination" -} - -func (workaroundNamedLocation) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { - model, ok := models["NamedLocation"] - if !ok { - return fmt.Errorf("`NamedLocation` model not found") - } - - if _, ok = model.Fields["ODataType"]; !ok { - return fmt.Errorf("`ODataType` field not found in `NamedLocation` model") - } - - // Set the constant reference and discriminated flag for the @odata.type field - model.Fields["ODataType"].ConstantName = pointer.To("NamedLocationType") - model.Fields["ODataType"].DiscriminatedValue = true - model.TypeField = pointer.To("ODataType") - - if _, ok = constants["NamedLocationType"]; ok { - return fmt.Errorf("`NamedLocationType` constant already defined") - } - - // Add constant values for the discriminated type value - constants["NamedLocationType"] = &parser.Constant{ - Enum: []string{ - "#microsoft.graph.countryNamedLocation", - "#microsoft.graph.ipNamedLocation", - }, - Type: pointer.To(parser.DataTypeString), - } - - // Set the parent model and discriminated type value for CountryNamedLocation - model, ok = models["CountryNamedLocation"] - if !ok { - return fmt.Errorf("`CountryNamedLocation` model not found") - } - model.ParentModelName = pointer.To("NamedLocation") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.countryNamedLocation") - - // Set the parent model and discriminated type value for CountryNamedLocation - model, ok = models["IPNamedLocation"] - if !ok { - return fmt.Errorf("`IPNamedLocation` model not found") - } - model.ParentModelName = pointer.To("NamedLocation") - model.TypeField = pointer.To("ODataType") - model.TypeValue = pointer.To("#microsoft.graph.ipNamedLocation") - - return nil -} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go new file mode 100644 index 00000000000..79bbcc53723 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go @@ -0,0 +1,61 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + "strings" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" +) + +var _ workaround = workaroundODataBind{} + +// workaroundODataBind inserts an `@odata.bind` field where a field or collection refers to a DirectoryObject. The +// OpenAPI spec unfortunately does not document relationships between entities. +type workaroundODataBind struct{} + +func (workaroundODataBind) Name() string { + return "OData / add @odata.bind fields" +} + +func (workaroundODataBind) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { + for schemaName, model := range models { + for jsonField, field := range model.Fields { + if field.ReferenceName != nil && strings.EqualFold(*field.ReferenceName, "microsoft.graph.directoryObject") { + bindFieldName := fmt.Sprintf("%s_ODataBind", normalize.CleanName(jsonField)) + bindFieldJsonName := fmt.Sprintf("%s@odata.bind", jsonField) + + logging.Tracef("Adding `%s_ODataBind` (`%s@odata.bind`) field to %q model", bindFieldName, bindFieldJsonName, schemaName) + + description := "" + if field.Type != nil && *field.Type == parser.DataTypeArray { + description = fmt.Sprintf("List of OData IDs for `%s` to bind to this entity", normalize.CleanName(jsonField)) + } else { + description = fmt.Sprintf("OData ID for `%s` to bind to this entity", normalize.CleanName(jsonField)) + } + + var fieldType, itemType *parser.DataType + + fieldType = pointer.To(parser.DataTypeString) + if *field.Type == parser.DataTypeArray { + fieldType = pointer.To(parser.DataTypeArray) + itemType = pointer.To(parser.DataTypeString) + } + + model.Fields[bindFieldJsonName] = &parser.ModelField{ + Name: bindFieldName, + Description: description, + Type: fieldType, + ItemType: itemType, + } + } + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go index c3d2aba5c8d..afbdc577aa7 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go @@ -4,9 +4,8 @@ package workarounds import ( - "strings" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) var _ workaround = workaroundRepeatingResourceIdSegments{} @@ -21,47 +20,16 @@ func (workaroundRepeatingResourceIdSegments) Name() string { func (workaroundRepeatingResourceIdSegments) Process(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { for resourceIdName := range resourceIds { var invalid bool + resourceId := *resourceIds[resourceIdName] - // Repeating segments, which are not supported - if strings.Contains(resourceIdName, "SiteIdSiteId") { - invalid = true - } - - // GroupSiteTermStore resources have repeating ID segments which are not supported at this time - if strings.Contains(resourceIdName, "TermStore") { - invalid = true - } - - // Onenote resources have repeating ID segments which are not supported at this time - if strings.Contains(resourceIdName, "Onenote") { - invalid = true - } - - // These contain IDs with repeating segments, which are not supported at this time - prefixes := []string{ - "IdentityGovernanceEntitlementManagementAccessPackageIdResourceRoleScopeIdRoleResourceScopeIdResourceRoleId", - "IdentityGovernanceEntitlementManagementAccessPackageIdResourceRoleScopeIdScopeResourceRoleIdResourceScopeId", - "IdentityGovernanceEntitlementManagementCatalogIdResourceRoleIdResourceScopeIdResourceRoleId", - "IdentityGovernanceEntitlementManagementCatalogIdResourceScopeIdResourceRoleIdResourceScopeId", - "IdentityGovernanceEntitlementManagementResourceRequestIdCatalogResourceRoleIdResourceScopeIdResourceRoleId", - "IdentityGovernanceEntitlementManagementResourceRequestIdCatalogResourceScopeIdResourceRoleIdResourceScopeId", - "IdentityGovernanceEntitlementManagementResourceRequestIdResourceRoleIdResourceScopeId", - "IdentityGovernanceEntitlementManagementResourceRequestIdResourceScopeIdResourceRoleId", - "IdentityGovernanceEntitlementManagementResourceRoleScopeIdRoleResourceScopeIdResourceRoleId", - "IdentityGovernanceEntitlementManagementResourceRoleScopeIdScopeResourceRoleIdResourceScopeId", - "MePendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionId", - "MePendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionIdInsightId", - "MePendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionId", - "MePendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionIdInsightId", - "UserIdPendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionId", - "UserIdPendingAccessReviewInstanceIdDecisionIdInstanceStageIdDecisionIdInsightId", - "UserIdPendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionId", - "UserIdPendingAccessReviewInstanceIdStageIdDecisionIdInstanceDecisionIdInsightId", - } - for _, prefix := range prefixes { - if strings.HasPrefix(resourceIdName, prefix) { + seenSegmentNames := map[string]struct{}{} + for _, segment := range resourceId.Segments { + if _, ok := seenSegmentNames[segment.Value]; ok { + logging.Warnf("Dropping resource ID due to duplicate segment %q: %s", resourceIdName, segment.Value) invalid = true + break } + seenSegmentNames[segment.Value] = struct{}{} } if invalid { diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index e44c3345aac..be9f0d1d827 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -11,19 +11,18 @@ import ( ) var workarounds = []workaround{ + workaroundODataBind{}, + workaroundRepeatingResourceIdSegments{}, + workaroundApplication{}, workaroundConditionalAccessPolicy{}, - workaroundDirectoryObject{}, - workaroundNamedLocation{}, - workaroundIPRange{}, - workaroundRepeatingResourceIdSegments{}, } type workaround interface { // Name returns the Service Name and associated Pull Request number Name() string - // Process takes the apiDefinition and applies the Workaround to this AzureApiDefinition + // Process performs any necessary fixes to constants, models and/or resource IDs Process(string, parser.Models, parser.Constants, parser.ResourceIds) error } @@ -31,7 +30,7 @@ type workaround interface { func ApplyWorkarounds(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { logging.Tracef("Processing Data Workarounds..") for _, fix := range workarounds { - logging.Tracef("Applying Data Workaround %q to Model %q", fix.Name()) + logging.Tracef("Applying Data Workaround %q", fix.Name()) if err := fix.Process(apiVersion, models, constants, resourceIds); err != nil { return fmt.Errorf("applying Data Workaround %q: %v", fix.Name(), err) } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go index e1180d7bc79..ae2bfcfe3a7 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/importer.go @@ -58,7 +58,7 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } logging.Infof("Parsing models and constants...") - p.models, p.constants, err = parser.Common(p.spec.Components.Schemas) + p.models, p.constants, err = parser.ModelsAndConstants(p.spec.Components.Schemas) if err != nil { return err } @@ -74,11 +74,6 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha return err } - logging.Infof("Cleaning up models...") - if err = p.cleanupModels(); err != nil { - return err - } - serviceTags, err := tags.Parse(p.spec.Tags) if err != nil { return err diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go b/tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go deleted file mode 100644 index c81c6617e6b..00000000000 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_cleanup_models.go +++ /dev/null @@ -1,31 +0,0 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - -package pipeline - -func (p pipeline) cleanupModels() error { - // First delete all invalid models (i.e. those without fields) - modelsToDelete := make([]string, 0) - for modelName, model := range p.models { - if !model.IsValid() { - modelsToDelete = append(modelsToDelete, modelName) - } - } - - for _, modelName := range modelsToDelete { - delete(p.models, modelName) - } - - // Look for invalid references due to deleted models, and remove them - for _, model := range p.models { - for _, field := range model.Fields { - if field.ModelName != nil { - if !p.models.Found(*field.ModelName) { - field.ModelName = nil - } - } - } - } - - return nil -} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index f1379c1e3f2..03600e7dfd9 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -15,6 +15,8 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) +const directoryObjectSchemaName = "microsoft.graph.directoryObject" + func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, models parser.Models, constants parser.Constants) (resources parser.Resources, err error) { resources = make(parser.Resources) for pathKey, pathItem := range p.spec.Paths { @@ -44,6 +46,10 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model parsedPath := parser.NewResourceId(path, operationTags) lastSegment := parsedPath.Segments[len(parsedPath.Segments)-1] + if len(parsedPath.Segments) == 0 { + continue + } + // Determine whether to skip a path containing unsupported segment types for idx, segment := range parsedPath.Segments { if segment.Type == parser.SegmentCast || segment.Type == parser.SegmentFunction { @@ -134,8 +140,10 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model if operation.Responses != nil { for stat, resp := range operation.Responses { var status int - var contentType, responseModel *string + var responseContentType, responseReferenceName *string var responseType *parser.DataType + var responseConstant *parser.Constant + var responseModel *parser.Model if s, err := strconv.Atoi(strings.ReplaceAll(stat, "X", "0")); err == nil { status = s @@ -144,61 +152,133 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model continue } - if resp.Value != nil && len(resp.Value.Content) > 0 { - if resp.Value.Description != nil && strings.Contains(strings.ToLower(*resp.Value.Description), "collection") { - listOperation = true - } - for t, m := range resp.Value.Content { - contentType = &t + if resp.Value != nil { + for contentType, content := range resp.Value.Content { + responseContentType = &contentType - // Prefer model name from Ref - if strings.HasPrefix(m.Schema.Ref, parser.RefPrefix) { - modelName := normalize.CleanName(m.Schema.Ref[len(parser.RefPrefix):]) - responseModel = &modelName + if content.Schema == nil { + continue } - if m.Schema != nil { - // Flatten the response SchemaRef for inspection - if f, _ := parser.FlattenSchemaRef(m.Schema, nil); f != nil { - if f.Format == "binary" { - responseType = pointer.To(parser.DataTypeBinary) - break - } + switch strings.ToLower(contentType) { + // Binary payloads are handled by the SDK and unmarshalled into []byte + case "application/octet-stream", "text/plain", "text/powershell": + responseType = pointer.To(parser.DataTypeBinary) + + // Supported content types for unmarshalling into a model or other type + case "application/json", "application/xml", "text/xml": + // This is a temporary name for the response model until we determine the name of the operation, or find a referenced type name + schemaName := "ResponseObject" + + // Try to identify constant or model from ref + if ref := parser.TrimRefPrefix(content.Schema.Ref); ref != "" { + schemaName = ref + } else if value := content.Schema.Value; value != nil && value.AnyOf != nil { + for _, ref := range value.AnyOf { + if referencedSchemaName := parser.TrimRefPrefix(ref.Ref); referencedSchemaName != "" { + if schemaName != "ResponseObject" { + return nil, fmt.Errorf("identifying response object: reference %q already set, cannot set new reference %q", schemaName, referencedSchemaName) + } - // Derive model from schema title and/or and response type from schema type - if title := f.Title; title != "" || f.Type != "" { - if strings.HasPrefix(strings.ToLower(title), "collectionof") { - title = title[12:] - listOperation = true + schemaName = referencedSchemaName } + } + } - if responseModel == nil && title != "" { - if modelName := normalize.CleanName(title); models.Found(modelName) { - responseModel = &modelName - } + // Look for a referenced constant or model + if schemaName != "ResponseObject" { + if constants.Found(schemaName) { + responseConstant = constants[schemaName] + responseReferenceName = &schemaName + responseType = pointer.To(parser.DataTypeReference) + } else if models.Found(schemaName) { + responseModel = models[schemaName] + responseReferenceName = &schemaName + responseType = pointer.To(parser.DataTypeReference) + } + } + + // If no constant/model was found, build one from the provided schema + if responseConstant == nil && responseModel == nil { + model, constant, err := parser.ModelOrConstant(schemaName, content.Schema, false) + if err != nil { + return nil, fmt.Errorf("building response model: %v", err) + } + + if constant == nil && model == nil { + return nil, fmt.Errorf("building response object: could not find a constant or model for the reference %q", schemaName) + } + if constant != nil && model != nil { + return nil, fmt.Errorf("building response object: received a constant and a model for the reference %q", schemaName) + } + + if constant != nil { + responseConstant = constant + responseReferenceName = &schemaName + responseType = pointer.To(parser.DataTypeReference) + + } else if model != nil { + // An empty model will be returned if there is no schema, skip this + if len(model.Fields) > 0 || model.ParentModel != nil { + responseModel = model + responseReferenceName = &schemaName + responseType = pointer.To(parser.DataTypeReference) } + } + } + + // GET requests to URIs ending in '/$ref' should always return one or more directoryObjects, + // but this isn't always described in the spec, so hardcode this. + if strings.ToUpper(method) == "GET" && lastSegment.Type == parser.SegmentODataReference && lastSegment.Value == "$ref" { + var value *parser.ModelField + if responseModel != nil { + value, _ = responseModel.Fields["value"] + } + if value != nil && value.Type != nil && *value.Type == parser.DataTypeArray { + value.ItemType = pointer.To(parser.DataTypeReference) + value.ReferenceName = pointer.To(directoryObjectSchemaName) + } else { + responseModel = models[directoryObjectSchemaName] + responseReferenceName = pointer.To(directoryObjectSchemaName) + } + } - if l := parser.FieldType(f.Type, title, responseModel != nil); l != nil { - responseType = l + if responseModel != nil { + // When the response comprises a `value` field containing an array, this is probably a + // paginated list operation, so mark it as such, and take the item type as the response + // type, since the SDK handles pagination without needing an intermediary model. + if value, ok := responseModel.Fields["value"]; ok && value.Type != nil && *value.Type == parser.DataTypeArray { + // Make this a list operation to enable pagination + listOperation = true + + // Prefer the type of the `value` array item + if value.ReferenceName != nil { + // `value` items contain a referenced model + responseReferenceName = value.ReferenceName + responseType = pointer.To(parser.DataTypeReference) + } else if value.ItemType != nil { + // `value` items contain a simple type + responseReferenceName = nil + responseType = value.ItemType } } } + + default: + return nil, fmt.Errorf("unsupported content-type: %q", contentType) } break } } - // Use generic DirectoryObject model for List operations ending in "/$ref" where no other model was found - if listOperation && responseModel == nil && resourceId != nil && len(resourceId.Segments) > 0 && resourceId.Segments[len(resourceId.Segments)-1].Value == "$ref" { - responseModel = pointer.To("DirectoryObject") - } - responses = append(responses, parser.Response{ - Status: status, - ContentType: contentType, - ModelName: responseModel, - Type: responseType, + Status: status, + ContentType: responseContentType, + ReferenceName: responseReferenceName, + Type: responseType, + Constant: responseConstant, + Model: responseModel, }) } } @@ -259,46 +339,92 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + // Now that we have determined the operation name, we can rename any ad-hoc response models accordingly and + // persist them in the global constants or models map. We _might_ have more than one ad-hoc response model, + // so for subsequent ones we just append an integer to ensure the resulting SDK code compiles. Ad-hoc + // response models are those that are described in the response but don't exist as a formal constant/model. + if len(responses) > 0 { + usedNames := make(map[string]struct{}) + + nextAvailableObjectName := func() string { + proposed := fmt.Sprintf("%sResult", operationName) + var count int + for { + if _, ok := usedNames[proposed]; !ok { + break + } + count++ + proposed = fmt.Sprintf("%s%dResult", operationName, count) + } + usedNames[proposed] = struct{}{} + return proposed + } + + for i, response := range responses { + if response.ReferenceName != nil && *response.ReferenceName == "ResponseObject" { + objectName := nextAvailableObjectName() + responses[i].ReferenceName = pointer.To(objectName) + + if response.Constant != nil && response.Constant.Name == "ResponseObject" { + response.Constant.Name = objectName + + // Persist the constant to the global constants + constants[objectName] = response.Constant + } else if response.Model != nil && response.Model.Name == "ResponseObject" { + response.Model.Name = objectName + + // Persist the model to the global models + models[objectName] = response.Model + } + } + } + } + // Determine request model - var requestModel *string + var requestContentType, requestModelName *string var requestType *parser.DataType if operation.RequestBody != nil && operation.RequestBody.Value != nil { for contentType, content := range operation.RequestBody.Value.Content { - if content.Schema != nil { - if schema, _ := parser.FlattenSchemaRef(content.Schema, nil); schema != nil { - if strings.ToLower(schema.Format) == "binary" { - requestType = pointer.To(parser.DataTypeBinary) - break - } + requestContentType = &contentType - if strings.HasPrefix(strings.ToLower(contentType), "application/json") { - var modelName string - if strings.HasPrefix(content.Schema.Ref, parser.RefPrefix) { - // Should be a known model - if modelName = normalize.CleanName(content.Schema.Ref[len(parser.RefPrefix):]); models.Found(modelName) { - requestModel = &modelName - break - } - } else if schema.Title != "" { - // Should be a known model - if modelName = normalize.CleanName(schema.Title); models.Found(modelName) { - requestModel = &modelName - break - } - } else if len(schema.Schemas) > 0 { - // Unique object for this operation - modelName = fmt.Sprintf("%sRequest", operationName) - models, constants = parser.Schemas(*schema, modelName, models, constants, false) - - // Only assign requestModel if the parsed model was valid and added - if models.Found(modelName) { - requestModel = &modelName - } - break - } - } + if content.Schema == nil { + continue + } + + // Binary payloads are handled by the SDK + if content.Schema.Value != nil && content.Schema.Value.Format == "binary" { + requestType = pointer.To(parser.DataTypeBinary) + break + } + + // Try to locate model from ref + var requestModel *parser.Model + if referencedSchemaName := parser.TrimRefPrefix(content.Schema.Ref); referencedSchemaName != "" { + if models.Found(referencedSchemaName) { + requestModel = models[referencedSchemaName] + requestModelName = &referencedSchemaName + } + } + + // If no model was found, build one + if requestModel == nil && content.Schema != nil { + requestModelName = pointer.To(fmt.Sprintf("%sRequest", operationName)) + + model, constant, err := parser.ModelOrConstant(*requestModelName, content.Schema, false) + if err != nil { + return nil, fmt.Errorf("building request model: %v", err) + } + if constant != nil { + return nil, fmt.Errorf("building request model: received a constant but expected a model") } + if model == nil { + return nil, fmt.Errorf("building request model: received a nil model") + } + + models[*requestModelName] = model } + + break } } @@ -308,12 +434,11 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model headers := make(parser.Headers, 0) params := make(parser.Params, 0) for _, param := range operation.Parameters { - if param.Value == nil { + if param.Value == nil || param.Value.Schema == nil || param.Value.Schema.Value == nil { continue } - result, _ := parser.FlattenSchemaRef(param.Value.Schema, nil) - paramType := parser.FieldType(result.Type, result.Format, false) + paramType := parser.FieldType(param.Value.Schema.Value.Type, param.Value.Schema.Value.Format, false) switch param.Value.In { case "header": @@ -324,8 +449,13 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model case "query": var itemType *parser.DataType if paramType != nil && *paramType == parser.DataTypeArray { - itemType = parser.FieldType(result.Type, result.Format, false) + items := param.Value.Schema.Value.Items + if items == nil || items.Value == nil { + return nil, fmt.Errorf("encountered query parameter %q with Array type but no Items", param.Value.Name) + } + itemType = parser.FieldType(items.Value.Type, items.Value.Format, false) } + params = append(params, parser.Param{ Name: param.Value.Name, Type: paramType, @@ -340,12 +470,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model requestParams = ¶ms } - if operationType == parser.OperationTypeCreate || operationType == parser.OperationTypeUpdate || operationType == parser.OperationTypeCreateUpdate { - if resourceId != nil && len(resourceId.Segments) > 0 && resourceId.Segments[len(resourceId.Segments)-1].Value == "$ref" { - requestModel = pointer.To("DirectoryObject") - } - } - + // Construct a description for the operation method descriptionChunks := make([]string, 0) if operation.Summary != "" { descriptionChunks = append(descriptionChunks, operation.Summary) @@ -360,19 +485,20 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model operationDescription = strings.TrimPrefix(operationDescription, ";") resources[resourceName].Operations = append(resources[resourceName].Operations, parser.Operation{ - Name: operationName, - Description: operationDescription, - Type: operationType, - Method: method, - ResourceId: resourceId, - UriSuffix: uriSuffix, - RequestModel: requestModel, - RequestHeaders: requestHeaders, - RequestParams: requestParams, - RequestType: requestType, - Responses: responses, - PaginationField: paginationField, - Tags: operation.Tags, + Name: operationName, + Description: operationDescription, + Type: operationType, + Method: method, + ResourceId: resourceId, + UriSuffix: uriSuffix, + RequestContentType: requestContentType, + RequestModel: requestModelName, + RequestHeaders: requestHeaders, + RequestParams: requestParams, + RequestType: requestType, + Responses: responses, + PaginationField: paginationField, + Tags: operation.Tags, }) } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_common_types.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_common_types.go index 9eb554e4bbf..afbceceaff6 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_common_types.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_common_types.go @@ -16,28 +16,28 @@ func (p pipeline) translateCommonTypesToDataApiSdkTypes() (*sdkModels.CommonType sdkModelsMap := make(map[string]sdkModels.SDKModel) sdkResourceIdsMap := make(map[string]sdkModels.ResourceID) - for modelName, model := range p.models { + for schemaName, model := range p.models { if model.Common { - sdkModel, err := model.DataApiSdkModel(p.models) + sdkModel, err := model.DataApiSdkModel(p.models, p.constants) if err != nil { return nil, err } if sdkModel == nil { - logging.Warnf("skipping invalid model %q as it has no fields", modelName) + logging.Warnf("skipping invalid model %q as it has no fields", schemaName) continue } - sdkModelsMap[modelName] = *sdkModel + sdkModelsMap[model.Name] = *sdkModel } } - for constantName, constant := range p.constants { + for _, constant := range p.constants { constantValues := make(map[string]string) for _, value := range constant.Enum { - // prefix constant value names with underscore to prevent naming conflicts with similarly named models in the generated SDK + // Prefix constant-value names with underscore to prevent naming conflicts with similarly named models in the generated SDK constantValues[fmt.Sprintf("_%s", normalize.CleanName(value))] = value } - sdkConstantsMap[constantName] = sdkModels.SDKConstant{ + sdkConstantsMap[constant.Name] = sdkModels.SDKConstant{ // TODO support additional types, if there are any Type: sdkModels.StringSDKConstantType, Values: constantValues, diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index e632e89fb61..ea56c2d6697 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -50,44 +50,37 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } } + serviceConstants := make(parser.Constants) serviceModels := make(parser.Models) // Populate everything else for _, operation := range resource.Operations { var resourceIdName *string + // Note we longer output resource IDs per service, they are now common types if operation.ResourceId != nil { resourceIdName = &operation.ResourceId.Name - - // No longer output resource IDs per service, they are now common types - //sdkResourceId, err := operation.ResourceId.DataApiSdkResourceId() - //if err != nil { - // return nil, err - //} - // - //sdkServices[resource.Service].APIVersions[resource.Version].Resources[resource.Category].ResourceIDs[operation.ResourceId.Name] = *sdkResourceId } var requestObject *sdkModels.SDKObjectDefinition requestObjectIsCommonType := true if operation.RequestModel != nil { - if !p.models.Found(*operation.RequestModel) { - return nil, fmt.Errorf("request model %q was not found for operation: %s", *operation.RequestModel, operation.Name) + schemaName := *operation.RequestModel + + if !p.models.Found(schemaName) { + return nil, fmt.Errorf("request model %q was not found for operation: %s", schemaName, operation.Name) } - if model := p.models[*operation.RequestModel]; !model.IsValid() { - return nil, fmt.Errorf("request model %q was invalid for operation: %s", *operation.RequestModel, operation.Name) - } else if !model.Common { - requestObjectIsCommonType = false + model := p.models[schemaName] - if err := serviceModels.MergeDependants(p.models, *operation.RequestModel, false); err != nil { - return nil, err - } + if !model.Common { + requestObjectIsCommonType = false + serviceModels[schemaName] = model } requestObject = &sdkModels.SDKObjectDefinition{ - ReferenceName: operation.RequestModel, + ReferenceName: pointer.To(normalize.CleanName(*operation.RequestModel)), ReferenceNameIsCommonType: &requestObjectIsCommonType, Type: sdkModels.ReferenceSDKObjectDefinitionType, } @@ -234,46 +227,36 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv responseObjectIsCommonType := true for _, response := range operation.Responses { - if response.Type != nil && *response.Type == parser.DataTypeModel && response.ModelName != nil { - modelName := *response.ModelName + if response.Type != nil && *response.Type == parser.DataTypeReference && response.ReferenceName != nil { + schemaName := *response.ReferenceName - if !p.models.Found(modelName) { - return nil, fmt.Errorf("response model %q was not found for operation: %s", modelName, operation.Name) + if !p.constants.Found(schemaName) && !p.models.Found(schemaName) { + return nil, fmt.Errorf("response constant or model %q was not found for operation: %s", schemaName, operation.Name) + } + if p.constants.Found(schemaName) && p.models.Found(schemaName) { + return nil, fmt.Errorf("response object %q was found as both a constant and a model for operation: %s", schemaName, operation.Name) } - model := p.models[modelName] - - if !model.IsValid() { - return nil, fmt.Errorf("response model %q was invalid for operation: %s", modelName, operation.Name) - } else if !model.Common { - responseObjectIsCommonType = false + if p.constants.Found(schemaName) { + constant := p.constants[schemaName] - if err := serviceModels.MergeDependants(p.models, modelName, false); err != nil { - return nil, err + if !constant.Common { + responseObjectIsCommonType = false + serviceConstants[schemaName] = constant } } - // List operations return a "CollectionResponse" object, which we are not interested in - // We want the actual underlying model, expected to be in the `value` field - if operation.Type == parser.OperationTypeList { - if value, ok := model.Fields["Value"]; ok && value != nil && *value.Type == parser.DataTypeArray && value.ModelName != nil { - responseObjectIsCommonType = true - modelName = *value.ModelName - - if !p.models.Found(modelName) { - return nil, fmt.Errorf("nested response model %q was not found for operation: %s", modelName, operation.Name) - } else if !model.Common { - responseObjectIsCommonType = false - - if err := serviceModels.MergeDependants(p.models, modelName, false); err != nil { - return nil, err - } - } + if p.models.Found(schemaName) { + model := p.models[schemaName] + + if !model.Common { + responseObjectIsCommonType = false + serviceModels[schemaName] = model } } responseObject = &sdkModels.SDKObjectDefinition{ - ReferenceName: &modelName, + ReferenceName: pointer.To(normalize.CleanName(schemaName)), ReferenceNameIsCommonType: &responseObjectIsCommonType, Type: sdkModels.ReferenceSDKObjectDefinitionType, } @@ -282,6 +265,11 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv Type: response.Type.DataApiSdkObjectDefinitionType(), } } + + // Only one response object is currently supported by the SDK, so break here. This doesn't affect + // us right now since we already ignored error responses during parsing, and to date the specs + // have at most one "success" response per operation. + break } contentType := "application/json" @@ -310,30 +298,13 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } } - for modelName, model := range serviceModels { - sdkModel, err := model.DataApiSdkModel(p.models) + for _, model := range serviceModels { + sdkModel, err := model.DataApiSdkModel(p.models, p.constants) if err != nil { return nil, err } - sdkService.APIVersions[resource.Version].Resources[resource.Category].Models[modelName] = *sdkModel - - for _, field := range model.Fields { - if field.ConstantName != nil { - constantValues := make(map[string]string) - if constant, ok := p.constants[*field.ConstantName]; ok { - for _, value := range constant.Enum { - constantValues[normalize.CleanName(value)] = value - } - } - - // TODO support additional types, if there are any - sdkService.APIVersions[resource.Version].Resources[resource.Category].Constants[*field.ConstantName] = sdkModels.SDKConstant{ - Type: sdkModels.StringSDKConstantType, - Values: constantValues, - } - } - } + sdkService.APIVersions[resource.Version].Resources[resource.Category].Models[model.Name] = *sdkModel } } From 99867f6e8a2dbe34360ac87c0b3eec8b3ecff6c4 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 4 Sep 2024 21:43:12 +0100 Subject: [PATCH 089/117] importer-msgraph-metadata: explicitly flag parent models at import time to avoid expensive iteration+computation at generation time --- .../repository/internal/models/models.go | 3 +++ .../repository/internal/transforms/sdk_model.go | 6 ++++-- tools/data-api-sdk/v1/models/sdk_model.go | 5 ++++- tools/importer-msgraph-metadata/components/parser/types.go | 5 ++++- 4 files changed, 15 insertions(+), 4 deletions(-) diff --git a/tools/data-api-repository/repository/internal/models/models.go b/tools/data-api-repository/repository/internal/models/models.go index 54af13b2126..23c77fd6c8b 100644 --- a/tools/data-api-repository/repository/internal/models/models.go +++ b/tools/data-api-repository/repository/internal/models/models.go @@ -13,6 +13,9 @@ type Model struct { // Fields is an array of fields contained in the Model Fields []ModelField `json:"fields"` + // IsParent specifies whether this model is a known parent model, typically for discriminated child models. + IsParent bool + // NOTE: If DiscriminatedParentModelName and DiscriminatedTypeValue are both populated then this Model // represents a discriminated type // DiscriminatedParentModelName contains the name of the Parent Model that this Model would implement diff --git a/tools/data-api-repository/repository/internal/transforms/sdk_model.go b/tools/data-api-repository/repository/internal/transforms/sdk_model.go index 47da9488c00..051348c0720 100644 --- a/tools/data-api-repository/repository/internal/transforms/sdk_model.go +++ b/tools/data-api-repository/repository/internal/transforms/sdk_model.go @@ -27,6 +27,7 @@ func MapSDKModelFromRepository(input repositoryModels.Model) (*sdkModels.SDKMode DiscriminatedValue: input.DiscriminatedTypeValue, FieldNameContainingDiscriminatedValue: input.TypeHintIn, Fields: fields, + IsParent: input.IsParent, ParentTypeName: input.DiscriminatedParentModelName, }, nil } @@ -42,8 +43,9 @@ func MapSDKModelToRepository(modelName string, model sdkModels.SDKModel, parentM } dataApiModel := repositoryModels.Model{ - Name: modelName, - Fields: *fields, + Name: modelName, + Fields: *fields, + IsParent: model.IsParent, } // NOTE: `Parent` types don't get a `DiscriminatedValue` diff --git a/tools/data-api-sdk/v1/models/sdk_model.go b/tools/data-api-sdk/v1/models/sdk_model.go index 35eba91b030..f57b5af8045 100644 --- a/tools/data-api-sdk/v1/models/sdk_model.go +++ b/tools/data-api-sdk/v1/models/sdk_model.go @@ -19,6 +19,9 @@ type SDKModel struct { // NOTE: the Field Name is a valid Identifier. Fields map[string]SDKField `json:"fields"` + // IsParent specifies whether this model is a known parent model, typically for discriminated child models. + IsParent bool + // ParentTypeName optionally specifies the name of the Parent Type for this Model. // If a ParentTypeName is present then this SDKModel will be a Discriminated Implementation // meaning that a TypeHintIn and TypeHintValue must also be specified in order to uniquely @@ -35,5 +38,5 @@ func (m SDKModel) IsDiscriminatedImplementation() bool { // IsDiscriminatedParentType returns whether this SDKModel is a Discriminated Parent Type. // This means that this SDKModel will have associated Discriminated Implementations. func (m SDKModel) IsDiscriminatedParentType() bool { - return m.ParentTypeName == nil && m.FieldNameContainingDiscriminatedValue != nil + return m.IsParent || (m.ParentTypeName == nil && m.FieldNameContainingDiscriminatedValue != nil) } diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 023d2d58f9a..85fda525303 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -79,6 +79,7 @@ type Model struct { Name string Fields map[string]*ModelField Common bool + Parent bool TypeField *string TypeValue *string ParentModel *string @@ -139,6 +140,7 @@ func (m *Model) DataApiSdkModel(models Models, constants Constants) (*sdkModels. return &sdkModels.SDKModel{ Fields: sdkFields, + IsParent: m.Parent, DiscriminatedValue: m.TypeValue, FieldNameContainingDiscriminatedValue: m.TypeField, ParentTypeName: parentTypeName, @@ -441,7 +443,7 @@ func ModelsAndConstants(schemas openapi3.Schemas) (Models, Constants, error) { } } - // Now iterate models and populate discriminated children + // Now iterate models, mark parent models as such and populate discriminated children for schemaName, model := range models { if model.ParentModel == nil { continue @@ -453,6 +455,7 @@ func ModelsAndConstants(schemas openapi3.Schemas) (Models, Constants, error) { } parentModel.Fields["@odata.type"].DiscriminatedValue = true + parentModel.Parent = true parentModel.TypeField = pointer.To("ODataType") model.TypeField = pointer.To("ODataType") model.TypeValue = pointer.To("#" + schemaName) From 23fa8046591fd2fd814096c01f6ce727bfedd0d1 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 4 Sep 2024 21:44:38 +0100 Subject: [PATCH 090/117] generator-go-sdk: support for parsing and generating extended model ancestry For microsoft-graph SDK, recursively traverse model ancestry and represent all relationships in the generated SDK. This involves: - Generating an interface for every parent model. - Embed (grand)parent model interfaces. - Generating a `Base{Model}Impl` struct for every parent model. - Add a method to each model interface for returning a `Base{Model}Impl` struct, which must be implemented on child model structs. - Inherit fields in model structs from all ancestors. - Adjust marshal/unmarshal methods and funcs accordingly. This enables child models to hold inherited state from any ancestors, whilst being able to return a standalone representation for any of their ancestors. This maintains current behavior when generating resource-manager SDK (traverses model ancestors until one with valid discriminator field is found, and only generates one parent->child relationship). This is controlled by a code setting, so the above new behavior can be enabled for resource-manager once the importer has been updated to represent ancestry in API definitions (not forgetting the new `IsParent` field on each model which is needed to facilitate this). --- .../generator-go-sdk/internal/cmd/generate.go | 5 + .../internal/generator/data.go | 5 + .../internal/generator/settings.go | 1 + .../internal/generator/templater_models.go | 341 +++++++++++------- 4 files changed, 231 insertions(+), 121 deletions(-) diff --git a/tools/generator-go-sdk/internal/cmd/generate.go b/tools/generator-go-sdk/internal/cmd/generate.go index c610f45a74d..6a479f29cb1 100644 --- a/tools/generator-go-sdk/internal/cmd/generate.go +++ b/tools/generator-go-sdk/internal/cmd/generate.go @@ -60,14 +60,19 @@ func (g GenerateCommand) Run(args []string) int { if g.sourceDataType == models.MicrosoftGraphSourceDataType { input.settings.GenerateDescriptionsForModels = true + input.settings.RecurseParentModels = false + input.settings.CanonicalApiVersions = map[string]string{ "stable": "v1.0", } + input.settings.VersionsToGenerateCommonTypes = map[string]models.SourceDataOrigin{ "stable": models.MicrosoftGraphMetaDataSourceDataOrigin, "beta": models.MicrosoftGraphMetaDataSourceDataOrigin, } } else if g.sourceDataType == models.ResourceManagerSourceDataType { + input.settings.RecurseParentModels = true + input.settings.UseOldBaseLayerFor( // @tombuildsstuff: New Services should now use the `hashicorp/go-azure-sdk` base layer by default // instead of the base layer from `Azure/go-autorest` - as such this list is for compatibility purposes diff --git a/tools/generator-go-sdk/internal/generator/data.go b/tools/generator-go-sdk/internal/generator/data.go index ddf14847d47..7286b8a5eee 100644 --- a/tools/generator-go-sdk/internal/generator/data.go +++ b/tools/generator-go-sdk/internal/generator/data.go @@ -80,6 +80,9 @@ type GeneratorData struct { // whether this is a data plane SDK (omits certain Resource Manager specific features, currently used in ID parsers) isDataPlane bool + // whether parent models should be traversed recursively so that the most distant ancestor is allocated as the parent + recurseParentModels bool + // development feature flag - should this service use the new transport layer from `hashicorp/go-azure-sdk` // rather than the existing Autorest base layer? useNewBaseLayer bool @@ -119,6 +122,7 @@ func (i ServiceGeneratorInput) generatorData(settings Settings) GeneratorData { models: i.ResourceDetails.Models, operations: i.ResourceDetails.Operations, packageName: resourcePackageName, + recurseParentModels: settings.RecurseParentModels, resourceIds: i.ResourceDetails.ResourceIDs, resourceOutputPath: resourceOutputPath, serviceClientName: fmt.Sprintf("%sClient", strings.Title(i.ResourceName)), @@ -169,6 +173,7 @@ func (i VersionGeneratorInput) generatorData(settings Settings) VersionGenerator isDataPlane: models.SourceDataTypeIsDataPlane(i.Type), models: i.CommonTypes.Models, packageName: versionPackageName, + recurseParentModels: settings.RecurseParentModels, servicePackageName: strings.ToLower(i.ServiceName), source: i.Source, sourceType: i.Type, diff --git a/tools/generator-go-sdk/internal/generator/settings.go b/tools/generator-go-sdk/internal/generator/settings.go index 86fefd6bde4..7aac2a46055 100644 --- a/tools/generator-go-sdk/internal/generator/settings.go +++ b/tools/generator-go-sdk/internal/generator/settings.go @@ -13,6 +13,7 @@ type Settings struct { CanonicalApiVersions map[string]string CommonTypesPackageName string GenerateDescriptionsForModels bool + RecurseParentModels bool VersionsToGenerateCommonTypes map[string]models.SourceDataOrigin servicesUsingOldBaseLayer map[string]struct{} } diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 66c89cdf5d5..8e978702802 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -67,9 +67,9 @@ func (c modelsTemplater) structCode(data GeneratorData) (*string, error) { out := "" structName := c.name - // parent models get a {model}Base struct + // parent models get a Base{model}Impl struct so as not to conflict with their interface name if c.model.IsDiscriminatedParentType() { - structName = fmt.Sprintf("%sBase", c.name) + structName = fmt.Sprintf("Base%sImpl", c.name) } fields := make([]string, 0) @@ -98,69 +98,91 @@ func (c modelsTemplater) structCode(data GeneratorData) (*string, error) { // then add any inherited fields parentAssignmentInfo := "" - parentTypeName := "" + ancestorTypeNames := make([]string, 0) if c.model.ParentTypeName != nil { + ancestorTypeNames = append(ancestorTypeNames, *c.model.ParentTypeName) if c.model.FieldNameContainingDiscriminatedValue != nil { - _, foundParentTypeName, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + _, foundAncestorTypeNames, err := c.findModelAncestry(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) if err != nil { return nil, err } - parentTypeName = *foundParentTypeName - - } else { - parentTypeName = *c.model.ParentTypeName + ancestorTypeNames = *foundAncestorTypeNames } - parentAssignmentInfo = fmt.Sprintf("var _ %[1]s = %[2]s{}", parentTypeName, structName) - parentFields := make(map[string]models.SDKField) + // Since ancestor interfaces embed each other, we only need to satisfy the immediate parent interface + parentAssignmentInfo = fmt.Sprintf("var _ %[1]s = %[2]s{}", ancestorTypeNames[0], structName) - parent, ok := data.models[*c.model.ParentTypeName] - if !ok { - return nil, fmt.Errorf("couldn't find Parent Model %q for Model %q", *c.model.ParentTypeName, c.name) - } - for fieldName, fieldDetails := range parent.Fields { - parentFields[fieldName] = fieldDetails - } - - // Also include fields from the grandparent model - // Related to: https://github.com/hashicorp/pandora/issues/1235 - if parentTypeName != *c.model.ParentTypeName { - grandParent, ok := data.models[parentTypeName] + // We want to include fields from all ancestors, grouped by ancestor name + ancestorFields := make(map[string]map[string]models.SDKField) + for _, ancestorTypeName := range ancestorTypeNames { + parent, ok := data.models[ancestorTypeName] if !ok { - return nil, fmt.Errorf("couldn't find [Grand]Parent Model %q for Model %q", parentTypeName, c.name) + return nil, fmt.Errorf("couldn't find Ancestor Model %q for Model %q", ancestorTypeName, c.name) } - for fieldName, fieldDetails := range grandParent.Fields { - parentFields[fieldName] = fieldDetails + ancestorFields[ancestorTypeName] = make(map[string]models.SDKField) + for fieldName, fieldDetails := range parent.Fields { + ancestorFields[ancestorTypeName][fieldName] = fieldDetails } } - parentFieldNames := make([]string, 0, len(parentFields)) - for fieldName := range parentFields { - parentFieldNames = append(parentFieldNames, fieldName) - } - sort.Strings(parentFieldNames) - - if len(parentFieldNames) > 0 { - structLines = append(structLines, fmt.Sprintf("\n// Fields inherited from %s", *c.model.ParentTypeName)) - for _, fieldName := range parentFieldNames { - fieldDetails := parentFields[fieldName] - fieldTypeName := "FIXME" - fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, data.commonTypesPackageName) - if err != nil { - return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) + // Get sorted slices of ancestors' field names + ancestorFieldNames := make(map[string][]string) + for ancestorName := range ancestorFields { + ancestorFieldNames[ancestorName] = make([]string, 0, len(ancestorFields[ancestorName])) + for fieldName := range ancestorFields[ancestorName] { + ancestorFieldNames[ancestorName] = append(ancestorFieldNames[ancestorName], fieldName) + } + sort.Strings(ancestorFieldNames[ancestorName]) + } + + // Append fields from all ancestors to struct + for _, ancestorName := range ancestorTypeNames { + if len(ancestorFieldNames[ancestorName]) > 0 { + structLines = append(structLines, fmt.Sprintf("\n// Fields inherited from %s", ancestorName)) + for _, fieldName := range ancestorFieldNames[ancestorName] { + fieldDetails := ancestorFields[ancestorName][fieldName] + fieldTypeName := "FIXME" + fieldTypeVal, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, data.commonTypesPackageName) + if err != nil { + return nil, fmt.Errorf("determining type information for %q: %+v", fieldName, err) + } + fieldTypeName = *fieldTypeVal + + structLine, err := c.structLineForField(fieldName, fieldTypeName, fieldDetails, data) + if err != nil { + return nil, err + } + + structLines = append(structLines, *structLine) } - fieldTypeName = *fieldTypeVal + } + } + } - structLine, err := c.structLineForField(fieldName, fieldTypeName, fieldDetails, data) - if err != nil { - return nil, err - } + // If this is a parent model, we output an Interface with a manual unmarshal func that gets called wherever it's used + if c.model.IsDiscriminatedParentType() { + interfaceLines := make([]string, 0, len(ancestorTypeNames)+1) - structLines = append(structLines, *structLine) + // First we embed any ancestor types (in reverse ancestral order for neatness) + if len(ancestorTypeNames) > 0 { + for i := len(ancestorTypeNames) - 1; i >= 0; i-- { + interfaceLines = append(interfaceLines, ancestorTypeNames[i]) } } + + // Then we implement a method for the base type + interfaceLines = append(interfaceLines, fmt.Sprintf(`%[1]s() %[2]s`, c.name, structName)) + + // Output an interface for the parent type + out += fmt.Sprintf(` +type %[1]s interface { + %[2]s +} + +`, c.name, strings.Join(interfaceLines, "\n")) } + // Output a model struct formattedStructLines := make([]string, 0) for i, v := range structLines { if strings.HasPrefix(strings.TrimSpace(v), "//") { @@ -181,31 +203,47 @@ type %[1]s struct { } `, structName, strings.Join(formattedStructLines, "\n"), parentAssignmentInfo) - // if this is an Abstract/Type Hint, we output an Interface with a manual unmarshal func that gets called wherever it's used + parentModelFunctions, err := c.codeForParentStructFunctions(data) + if err != nil { + return nil, fmt.Errorf("generating parent model functions: %+v", err) + } + out += *parentModelFunctions + + // Parent models also get an implementation struct for use as a fallback when unmarshalling and no child type is found if c.model.IsDiscriminatedParentType() { + // Output a Raw{Type}Impl struct for use as a fallback when unmarshalling an unimplemented discriminated type + implementationStructName := fmt.Sprintf("Raw%sImpl", c.name) + out += fmt.Sprintf(` -type %[1]s interface { - %[1]s() %[1]sBase -} +var _ %[1]s = %[3]s{} -// Raw%[1]sImpl is returned when the Discriminated Value -// doesn't match any of the defined types +// %[3]s is returned when the Discriminated Value doesn't match any of the defined types // NOTE: this should only be used when a type isn't defined for this type of Object (as a workaround) // and is used only for Deserialization (e.g. this cannot be used as a Request Payload). -type Raw%[1]sImpl struct { - %[2]s %[1]sBase +type %[3]s struct { + %[2]s %[4]s Type string Values map[string]interface{} } -`, c.name, camelCase(c.name)) - - out += fmt.Sprintf(` -func (s Raw%[1]sImpl) %[1]s() %[1]sBase { +func (s %[3]s) %[1]s() %[4]s { return s.%[2]s } -`, c.name, camelCase(c.name)) +`, c.name, camelCase(c.name), implementationStructName, structName) + + // Output functions for Raw{Type}Impl to satisfy ancestor interfaces + for _, ancestorName := range ancestorTypeNames { + // the parent model we're returning will get a Base{model}Impl struct + ancestorStructName := fmt.Sprintf("Base%sImpl", ancestorName) + + out += fmt.Sprintf(` +func (s %[1]s) %[2]s() %[3]s { + return s.%[4]s.%[2]s() +} + +`, implementationStructName, ancestorName, ancestorStructName, camelCase(c.name)) + } } return &out, nil @@ -220,12 +258,6 @@ func (c modelsTemplater) methods(data GeneratorData) (*string, error) { } code = append(code, *dateFunctions) - parentModelFunctions, err := c.codeForParentStructFunctions(data) - if err != nil { - return nil, fmt.Errorf("generating parent model functions: %+v", err) - } - code = append(code, *parentModelFunctions) - marshalFunctions, err := c.codeForMarshalFunctions(data) if err != nil { return nil, fmt.Errorf("generating marshal functions: %+v", err) @@ -247,24 +279,7 @@ func (c modelsTemplater) methods(data GeneratorData) (*string, error) { func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDetails models.SDKField, data GeneratorData) (*string, error) { jsonDetails := fieldDetails.JsonName - isOptional := false - if fieldDetails.Optional { - isOptional = true - - // TODO: this'll want rolling out by Service Package (likely using the new base layer toggle, for now) - // but I'm disabling this entirely for the moment to workaround the issue. - if !featureflags.OptionalDiscriminatorsShouldBeOutputWithoutOmitEmpty { - // however if the immediate (not top-level) object definition is a Reference to a Parent it's Optional - // by default since Parent types are output as an interface (which is implied nullable) - if fieldDetails.ObjectDefinition.Type == models.ReferenceSDKObjectDefinitionType { - model, ok := data.models[*fieldDetails.ObjectDefinition.ReferenceName] - if ok && model.FieldNameContainingDiscriminatedValue != nil && model.ParentTypeName == nil { - isOptional = false - } - } - } - } - if isOptional || fieldDetails.ReadOnly { + if c.fieldIsOptional(data, fieldDetails) || fieldDetails.ReadOnly { if !strings.HasPrefix(fieldType, "nullable.") { fieldType = fmt.Sprintf("*%s", fieldType) } @@ -391,51 +406,74 @@ func (c modelsTemplater) dateFunctionForField(fieldName string, fieldDetails mod func (c modelsTemplater) codeForParentStructFunctions(data GeneratorData) (*string, error) { out := "" + structName := c.name if c.model.ParentTypeName == nil { return &out, nil } - parentTypeName := "" - structFields := make([]string, 0) + // parent models get a Base{model}Impl struct so as not to conflict with their interface name + if c.model.IsDiscriminatedParentType() { + structName = fmt.Sprintf("Base%sImpl", c.name) + } + + ancestorTypeNames := make([]string, 0) + ancestorTypeNames = append(ancestorTypeNames, *c.model.ParentTypeName) if c.model.FieldNameContainingDiscriminatedValue != nil { - _, foundParentTypeName, err := c.recurseParentModels(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + _, foundAncestorTypeNames, err := c.findModelAncestry(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) if err != nil { return nil, err } - parentTypeName = *foundParentTypeName - - } else { - parentTypeName = *c.model.ParentTypeName + ancestorTypeNames = *foundAncestorTypeNames } - // Intentionally only setting fields from the outermost parent model - parent, ok := data.models[parentTypeName] - if !ok { - return nil, fmt.Errorf("couldn't find Parent Model %q for Model %q", *c.model.ParentTypeName, c.name) + // We want to include fields from all ancestors, grouped by ancestor name + ancestorFields := make(map[string]map[string]models.SDKField) + for _, ancestorTypeName := range ancestorTypeNames { + parent, ok := data.models[ancestorTypeName] + if !ok { + return nil, fmt.Errorf("couldn't find Ancestor Model %q for Model %q", ancestorTypeName, c.name) + } + ancestorFields[ancestorTypeName] = make(map[string]models.SDKField) + for fieldName, fieldDetails := range parent.Fields { + ancestorFields[ancestorTypeName][fieldName] = fieldDetails + } } - parentFields := make([]string, 0) - for fieldName := range parent.Fields { - parentFields = append(parentFields, fieldName) + // Get sorted slices of ancestors' field names + ancestorFieldNames := make(map[string][]string) + for ancestorName := range ancestorFields { + ancestorFieldNames[ancestorName] = make([]string, 0, len(ancestorFields[ancestorName])) + for fieldName := range ancestorFields[ancestorName] { + ancestorFieldNames[ancestorName] = append(ancestorFieldNames[ancestorName], fieldName) + } + sort.Strings(ancestorFieldNames[ancestorName]) } - sort.Strings(parentFields) - if len(parentFields) > 0 { - for _, fieldName := range parentFields { - structFields = append(structFields, fmt.Sprintf(`%[1]s: s.%[1]s,`, fieldName)) + // Output functions to satisfy ancestor interfaces + for i, ancestorName := range ancestorTypeNames { + // parent models get a Base{model}Impl struct so as not to conflict with their interface name + ancestorStructName := fmt.Sprintf("Base%sImpl", ancestorName) + ancestorStructFields := make([]string, 0) + + // Get fields from the current/next ancestor, plus any older ancestors + for j := i; j < len(ancestorTypeNames); j++ { + nextAncestorName := ancestorTypeNames[j] + for _, fieldName := range ancestorFieldNames[nextAncestorName] { + ancestorStructFields = append(ancestorStructFields, fmt.Sprintf(`%[1]s: s.%[1]s,`, fieldName)) + } } - } - out += fmt.Sprintf(` -func (s %[1]s) %[2]s() %[2]sBase { - return %[2]sBase{ - %[3]s + out += fmt.Sprintf(` +func (s %[1]s) %[2]s() %[3]s { + return %[3]s{ + %[4]s } } -`, c.name, parentTypeName, strings.Join(structFields, "\n")) +`, structName, ancestorName, ancestorStructName, strings.Join(ancestorStructFields, "\n")) + } return &out, nil } @@ -467,9 +505,9 @@ func (c modelsTemplater) codeForMarshalFunctions(data GeneratorData) (*string, e structName := c.name - // parent models get a {model}Base struct + // parent models get a Base{model}Impl struct so as not to conflict with their interface name if c.model.IsDiscriminatedParentType() { - structName = fmt.Sprintf("%sBase", c.name) + structName = fmt.Sprintf("Base%sImpl", c.name) } output += fmt.Sprintf(` @@ -484,7 +522,7 @@ func (s %[1]s) MarshalJSON() ([]byte, error) { } var decoded map[string]interface{} - if err := json.Unmarshal(encoded, &decoded); err != nil { + if err = json.Unmarshal(encoded, &decoded); err != nil { return nil, fmt.Errorf("unmarshaling %[1]s: %%+v", err) } @@ -556,6 +594,9 @@ func (c modelsTemplater) codeForUnmarshalParentFunction(data GeneratorData) (*st return &output, nil } + // parent models get a Base{model}Impl struct so as not to conflict with their interface name + structName := fmt.Sprintf("Base%sImpl", c.name) + // if this is a Discriminated Type (e.g. Parent) then we need to generate an Unmarshal{Name}Implementation // function which can be used in any usages lines := make([]string, 0) @@ -618,19 +659,19 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { // if it doesn't match - we generate and deserialize into a 'Raw{Name}Impl' type - named intentionally // so that we don't conflict with a generated 'Raw{Name}' type which exists in a handful of Swaggers + implementationStructName := fmt.Sprintf("Raw%sImpl", c.name) lines = append(lines, fmt.Sprintf(` - var parent %[1]sBase + var parent %[1]s if err := json.Unmarshal(input, &parent); err != nil { - return nil, fmt.Errorf("unmarshaling into %[1]sBase: %%+v", err) + return nil, fmt.Errorf("unmarshaling into %[1]s: %%+v", err) } - out := Raw%[1]sImpl{ - %[2]s: parent, + return %[2]s{ + %[3]s: parent, Type: value, Values: temp, - } - return out, nil -`, c.name, camelCase(c.name))) + }, nil +`, structName, implementationStructName, camelCase(c.name))) lines = append(lines, "}") @@ -641,9 +682,9 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { func (c modelsTemplater) codeForUnmarshalStructFunction(data GeneratorData) (*string, error) { structName := c.name - // parent models get a {model}Base struct + // parent models get a Base{model}Impl struct so as not to conflict with their interface name if c.model.IsDiscriminatedParentType() { - structName = fmt.Sprintf("%sBase", c.name) + structName = fmt.Sprintf("Base%sImpl", c.name) } lines := make([]string, 0) @@ -816,9 +857,14 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, structName)) }`, fieldName, *topLevelObjectDef.ReferenceName, structName, assignmentPrefix, fieldDetails.JsonName)) } - // if the field is read-only, we need to assign the pointer value assignmentPrefix := "" - if fieldDetails.ReadOnly { + fieldType, err := helpers.GolangTypeForSDKObjectDefinition(fieldDetails.ObjectDefinition, nil, data.commonTypesPackageName) + if err != nil { + return nil, err + } + + // if the field is optional, read-only, or nullable, but not a nullable type, we need to assign the pointer value + if (c.fieldIsOptional(data, fieldDetails) || fieldDetails.ReadOnly || fieldDetails.ObjectDefinition.Nullable) && !strings.HasPrefix(*fieldType, "nullable.") { assignmentPrefix = "&" } @@ -842,7 +888,43 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, structName)) return &output, nil } -// recurseParentModels walks the models hierarchy to find the parentName and field details of the model for disciminated types +// findModelAncestry walks the models hierarchy to find all ancestors of the specified model for discriminated types. +func (c modelsTemplater) findModelAncestry(data GeneratorData, parentModelName, typeHint string) (*models.SDKField, *[]string, error) { + if data.recurseParentModels { + field, model, err := c.recurseParentModels(data, parentModelName, typeHint) + if err != nil { + return nil, nil, err + } + return field, &[]string{*model}, nil + } + + parentModel, ok := data.models[parentModelName] + if !ok { + return nil, nil, fmt.Errorf("the parent model %q for model %q was not found", parentModelName, c.name) + } + + ancestors := []string{parentModelName} + var field *models.SDKField + + if parentField, ok := parentModel.Fields[typeHint]; ok { + field = &parentField + } + + if parentModel.ParentTypeName != nil { + parentField, older, err := c.findModelAncestry(data, *parentModel.ParentTypeName, typeHint) + if err != nil { + return nil, nil, err + } + if field == nil { + field = parentField + } + ancestors = append(ancestors, *older...) + } + + return field, &ancestors, nil +} + +// recurseParentModels walks the models hierarchy to find the parentName and field details of the model for discriminated types // This is a temporary measure until we update the swagger importer to connect the model fields inheritance for multiple parents. // Tracked at: https://github.com/hashicorp/pandora/issues/1235 func (c modelsTemplater) recurseParentModels(data GeneratorData, model string, typeHint string) (*models.SDKField, *string, error) { @@ -866,3 +948,20 @@ func (c modelsTemplater) recurseParentModels(data GeneratorData, model string, t return &field, &model, nil } + +func (c modelsTemplater) fieldIsOptional(data GeneratorData, fieldDetails models.SDKField) (isOptional bool) { + if fieldDetails.Optional { + isOptional = true + + // If the immediate (not top-level) object definition is a Reference to a Parent it's Optional + // by default since Parent types are output as an interface (which is implied nullable) + if fieldDetails.ObjectDefinition.Type == models.ReferenceSDKObjectDefinitionType { + model, ok := data.models[*fieldDetails.ObjectDefinition.ReferenceName] + if ok && model.FieldNameContainingDiscriminatedValue != nil && model.ParentTypeName == nil { + isOptional = false + } + } + } + + return +} From 5e92c96dfd6e87a828d066a8eef4c3e48f48eec3 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 4 Sep 2024 21:57:41 +0100 Subject: [PATCH 091/117] importer-msgraph-metadata: new! improved! method naming --- .../components/normalize/normalize_test.go | 16 ++++ .../components/parser/resourceids.go | 39 ++++++--- .../internal/pipeline/task_parse_resources.go | 86 ++++++++++++------- .../pipeline/task_translate_service.go | 2 +- 4 files changed, 96 insertions(+), 47 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize_test.go b/tools/importer-msgraph-metadata/components/normalize/normalize_test.go index 388f7d4f75c..a11404ff37d 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize_test.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize_test.go @@ -4,6 +4,7 @@ import "testing" func TestSingularize(t *testing.T) { testCases := map[string]string{ + // Plurals to be singularized "Access": "Access", "Apps": "App", "Applications": "Application", @@ -27,6 +28,13 @@ func TestSingularize(t *testing.T) { "Sortby": "Sortby", "Success": "Success", "Successes": "Success", + + // Singulars to remain the same + "Application": "Application", + "Group": "Group", + "Property": "Property", + "Ref": "Ref", + "Reference": "Reference", } for input, expected := range testCases { @@ -39,6 +47,7 @@ func TestSingularize(t *testing.T) { func TestPluralize(t *testing.T) { testCases := map[string]string{ + // Singulars to be pluralized "Access": "Accesses", "App": "Apps", "Application": "Applications", @@ -62,6 +71,13 @@ func TestPluralize(t *testing.T) { "Sortby": "Sortby", "Success": "Successes", "Successes": "Successes", + + // Plurals to remain the same + "Applications": "Applications", + "Groups": "Groups", + "Properties": "Properties", + "Refs": "Refs", + "References": "References", } for input, expected := range testCases { diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 1da8aec926a..0ea8c7283ed 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -84,7 +84,7 @@ func (r ResourceId) DataApiSdkResourceId() (*sdkModels.ResourceID, error) { case SegmentAction, SegmentCast, SegmentFunction, SegmentLabel, SegmentODataReference: sdkSegments = append(sdkSegments, sdkModels.NewStaticValueResourceIDSegment(segment.Value, segment.Value)) case SegmentUserValue: - sdkSegments = append(sdkSegments, sdkModels.NewUserSpecifiedResourceIDSegment(normalize.CleanNameCamel(*segment.Field), segment.Value)) + sdkSegments = append(sdkSegments, sdkModels.NewUserSpecifiedResourceIDSegment(normalize.CleanNameCamel(*segment.field), segment.Value)) default: return nil, fmt.Errorf("unknown segment type %q at index %d for resource ID: %q", segment.Type, i, r.Name) } @@ -160,13 +160,18 @@ func (r ResourceId) FullyQualifiedResourceName(suffixQualification *string) (*st } } - if len(r.Segments) > i+1 && r.Segments[i+1].Type == SegmentODataReference && (r.Segments[i+1].Value == "$count" || r.Segments[i+1].Value == "$ref") { - // $count and $ref indicate a plural entity (noting that this only applies when the current + if len(r.Segments) > i+1 && r.Segments[i+1].Type == SegmentODataReference && (r.Segments[i+1].Value == "$count") { + // $count indicates a plural entity (noting that this only applies when the current // segment is a label and not user-specified). shouldSingularize = false } } + // Explicitly pluralize a $ref segment when the previous segment was a label and plural + if segment.Type == SegmentODataReference && segment.Value == "$ref" && r.Segments[i-1].plural { + newName = normalize.Pluralize(newName) + } + // Note we intentionally match verbs on any segment type, not just SegmentTypeAction if v, ok := normalize.Verbs.Match(newName); ok && verb == "" { verb = *v @@ -333,7 +338,12 @@ func (r ResourceId) TruncateToLastSegmentOfTypeBeforeSegment(types []ResourceIdS type ResourceIdSegment struct { Type ResourceIdSegmentType Value string - Field *string + + // indicates the name to use when converting a SegmentUserValue to an sdkModels.ResourceIDSegment + field *string + + // indicates whether the original value for a SegmentLabel was a plural + plural bool } type ResourceIdSegmentType string @@ -371,13 +381,13 @@ func NewResourceId(path string, tags []string) (id ResourceId) { segment = ResourceIdSegment{ Type: SegmentUserValue, Value: fmt.Sprintf("{%s}", value), - Field: &field, + field: &field, } } else if strings.Contains(s, "(") { segment = ResourceIdSegment{ Type: SegmentFunction, Value: s, - Field: nil, + field: nil, } } else if strings.HasPrefix(strings.ToLower(s), "microsoft.graph.") || strings.HasPrefix(strings.ToLower(s), "graph.") { if tagSuffix(".actions") { @@ -394,38 +404,39 @@ func NewResourceId(path string, tags []string) (id ResourceId) { segment = ResourceIdSegment{ Type: SegmentAction, Value: value, - Field: nil, + field: nil, } } else { segment = ResourceIdSegment{ Type: SegmentCast, Value: s, - Field: nil, + field: nil, } } } else if strings.HasPrefix(s, "$") { segment = ResourceIdSegment{ Type: SegmentODataReference, Value: s, - Field: nil, + field: nil, } } else if i == len(segments)-1 && tagSuffix(".actions") { segment = ResourceIdSegment{ Type: SegmentAction, Value: s, - Field: nil, + field: nil, } } else if i == len(segments)-1 && tagSuffix(".functions") { segment = ResourceIdSegment{ Type: SegmentFunction, Value: s, - Field: nil, + field: nil, } } else { segment = ResourceIdSegment{ - Type: SegmentLabel, - Value: s, - Field: nil, + Type: SegmentLabel, + Value: s, + field: nil, + plural: normalize.Pluralize(s) == s, } } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 03600e7dfd9..1c046bd4582 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -63,6 +63,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model continue } + // Determine the resource name resourceName := "" if r, ok := parsedPath.FullyQualifiedResourceName(pointer.To(parser.ResourceSuffix)); ok { resourceName = *r @@ -101,38 +102,41 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model resources[resourceName].Paths = append(resources[resourceName].Paths, parsedPath) } - for method, operation := range operations { - if !tags.Matches(p.service, operation.Tags) { - continue + // Determine resource ID and/or URI suffix + var resourceId *parser.ResourceId + var uriSuffix *string + match, ok := resourceIds.MatchIdOrAncestor(parsedPath) + if ok { + if match.Id != nil { + resourceId = match.Id } - - // Determine resource ID and/or URI suffix - var resourceId *parser.ResourceId - var uriSuffix *string - match, ok := resourceIds.MatchIdOrAncestor(parsedPath) - if ok { - if match.Id != nil { - resourceId = match.Id + if match.Remainder != nil && len(match.Remainder.Segments) > 0 { + uriSuffix = pointer.To(match.Remainder.ID()) + + // When last segment is not a label (e.g. an action, function or cast), adopt the parent resource category, + // but only if the suffix has one segment, else this could indicate a different parent, in which case + // we'll attempt a match after parsing all resources. + if resourceCategory == "" && len(match.Remainder.Segments) == 1 && + (match.Remainder.Segments[0].Type == parser.SegmentAction || match.Remainder.Segments[0].Type == parser.SegmentCast || match.Remainder.Segments[0].Type == parser.SegmentFunction) { + resourceCategory = parsedPath.ID() } - if match.Remainder != nil && len(match.Remainder.Segments) > 0 { - uriSuffix = pointer.To(match.Remainder.ID()) - - // When last segment is not a label (e.g. an action, function or cast), adopt the parent resource category, - // but only if the suffix has one segment, else this could indicate a different parent, in which case - // we'll attempt a match after parsing all resources. - if resourceCategory == "" && strings.Count(*uriSuffix, "/") == 1 { - resourceCategory = resourceId.Name - } - } - } else { - uriSuffix = &path } + } else { + uriSuffix = pointer.To(parsedPath.ID()) + } - if uriSuffix != nil { - if uriSuffixParsed := parser.NewResourceId(*uriSuffix, operationTags); uriSuffixParsed.HasUserValue() { - logging.Infof(fmt.Sprintf("Skipping URI suffix containing user value in resource %q (category %q, service %q, version %q): %q", resourceName, resourceCategory, p.service, p.apiVersion, *uriSuffix)) - continue - } + // Skip this path when the uriSuffix contains a user-specified value. This should ordinarily not occur, but + // can happen when no parent resource ID has been matched, for example if a resource ID was blacklisted. + if uriSuffix != nil { + if uriSuffixParsed := parser.NewResourceId(*uriSuffix, operationTags); uriSuffixParsed.HasUserValue() { + logging.Infof(fmt.Sprintf("Skipping URI suffix containing user value in resource %q (category %q, service %q, version %q): %q", resourceName, resourceCategory, p.service, p.apiVersion, *uriSuffix)) + continue + } + } + + for method, operation := range operations { + if !tags.Matches(p.service, operation.Tags) { + continue } listOperation := false @@ -291,25 +295,38 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model paginationField = pointer.To("@odata.nextLink") } - // Skip unknown operations + // Skip unknown operation types if operationType == parser.OperationTypeUnknown { logging.Warnf(fmt.Sprintf("Skipping unknown operation type for %q: %v", p.service, path)) continue } + // Determine the base name of the operation, attempting to trim a leading service name since that is redundant operationName := "" - prefixToTrim := normalize.Singularize(normalize.CleanName(p.service)) + prefixesToTrim := []string{ + normalize.CleanName(p.service), + normalize.Singularize(normalize.CleanName(p.service)), + } if resourceId != nil && uriSuffix == nil { - prefixToTrim = fmt.Sprintf("%s%s", prefixToTrim, parser.ResourceSuffix) + for i := range prefixesToTrim { + prefixesToTrim[i] = fmt.Sprintf("%s%s", prefixesToTrim[i], parser.ResourceSuffix) + } + } + shortResourceName := resourceName + for _, prefixToTrim := range prefixesToTrim { + shortResourceName = strings.TrimPrefix(shortResourceName, prefixToTrim) } - shortResourceName := strings.TrimPrefix(resourceName, prefixToTrim) operationName = shortResourceName if len(operationName) == 0 { operationName = resourceName } + // Now qualify the operation name based on the type of operation. Additionally, if the operation name + // matches a known verb, move that verb to the beginning and use it instead of a standard verb. + // For example, a Create operation for "Application" should get an operation name of "CreateApplication", + // but a Create operation for "ApplicationTemplateInstantiate" should get an operation name of "InstantiateApplicationTemplate" switch operationType { case parser.OperationTypeList: if _, ok = normalize.Verbs.Match(shortResourceName); ok { @@ -317,8 +334,10 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } else { operationName = fmt.Sprintf("List%s", normalize.Pluralize(normalize.Singularize(operationName))) } + case parser.OperationTypeRead: operationName = fmt.Sprintf("Get%s", normalize.Singularize(operationName)) + case parser.OperationTypeCreate: if _, ok = normalize.Verbs.Match(shortResourceName); ok { operationName = normalize.Singularize(operationName) @@ -327,10 +346,13 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } else { operationName = fmt.Sprintf("Create%s", normalize.Singularize(operationName)) } + case parser.OperationTypeCreateUpdate: operationName = fmt.Sprintf("CreateUpdate%s", normalize.Singularize(operationName)) + case parser.OperationTypeUpdate: operationName = fmt.Sprintf("Update%s", normalize.Singularize(operationName)) + case parser.OperationTypeDelete: if lastSegment.Type == parser.SegmentODataReference { operationName = fmt.Sprintf("Remove%s", normalize.Singularize(operationName)) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index ea56c2d6697..92175668474 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -30,7 +30,7 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv continue } - // First scaffold the version and resources (categories) + // First scaffold the version and SDK resources (categories) if _, ok := sdkService.APIVersions[resource.Version]; !ok { sdkService.APIVersions[resource.Version] = sdkModels.APIVersion{ APIVersion: resource.Version, From 230f9a9fc5556aefef8663496229b5e1af4cd278 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Wed, 4 Sep 2024 22:14:44 +0100 Subject: [PATCH 092/117] generator-go-sdk: update test fixtures --- .../templater_models_discriminators_test.go | 40 +++++++++---------- .../generator/templater_models_test.go | 2 +- 2 files changed, 21 insertions(+), 21 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index 825153f7542..f9fb6f4baae 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -75,29 +75,30 @@ import ( // acctests licence placeholder -type ModeOfTransitBase struct { - Type string ''json:"type"'' +type ModeOfTransit interface { + ModeOfTransit() BaseModeOfTransitImpl } -type ModeOfTransit interface { - ModeOfTransit() ModeOfTransitBase +type BaseModeOfTransitImpl struct { + Type string ''json:"type"'' } -// RawModeOfTransitImpl is returned when the Discriminated Value -// doesn't match any of the defined types +var _ ModeOfTransit = RawModeOfTransitImpl{} + +// RawModeOfTransitImpl is returned when the Discriminated Value doesn't match any of the defined types // NOTE: this should only be used when a type isn't defined for this type of Object (as a workaround) // and is used only for Deserialization (e.g. this cannot be used as a Request Payload). type RawModeOfTransitImpl struct { - modeOfTransit ModeOfTransitBase + modeOfTransit BaseModeOfTransitImpl Type string Values map[string]interface{} } -func (s RawModeOfTransitImpl) ModeOfTransit() ModeOfTransitBase { +func (s RawModeOfTransitImpl) ModeOfTransit() BaseModeOfTransitImpl { return s.modeOfTransit } -func unmarshalModeOfTransitImplementation(input []byte) (ModeOfTransit, error) { +func UnmarshalModeOfTransitImplementation(input []byte) (ModeOfTransit, error) { if input == nil { return nil, nil } @@ -128,17 +129,16 @@ func unmarshalModeOfTransitImplementation(input []byte) (ModeOfTransit, error) { return out, nil } - var parent ModeOfTransitBase + var parent BaseModeOfTransitImpl if err := json.Unmarshal(input, &parent); err != nil { - return nil, fmt.Errorf("unmarshaling into ModeOfTransitBase: %+v", err) + return nil, fmt.Errorf("unmarshaling into BaseModeOfTransitImpl: %+v", err) } - out := RawModeOfTransitImpl{ + return RawModeOfTransitImpl{ modeOfTransit: parent, Type: value, Values: temp, - } - return out, nil + }, nil } `, "''", "`") assertTemplatedCodeMatches(t, expected, *actual) @@ -246,8 +246,8 @@ type Train struct { Type string ''json:"type"'' } -func (s Train) ModeOfTransit() ModeOfTransitBase { - return ModeOfTransitBase{ +func (s Train) ModeOfTransit() BaseModeOfTransitImpl { + return BaseModeOfTransitImpl{ Name: s.Name, Type: s.Type, } @@ -264,7 +264,7 @@ func (s Train) MarshalJSON() ([]byte, error) { } var decoded map[string]interface{} - if err := json.Unmarshal(encoded, &decoded); err != nil { + if err = json.Unmarshal(encoded, &decoded); err != nil { return nil, fmt.Errorf("unmarshaling Train: %+v", err) } decoded["type"] = "train" @@ -485,8 +485,8 @@ type FirstImplementation struct { Type string ''json:"type"'' } -func (s FirstImplementation) First() FirstBase { - return FirstBase{ +func (s FirstImplementation) First() BaseFirstImpl { + return BaseFirstImpl{ Type: s.Type, } } @@ -502,7 +502,7 @@ func (s FirstImplementation) MarshalJSON() ([]byte, error) { } var decoded map[string]interface{} - if err := json.Unmarshal(encoded, &decoded); err != nil { + if err = json.Unmarshal(encoded, &decoded); err != nil { return nil, fmt.Errorf("unmarshaling FirstImplementation: %+v", err) } decoded["type"] = "first" diff --git a/tools/generator-go-sdk/internal/generator/templater_models_test.go b/tools/generator-go-sdk/internal/generator/templater_models_test.go index 763ca656fba..5ed08617cb7 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_test.go @@ -370,7 +370,7 @@ func (s Example) MarshalJSON() ([]byte, error) { } var decoded map[string]interface{} - if err := json.Unmarshal(encoded, &decoded); err != nil { + if err = json.Unmarshal(encoded, &decoded); err != nil { return nil, fmt.Errorf("unmarshaling Example: %+v", err) } From 7a2417e0ae693ff2ee82f8c6c93605a99ad1cb27 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 5 Sep 2024 00:03:45 +0100 Subject: [PATCH 093/117] importer-msgraph-metadata: fix clobbering bug in method names for delete operations --- .../components/parser/resourceids.go | 1 + .../internal/pipeline/task_parse_resources.go | 4 ++-- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 0ea8c7283ed..c0733caa4dc 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -170,6 +170,7 @@ func (r ResourceId) FullyQualifiedResourceName(suffixQualification *string) (*st // Explicitly pluralize a $ref segment when the previous segment was a label and plural if segment.Type == SegmentODataReference && segment.Value == "$ref" && r.Segments[i-1].plural { newName = normalize.Pluralize(newName) + shouldSingularize = false } // Note we intentionally match verbs on any segment type, not just SegmentTypeAction diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 1c046bd4582..d502cbb7b08 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -355,9 +355,9 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model case parser.OperationTypeDelete: if lastSegment.Type == parser.SegmentODataReference { - operationName = fmt.Sprintf("Remove%s", normalize.Singularize(operationName)) + operationName = fmt.Sprintf("Remove%s", operationName) } else { - operationName = fmt.Sprintf("Delete%s", normalize.Singularize(operationName)) + operationName = fmt.Sprintf("Delete%s", operationName) } } From bd2b9e88cb569e049fc0adf2a62fc60a2b923935 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 5 Sep 2024 00:44:49 +0100 Subject: [PATCH 094/117] generator-go-sdk: `Base{Model}Impl` structs should implement their own model interface --- .../internal/generator/templater_models.go | 18 +++++++++++++++++- 1 file changed, 17 insertions(+), 1 deletion(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 8e978702802..0fcf91b2aab 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -182,7 +182,7 @@ type %[1]s interface { `, c.name, strings.Join(interfaceLines, "\n")) } - // Output a model struct + // Format the model struct field lines formattedStructLines := make([]string, 0) for i, v := range structLines { if strings.HasPrefix(strings.TrimSpace(v), "//") { @@ -196,6 +196,12 @@ type %[1]s interface { formattedStructLines = append(formattedStructLines, v) } + // When the struct name doesn't match the model name, the struct should implement the model interface + if structName != c.name { + parentAssignmentInfo = fmt.Sprintf("var _ %[1]s = %[2]s{}", c.name, structName) + } + + // Output the model struct out += fmt.Sprintf(` %[3]s type %[1]s struct { @@ -203,6 +209,16 @@ type %[1]s struct { } `, structName, strings.Join(formattedStructLines, "\n"), parentAssignmentInfo) + // When the struct name doesn't match the model name, output a method to satisfy the model interface + if structName != c.name { + out += fmt.Sprintf(` + +func (s %[1]s) %[2]s() %[1]s { + return s +} +`, structName, c.name) + } + parentModelFunctions, err := c.codeForParentStructFunctions(data) if err != nil { return nil, fmt.Errorf("generating parent model functions: %+v", err) From 483014afab9ce6c440d0cb851babcd631ef75d98 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 5 Sep 2024 14:40:27 +0100 Subject: [PATCH 095/117] importer-msgraph-metadata: ensure leading service name is trimmed from operation names when using a recognised verb --- .../components/normalize/normalize.go | 75 +++++-------------- .../components/normalize/verbs.go | 43 +++++++++++ .../internal/pipeline/task_parse_resources.go | 10 ++- 3 files changed, 71 insertions(+), 57 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/normalize/verbs.go diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/normalize.go index b9f069115ff..349377d0787 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/normalize.go @@ -4,7 +4,6 @@ package normalize import ( - "fmt" "regexp" "strings" @@ -13,6 +12,22 @@ import ( "golang.org/x/text/language" ) +type plural struct { + singular string + plural string +} + +var pluralExceptions = []plural{ + {"By", "By"}, + {"Cache", "Caches"}, + {"Compatibility", "Compatibility"}, + {"Data", "Data"}, + {"Metadata", "Metadata"}, + {"Orderby", "Orderby"}, + {"Premise", "Premises"}, + {"Sortby", "Sortby"}, +} + func Singularize(name string) string { for _, v := range pluralExceptions { if name == v.plural { @@ -64,7 +79,7 @@ func CleanName(name string) string { // Innererror should be InnerError name = regexp.MustCompile("^Innererror").ReplaceAllString(name, "InnerError") - // Ip[A-Zv] should be IP[A-Zv] + // Ip[A-Zv] should be IP[A-Zv] (e.g. IpAddress, Ipv6) name = regexp.MustCompile("Ip([A-Zv])").ReplaceAllString(name, "IP${1}") // Cidr should be CIDR @@ -79,6 +94,9 @@ func CleanName(name string) string { // Orderby should be OrderBy name = regexp.MustCompile("^Orderby").ReplaceAllString(name, "OrderBy") + // Sortby should be SortBy + name = regexp.MustCompile("^Sortby").ReplaceAllString(name, "SortBy") + // (trailing) ID should be Id for compatibility with Go SDK generator name = regexp.MustCompile("([a-z])ID$").ReplaceAllString(name, "${1}Id") @@ -89,56 +107,3 @@ func CleanNameCamel(name string) string { name = CleanName(name) return strings.ToLower(name[0:1]) + name[1:] } - -type operationVerbs []string - -func (ov operationVerbs) Match(operation string) (*string, bool) { - for _, v := range ov { - if regexp.MustCompile(fmt.Sprintf("^%s$", v)).MatchString(operation) { - return &v, true - } - if regexp.MustCompile(fmt.Sprintf("^%s[A-Z]", v)).MatchString(operation) { - return &v, true - } - } - return nil, false -} - -var Verbs = operationVerbs{ - "Acquire", - "Add", - "Assign", - "Check", - "Discover", - "Get", - "Instantiate", - "Parse", - "Pause", - "Provision", - "Remove", - "Renew", - "Restart", - "Restore", - "Set", - "Start", - "Stop", - "Unset", - "Update", - "Validate", -} - -type plural struct { - singular string - plural string -} - -var pluralExceptions = []plural{ - {"By", "By"}, - {"Compatibility", "Compatibility"}, - {"Cache", "Caches"}, - {"Data", "Data"}, - {"Metadata", "Metadata"}, - {"Orderby", "Orderby"}, - {"Premise", "Premises"}, - {"Sortby", "Sortby"}, -} diff --git a/tools/importer-msgraph-metadata/components/normalize/verbs.go b/tools/importer-msgraph-metadata/components/normalize/verbs.go new file mode 100644 index 00000000000..0a8392aa08e --- /dev/null +++ b/tools/importer-msgraph-metadata/components/normalize/verbs.go @@ -0,0 +1,43 @@ +package normalize + +import ( + "fmt" + "regexp" +) + +type operationVerbs []string + +func (ov operationVerbs) Match(operation string) (*string, bool) { + for _, v := range ov { + if regexp.MustCompile(fmt.Sprintf("^%s$", v)).MatchString(operation) { + return &v, true + } + if regexp.MustCompile(fmt.Sprintf("^%s[A-Z]", v)).MatchString(operation) { + return &v, true + } + } + return nil, false +} + +var Verbs = operationVerbs{ + "Acquire", + "Add", + "Assign", + "Check", + "Discover", + "Get", + "Instantiate", + "Parse", + "Pause", + "Provision", + "Remove", + "Renew", + "Restart", + "Restore", + "Set", + "Start", + "Stop", + "Unset", + "Update", + "Validate", +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index d502cbb7b08..6405fdf1231 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -301,7 +301,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model continue } - // Determine the base name of the operation, attempting to trim a leading service name since that is redundant + // Determine the base name of the operation, whilst attempting to trim a leading service name since that is redundant operationName := "" prefixesToTrim := []string{ @@ -313,9 +313,15 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model prefixesToTrim[i] = fmt.Sprintf("%s%s", prefixesToTrim[i], parser.ResourceSuffix) } } + shortResourceName := resourceName for _, prefixToTrim := range prefixesToTrim { - shortResourceName = strings.TrimPrefix(shortResourceName, prefixToTrim) + if verb, ok := normalize.Verbs.Match(shortResourceName); ok { + // resource name starts with a standard verb, so remove it before trimming a prefix + shortResourceName = *verb + strings.TrimPrefix(strings.TrimPrefix(shortResourceName, *verb), prefixToTrim) + } else { + shortResourceName = strings.TrimPrefix(shortResourceName, prefixToTrim) + } } operationName = shortResourceName From ff34b34bfa941f26778682ac00f4e7ecf0bcb1a3 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 5 Sep 2024 14:44:59 +0100 Subject: [PATCH 096/117] generator-go-sdk: update test fixtures --- .../generator/templater_models_discriminators_test.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index f9fb6f4baae..4daf25c2f9a 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -79,10 +79,16 @@ type ModeOfTransit interface { ModeOfTransit() BaseModeOfTransitImpl } +var _ ModeOfTransit = BaseModeOfTransitImpl{} + type BaseModeOfTransitImpl struct { Type string ''json:"type"'' } +func (s BaseModeOfTransitImpl) ModeOfTransit() BaseModeOfTransitImpl { + return s +} + var _ ModeOfTransit = RawModeOfTransitImpl{} // RawModeOfTransitImpl is returned when the Discriminated Value doesn't match any of the defined types From 6cef487cad7b726a9941b7c9d6f68e4abe3a1f69 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 5 Sep 2024 23:45:43 +0100 Subject: [PATCH 097/117] importer-msgraph-metadata: add service workaround to add missing GET method for synchronization secrets --- .../workarounds/workaround_application.go | 2 +- .../workaround_conditionalaccesspolicy.go | 2 +- .../workarounds/workaround_odata_bind.go | 2 +- ...rkaround_repeating_resource_id_segments.go | 2 +- .../workaround_synchronizationsecrets.go | 78 +++++++++++++++++++ .../components/workarounds/workarounds.go | 30 ++++++- .../internal/pipeline/importer.go | 6 +- 7 files changed, 116 insertions(+), 6 deletions(-) create mode 100644 tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go index 4df7fed50c7..8d288436b96 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_application.go @@ -11,7 +11,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" ) -var _ workaround = workaroundApplication{} +var _ dataWorkaround = workaroundApplication{} // workaroundApplication works around missing fields in the Application model for the beta API. // 1. Missing `oauth2RequirePostResponse` field. diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index 86c1b584267..fea8279da61 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -10,7 +10,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" ) -var _ workaround = workaroundConditionalAccessPolicy{} +var _ dataWorkaround = workaroundConditionalAccessPolicy{} // workaroundConditionalAccessPolicy adds missing fields and fixes some field types. type workaroundConditionalAccessPolicy struct{} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go index 79bbcc53723..f8ed087bc4a 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go @@ -13,7 +13,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -var _ workaround = workaroundODataBind{} +var _ dataWorkaround = workaroundODataBind{} // workaroundODataBind inserts an `@odata.bind` field where a field or collection refers to a DirectoryObject. The // OpenAPI spec unfortunately does not document relationships between entities. diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go index afbdc577aa7..b7e4136e0a3 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go @@ -8,7 +8,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -var _ workaround = workaroundRepeatingResourceIdSegments{} +var _ dataWorkaround = workaroundRepeatingResourceIdSegments{} // workaroundRepeatingResourceIdSegments removes incompatible resource IDs due to repeating segments which are not supported at this time. type workaroundRepeatingResourceIdSegments struct{} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go new file mode 100644 index 00000000000..76a552b2c80 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go @@ -0,0 +1,78 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: MPL-2.0 + +package workarounds + +import ( + "fmt" + "net/http" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" + "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" +) + +var _ serviceWorkaround = workaroundSynchronizationSecrets{} + +// workaroundSynchronizationSecrets adds a missing GET method for synchronization secrets, which is absent from upstream specs. +type workaroundSynchronizationSecrets struct{} + +func (workaroundSynchronizationSecrets) Name() string { + return "Synchronization Secrets / add missing get method" +} + +func (workaroundSynchronizationSecrets) Process(apiVersion, serviceName string, resources parser.Resources, resourceIds parser.ResourceIds) error { + serviceNamesToPaths := map[string]string{ + "servicePrincipals": "/servicePrincipals/{servicePrincipal-id}/synchronization/secrets", + } + + for serviceNameToMatch, path := range serviceNamesToPaths { + if serviceNameToMatch != serviceName { + return nil + } + + resourceName := fmt.Sprintf("%sSynchronizationSecret", normalize.Singularize(normalize.CleanName(serviceName))) + resource, ok := resources[resourceName] + if !ok { + return fmt.Errorf("%q was not found for the service %q", resourceName, serviceName) + } + + tags := []string{serviceName + ".synchronization"} + + var resourceId *parser.ResourceId + var uriSuffix *string + + parsedPath := parser.NewResourceId(path, tags) + match, ok := resourceIds.MatchIdOrAncestor(parsedPath) + if ok { + if match.Id != nil { + resourceId = match.Id + } + if match.Remainder != nil && len(match.Remainder.Segments) > 0 { + uriSuffix = pointer.To(match.Remainder.ID()) + } + } else { + uriSuffix = pointer.To(parsedPath.ID()) + } + + resource.Operations = append(resource.Operations, parser.Operation{ + Name: "GetSynchronizationSecret", + Description: "Retrieve synchronization secrets.", + Type: parser.OperationTypeRead, + Method: http.MethodGet, + ResourceId: resourceId, + UriSuffix: uriSuffix, + Responses: parser.Responses{ + { + Status: http.StatusOK, + ContentType: pointer.To("application/json"), + ReferenceName: pointer.To("microsoft.graph.synchronizationSecret"), + Type: pointer.To(parser.DataTypeReference), + }, + }, + Tags: tags, + }) + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index be9f0d1d827..305d5061cd6 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -10,7 +10,7 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -var workarounds = []workaround{ +var workarounds = []dataWorkaround{ workaroundODataBind{}, workaroundRepeatingResourceIdSegments{}, @@ -18,14 +18,29 @@ var workarounds = []workaround{ workaroundConditionalAccessPolicy{}, } +var serviceWorkarounds = []serviceWorkaround{ + workaroundSynchronizationSecrets{}, +} + type workaround interface { // Name returns the Service Name and associated Pull Request number Name() string +} + +type dataWorkaround interface { + workaround // Process performs any necessary fixes to constants, models and/or resource IDs Process(string, parser.Models, parser.Constants, parser.ResourceIds) error } +type serviceWorkaround interface { + workaround + + // Process performs any necessary fixes to resources + Process(string, string, parser.Resources, parser.ResourceIds) error +} + // ApplyWorkarounds invokes the specified workarounds for models, constants and resource func ApplyWorkarounds(apiVersion string, models parser.Models, constants parser.Constants, resourceIds parser.ResourceIds) error { logging.Tracef("Processing Data Workarounds..") @@ -38,3 +53,16 @@ func ApplyWorkarounds(apiVersion string, models parser.Models, constants parser. return nil } + +// ApplyWorkaroundsForService invokes the specified workarounds for a given service +func ApplyWorkaroundsForService(apiVersion, serviceName string, resources parser.Resources, resourceIds parser.ResourceIds) error { + logging.Tracef("Processing Service Workarounds for %s..", serviceName) + for _, fix := range serviceWorkarounds { + logging.Tracef("Applying Service Workaround %q", fix.Name()) + if err := fix.Process(apiVersion, serviceName, resources, resourceIds); err != nil { + return fmt.Errorf("applying Service Workaround %q: %v", fix.Name(), err) + } + } + + return nil +} diff --git a/tools/importer-msgraph-metadata/internal/pipeline/importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go index ae2bfcfe3a7..5eaeef31a63 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/importer.go @@ -161,7 +161,10 @@ func (p pipelineForService) RunImport() error { return nil } - p.resources[p.service] = resources + // Apply workarounds + if err = workarounds.ApplyWorkaroundsForService(p.apiVersion, p.service, resources, p.resourceIds); err != nil { + return err + } // Consistency checks for discovered resources for resourceName, resource := range resources { @@ -177,6 +180,7 @@ func (p pipelineForService) RunImport() error { } } + p.resources[p.service] = resources return nil } From e1126b99ceef4130f48bb4aabdf5cf1bce721e99 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 6 Sep 2024 01:03:21 +0100 Subject: [PATCH 098/117] importer-msgraph-metadata: fix up response model for synchronization secrets service workaround --- .../workaround_synchronizationsecrets.go | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go index 76a552b2c80..2a93d4276af 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go @@ -56,17 +56,18 @@ func (workaroundSynchronizationSecrets) Process(apiVersion, serviceName string, } resource.Operations = append(resource.Operations, parser.Operation{ - Name: "GetSynchronizationSecret", - Description: "Retrieve synchronization secrets.", - Type: parser.OperationTypeRead, - Method: http.MethodGet, - ResourceId: resourceId, - UriSuffix: uriSuffix, + Name: "ListSynchronizationSecrets", + Description: "Retrieve synchronization secrets.", + Type: parser.OperationTypeList, + Method: http.MethodGet, + PaginationField: pointer.To("@odata.nextLink"), + ResourceId: resourceId, + UriSuffix: uriSuffix, Responses: parser.Responses{ { Status: http.StatusOK, ContentType: pointer.To("application/json"), - ReferenceName: pointer.To("microsoft.graph.synchronizationSecret"), + ReferenceName: pointer.To("microsoft.graph.synchronizationSecretKeyStringValuePair"), Type: pointer.To(parser.DataTypeReference), }, }, From 962737c83765f5a9f0fd3cdf1e9edd0dbc406c9f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 6 Sep 2024 13:06:28 +0100 Subject: [PATCH 099/117] importer-msgraph-metadata: add some more verbs found in `user` package --- .../components/normalize/verbs.go | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/normalize/verbs.go b/tools/importer-msgraph-metadata/components/normalize/verbs.go index 0a8392aa08e..baf6365803e 100644 --- a/tools/importer-msgraph-metadata/components/normalize/verbs.go +++ b/tools/importer-msgraph-metadata/components/normalize/verbs.go @@ -23,21 +23,32 @@ var Verbs = operationVerbs{ "Acquire", "Add", "Assign", + "Change", "Check", "Discover", + "Export", + "Find", "Get", "Instantiate", + "Invoke", + "Issue", "Parse", "Pause", + "Process", "Provision", "Remove", "Renew", "Restart", "Restore", + "Retry", + "Revoke", "Set", + "Send", "Start", "Stop", + "Translate", "Unset", "Update", "Validate", + "Wipe", } From e67b869694832a8e85f52beb538cb3038c0e100f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 6 Sep 2024 18:15:43 +0100 Subject: [PATCH 100/117] importer-msgraph-metadata: fix singularization & verb handling bugs, improve performance, add tests --- .../normalize/{normalize.go => cleanname.go} | 46 ----------- .../components/normalize/cleanname_test.go | 18 +++++ .../components/normalize/plurals.go | 63 +++++++++++++++ .../{normalize_test.go => plurals_test.go} | 76 ++++++++++++------- .../components/normalize/verbs.go | 76 +++++++++++++++---- .../components/normalize/verbs_test.go | 63 +++++++++++++++ 6 files changed, 254 insertions(+), 88 deletions(-) rename tools/importer-msgraph-metadata/components/normalize/{normalize.go => cleanname.go} (70%) create mode 100644 tools/importer-msgraph-metadata/components/normalize/cleanname_test.go create mode 100644 tools/importer-msgraph-metadata/components/normalize/plurals.go rename tools/importer-msgraph-metadata/components/normalize/{normalize_test.go => plurals_test.go} (54%) create mode 100644 tools/importer-msgraph-metadata/components/normalize/verbs_test.go diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize.go b/tools/importer-msgraph-metadata/components/normalize/cleanname.go similarity index 70% rename from tools/importer-msgraph-metadata/components/normalize/normalize.go rename to tools/importer-msgraph-metadata/components/normalize/cleanname.go index 349377d0787..676d3b2253c 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize.go +++ b/tools/importer-msgraph-metadata/components/normalize/cleanname.go @@ -1,59 +1,13 @@ -// Copyright (c) HashiCorp, Inc. -// SPDX-License-Identifier: MPL-2.0 - package normalize import ( "regexp" "strings" - "github.com/gertd/go-pluralize" "golang.org/x/text/cases" "golang.org/x/text/language" ) -type plural struct { - singular string - plural string -} - -var pluralExceptions = []plural{ - {"By", "By"}, - {"Cache", "Caches"}, - {"Compatibility", "Compatibility"}, - {"Data", "Data"}, - {"Metadata", "Metadata"}, - {"Orderby", "Orderby"}, - {"Premise", "Premises"}, - {"Sortby", "Sortby"}, -} - -func Singularize(name string) string { - for _, v := range pluralExceptions { - if name == v.plural { - return v.singular - } - } - - client := pluralize.NewClient() - output := client.Singular(name) - - return output -} - -func Pluralize(name string) string { - for _, v := range pluralExceptions { - if name == v.singular { - return v.plural - } - } - - client := pluralize.NewClient() - output := client.Plural(name) - - return output -} - func CleanName(name string) string { if name == "" { return name diff --git a/tools/importer-msgraph-metadata/components/normalize/cleanname_test.go b/tools/importer-msgraph-metadata/components/normalize/cleanname_test.go new file mode 100644 index 00000000000..9419ef22bf8 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/normalize/cleanname_test.go @@ -0,0 +1,18 @@ +package normalize + +import "testing" + +func TestCleanName(t *testing.T) { + testCases := map[string]string{ + "CloudPc": "CloudPC", + "Ref": "Ref", + "Reference": "Reference", + } + + for input, expected := range testCases { + result := CleanName(input) + if result != expected { + t.Errorf("CleanName(%q): got %q, expected %q", input, result, expected) + } + } +} diff --git a/tools/importer-msgraph-metadata/components/normalize/plurals.go b/tools/importer-msgraph-metadata/components/normalize/plurals.go new file mode 100644 index 00000000000..903b7b2c51d --- /dev/null +++ b/tools/importer-msgraph-metadata/components/normalize/plurals.go @@ -0,0 +1,63 @@ +package normalize + +import ( + "fmt" + "regexp" + + "github.com/gertd/go-pluralize" +) + +type plural struct { + singular string + plural string +} + +var pluralExceptions = []plural{ + {"By", "By"}, + {"Cache", "Caches"}, + {"Compatibility", "Compatibility"}, + {"Data", "Data"}, + {"Metadata", "Metadata"}, + {"Orderby", "Orderby"}, + {"Premise", "Premises"}, + {"Sortby", "Sortby"}, +} + +var singularMatchers = map[string]*regexp.Regexp{} +var pluralMatchers = map[string]*regexp.Regexp{} + +func init() { + singularMatchers = make(map[string]*regexp.Regexp) + pluralMatchers = make(map[string]*regexp.Regexp) + + for _, exception := range pluralExceptions { + singularMatchers[exception.singular] = regexp.MustCompile(fmt.Sprintf("^(.*[a-z]?)%s$", exception.plural)) + pluralMatchers[exception.plural] = regexp.MustCompile(fmt.Sprintf("^(.*[a-z]?)%s$", exception.singular)) + } +} + +func Singularize(name string) string { + for _, exception := range pluralExceptions { + if singularMatchers[exception.singular].MatchString(name) { + return singularMatchers[exception.singular].ReplaceAllString(name, "$1") + exception.singular + } + } + + client := pluralize.NewClient() + output := client.Singular(name) + + return output +} + +func Pluralize(name string) string { + for _, exception := range pluralExceptions { + if pluralMatchers[exception.plural].MatchString(name) { + return pluralMatchers[exception.plural].ReplaceAllString(name, "$1") + exception.plural + } + } + + client := pluralize.NewClient() + output := client.Plural(name) + + return output +} diff --git a/tools/importer-msgraph-metadata/components/normalize/normalize_test.go b/tools/importer-msgraph-metadata/components/normalize/plurals_test.go similarity index 54% rename from tools/importer-msgraph-metadata/components/normalize/normalize_test.go rename to tools/importer-msgraph-metadata/components/normalize/plurals_test.go index a11404ff37d..dd31956ca63 100644 --- a/tools/importer-msgraph-metadata/components/normalize/normalize_test.go +++ b/tools/importer-msgraph-metadata/components/normalize/plurals_test.go @@ -5,36 +5,44 @@ import "testing" func TestSingularize(t *testing.T) { testCases := map[string]string{ // Plurals to be singularized + "Apps": "App", + "Applications": "Application", + "Caches": "Cache", + "Entities": "Entity", + "Modalities": "Modality", + "Options": "Option", + "Premises": "Premise", + "Principals": "Principal", + "Services": "Service", + "Severities": "Severity", + "Successes": "Success", + + // Composite plurals to be singularized + "FolderProperties": "FolderProperty", + "SecurityPolicies": "SecurityPolicy", + "ServerDetails": "ServerDetail", + "UserKinds": "UserKind", + + // Singulars to remain the same "Access": "Access", - "Apps": "App", - "Applications": "Application", + "Application": "Application", + "Compatibility": "Compatibility", "By": "By", - "Caches": "Cache", "Data": "Data", - "Details": "Detail", - "Compatibility": "Compatibility", - "Entities": "Entity", - "Kinds": "Kind", + "Group": "Group", "Metadata": "Metadata", - "Modalities": "Modality", - "Options": "Option", "Orderby": "Orderby", - "Policies": "Policy", - "Premises": "Premise", - "Principals": "Principal", - "Properties": "Property", - "Services": "Service", - "Severities": "Severity", + "Ref": "Ref", + "Reference": "Reference", "Sortby": "Sortby", - "Success": "Success", - "Successes": "Success", - // Singulars to remain the same - "Application": "Application", - "Group": "Group", - "Property": "Property", - "Ref": "Ref", - "Reference": "Reference", + // Composite singulars to remain the same + "OrderBy": "OrderBy", + "SecurityGroup": "SecurityGroup", + "UserData": "UserData", + + // Already a singular, mistakenly treated as plural + "Success": "Success", } for input, expected := range testCases { @@ -51,33 +59,43 @@ func TestPluralize(t *testing.T) { "Access": "Accesses", "App": "Apps", "Application": "Applications", - "By": "By", "Compatibility": "Compatibility", "Cache": "Caches", - "Data": "Data", "Detail": "Details", "Entity": "Entities", "Kind": "Kinds", - "Metadata": "Metadata", "Modality": "Modalities", "Option": "Options", - "Orderby": "Orderby", "Policy": "Policies", "Premise": "Premises", "Principal": "Principals", "Property": "Properties", "Service": "Services", "Severity": "Severities", - "Sortby": "Sortby", "Success": "Successes", - "Successes": "Successes", + + // Composite plurals to stay the same + "FolderProperty": "FolderProperties", + "SystemService": "SystemServices", + "UserDetail": "UserDetails", // Plurals to remain the same "Applications": "Applications", + "By": "By", + "Data": "Data", "Groups": "Groups", + "Metadata": "Metadata", + "Orderby": "Orderby", "Properties": "Properties", "Refs": "Refs", "References": "References", + "Sortby": "Sortby", + "Successes": "Successes", + + // Composite plurals to remain the same + "FolderProperties": "FolderProperties", + "OrderBy": "OrderBy", + "UserData": "UserData", } for input, expected := range testCases { diff --git a/tools/importer-msgraph-metadata/components/normalize/verbs.go b/tools/importer-msgraph-metadata/components/normalize/verbs.go index baf6365803e..ac1d365327a 100644 --- a/tools/importer-msgraph-metadata/components/normalize/verbs.go +++ b/tools/importer-msgraph-metadata/components/normalize/verbs.go @@ -7,48 +7,98 @@ import ( type operationVerbs []string -func (ov operationVerbs) Match(operation string) (*string, bool) { - for _, v := range ov { - if regexp.MustCompile(fmt.Sprintf("^%s$", v)).MatchString(operation) { - return &v, true - } - if regexp.MustCompile(fmt.Sprintf("^%s[A-Z]", v)).MatchString(operation) { - return &v, true - } - } - return nil, false -} - +// Verbs is a slice of verbs to match against operation names. An operation is considered to match a verb only +// when it begins with a known verb, and verbs can span multiple words, e.g. "TentativelyAccept" var Verbs = operationVerbs{ + "Accept", "Acquire", "Add", "Assign", + "Cancel", "Change", "Check", + "Checkin", + "Checkout", + "Copy", + "Create", + "Decline", + "Delete", + "Discard", "Discover", + "Dismiss", + "End", + "Erase", "Export", + "Extract", "Find", + "Follow", + "Forward", "Get", + "Hide", + "Insert", "Instantiate", "Invoke", "Issue", + "Mark", "Parse", "Pause", + "PowerOff", + "PowerOn", + "Preview", "Process", "Provision", + "Reboot", "Remove", + "Rename", "Renew", + "Reprocess", + "Reprovision", + "Resize", "Restart", "Restore", "Retry", "Revoke", - "Set", "Send", + "Set", + "Share", + "Snooze", "Start", "Stop", + "TentativelyAccept", "Translate", + "Troubleshoot", + "Unfollow", + "Unhide", + "Unmark", "Unset", + "Unshare", "Update", "Validate", "Wipe", } + +var verbMatchers = map[string][]*regexp.Regexp{} + +func init() { + // Compile regexp structs for all verbs in advance for performance + verbMatchers = make(map[string][]*regexp.Regexp) + for _, verb := range Verbs { + verbMatchers[verb] = []*regexp.Regexp{ + regexp.MustCompile(fmt.Sprintf("^%s$", verb)), + regexp.MustCompile(fmt.Sprintf("^%s[A-Z]", verb)), + } + } +} + +// Match returns the matched verb and a boolean to indicate a positive match, when an operation name +// begins with a known verb - noting that verbs can span multiple words. +func (ov operationVerbs) Match(operationName string) (*string, bool) { + for _, v := range ov { + for _, re := range verbMatchers[v] { + if re.MatchString(operationName) { + return &v, true + } + } + } + return nil, false +} diff --git a/tools/importer-msgraph-metadata/components/normalize/verbs_test.go b/tools/importer-msgraph-metadata/components/normalize/verbs_test.go new file mode 100644 index 00000000000..de3061ce1cb --- /dev/null +++ b/tools/importer-msgraph-metadata/components/normalize/verbs_test.go @@ -0,0 +1,63 @@ +package normalize + +import ( + "fmt" + "testing" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" +) + +func TestVerbsMatch(t *testing.T) { + testCases := map[string]struct { + matched bool + verb *string + }{ + // Standalone verbs + "Discover": {true, pointer.To("Discover")}, + "PowerOff": {true, pointer.To("PowerOff")}, + "Provision": {true, pointer.To("Provision")}, + "Reprovision": {true, pointer.To("Reprovision")}, + "Troubleshoot": {true, pointer.To("Troubleshoot")}, + + // Verbs at start of name (should be matched) + "AssignDirectoryRole": {true, pointer.To("Assign")}, + "InstantiateApplication": {true, pointer.To("Instantiate")}, + "PowerOnCloudPC": {true, pointer.To("PowerOn")}, + "RevokeCertificate": {true, pointer.To("Revoke")}, + "TentativelyAcceptCalendarInvitation": {true, pointer.To("TentativelyAccept")}, + + // Verbs not at start of name (should not be matched) + "AdministratorSetRestriction": {false, nil}, + "FolderShareName": {false, nil}, + "ReportExtract": {false, nil}, + "SiteCheckout": {false, nil}, + "UserDeleted": {false, nil}, + + // No verbs + "Demonstrate": {false, nil}, + "MemberIsUser": {false, nil}, + "RestartedApplication": {false, nil}, + "ServicePrincipal": {false, nil}, + } + + for name, expected := range testCases { + verb, matched := Verbs.Match(name) + if matched != expected.matched { + t.Errorf("%s: expected %t match, got %t", name, expected.matched, matched) + } + if verb == nil && expected.verb == nil { + continue + } + if (verb == nil && expected.verb != nil) || (verb != nil && expected.verb == nil) || pointer.From(verb) != pointer.From(expected.verb) { + exp := "nil" + if expected.verb != nil { + exp = fmt.Sprintf("%q", *expected.verb) + } + got := "nil" + if verb != nil { + got = fmt.Sprintf("%q", *verb) + } + t.Errorf("%s: expected verb %s, got %s", name, exp, got) + } + } +} From d919e46761f35a31fc42d7efb47ababf6e6736fd Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 9 Sep 2024 20:30:15 +0100 Subject: [PATCH 101/117] importer-msgraph-metadata: more pluralization and verb exceptions --- .../components/normalize/plurals.go | 1 + .../components/normalize/plurals_test.go | 2 ++ .../components/normalize/verbs.go | 23 +++++++++++++++++++ 3 files changed, 26 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/normalize/plurals.go b/tools/importer-msgraph-metadata/components/normalize/plurals.go index 903b7b2c51d..82ec429f0fc 100644 --- a/tools/importer-msgraph-metadata/components/normalize/plurals.go +++ b/tools/importer-msgraph-metadata/components/normalize/plurals.go @@ -20,6 +20,7 @@ var pluralExceptions = []plural{ {"Metadata", "Metadata"}, {"Orderby", "Orderby"}, {"Premise", "Premises"}, + {"Sms", "Sms"}, {"Sortby", "Sortby"}, } diff --git a/tools/importer-msgraph-metadata/components/normalize/plurals_test.go b/tools/importer-msgraph-metadata/components/normalize/plurals_test.go index dd31956ca63..b8c261d718a 100644 --- a/tools/importer-msgraph-metadata/components/normalize/plurals_test.go +++ b/tools/importer-msgraph-metadata/components/normalize/plurals_test.go @@ -34,10 +34,12 @@ func TestSingularize(t *testing.T) { "Orderby": "Orderby", "Ref": "Ref", "Reference": "Reference", + "Sms": "Sms", "Sortby": "Sortby", // Composite singulars to remain the same "OrderBy": "OrderBy", + "ReminderSms": "ReminderSms", "SecurityGroup": "SecurityGroup", "UserData": "UserData", diff --git a/tools/importer-msgraph-metadata/components/normalize/verbs.go b/tools/importer-msgraph-metadata/components/normalize/verbs.go index ac1d365327a..dd3b73eb104 100644 --- a/tools/importer-msgraph-metadata/components/normalize/verbs.go +++ b/tools/importer-msgraph-metadata/components/normalize/verbs.go @@ -14,22 +14,29 @@ var Verbs = operationVerbs{ "Acquire", "Add", "Assign", + "Bypass", "Cancel", "Change", "Check", "Checkin", "Checkout", + "Clean", + "Clear", "Copy", + "CreateOrGet", // match before "Create" "Create", "Decline", "Delete", + "Disable", "Discard", "Discover", "Dismiss", + "Enable", "End", "Erase", "Export", "Extract", + "Forward", "Find", "Follow", "Forward", @@ -39,7 +46,11 @@ var Verbs = operationVerbs{ "Instantiate", "Invoke", "Issue", + "Locate", + "Login", + "Logout", "Mark", + "Move", "Parse", "Pause", "PowerOff", @@ -48,22 +59,31 @@ var Verbs = operationVerbs{ "Process", "Provision", "Reboot", + "Recover", + "RemoteLock", "Remove", "Rename", "Renew", + "ReplyAll", // match before "Reply" + "Reply", "Reprocess", "Reprovision", + "Request", + "Reset", "Resize", "Restart", "Restore", + "Retire", "Retry", "Revoke", "Send", "Set", "Share", + "ShutDown", "Snooze", "Start", "Stop", + "Sync", "TentativelyAccept", "Translate", "Troubleshoot", @@ -84,7 +104,10 @@ func init() { verbMatchers = make(map[string][]*regexp.Regexp) for _, verb := range Verbs { verbMatchers[verb] = []*regexp.Regexp{ + // Match when the entire name is a verb regexp.MustCompile(fmt.Sprintf("^%s$", verb)), + + // Match when the name starts with a verb and is followed by another title-cased word regexp.MustCompile(fmt.Sprintf("^%s[A-Z]", verb)), } } From 06c4a0947a2940864b5170abffd9ec395960dc67 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 9 Sep 2024 20:31:02 +0100 Subject: [PATCH 102/117] importer-msgraph-metadata: deduplicate resource (category) and operation names --- tools/importer-msgraph-metadata/README.md | 2 +- .../internal/pipeline/task_parse_resources.go | 24 ++++++++++++------- .../pipeline/task_translate_service.go | 5 ++-- 3 files changed, 20 insertions(+), 11 deletions(-) diff --git a/tools/importer-msgraph-metadata/README.md b/tools/importer-msgraph-metadata/README.md index 374f7b19ffb..edfcaa4a448 100644 --- a/tools/importer-msgraph-metadata/README.md +++ b/tools/importer-msgraph-metadata/README.md @@ -42,7 +42,7 @@ This global namespace of models leads to some excessive duplication if we define After importing and processing the API specs using the internal types in `components/parser`, they are then translated to the Data API native types to be persisted in the Data API. There are some things to be aware of in this translation: -* The internal types for Resources have a `Category` and a `Name` - the `Category` maps to the resource name in the data API and the `Name` is used to build operation names. The reason for this approach is that each "resource" in the data API contains multiple Graph resources and/or representations of resources, so the importer organises these internally as "categories" to facilitate grouping of resources prior to translation. +* The internal types for Resources have a `Category` and a `Name` - the `Category` maps to the resource name in the Data API and the `Name` is used to build operation names. The reason for this approach is that each "resource" in the Data API contains multiple Graph resources and/or representations of resources, so the importer organises these internally as "categories" to facilitate grouping of resources prior to translation. When translating resources from the importer's internal representation to the Data API SDK representation, categories are flattened so that operations having the same category get placed in the same resource within the Data API. * All models and constants described in the API spec are considered to be "common types", the only models which are local to a resource are those that represent complex request objects. * Resource IDs are all considered to be resource-local - even though they are intentionally fully-namespaced and are all parsed out together prior to parsing out the resources, so they could in theory _could_ be pooled together as "common resource IDs". However, this would require refactoring of the SDK generator, and at this time (unlike with models), there is no significant overhead to duplicating these. diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 6405fdf1231..67ddc77b107 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -87,7 +87,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model // len(resourceName)>27 && resourceName[:27]=="DirectoryAdministrativeUnit" if _, ok := resources[resourceName]; !ok { // Create a new resource if not already encountered - logging.Infof(fmt.Sprintf("Found new resource %q (category %q, service %q, version %q)", resourceName, resourceCategory, p.service, p.apiVersion)) + logging.Infof(fmt.Sprintf("Found new resource %q (service %q, version %q)", resourceName, p.service, p.apiVersion)) resources[resourceName] = &parser.Resource{ Name: resourceName, @@ -129,7 +129,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model // can happen when no parent resource ID has been matched, for example if a resource ID was blacklisted. if uriSuffix != nil { if uriSuffixParsed := parser.NewResourceId(*uriSuffix, operationTags); uriSuffixParsed.HasUserValue() { - logging.Infof(fmt.Sprintf("Skipping URI suffix containing user value in resource %q (category %q, service %q, version %q): %q", resourceName, resourceCategory, p.service, p.apiVersion, *uriSuffix)) + logging.Infof(fmt.Sprintf("Skipping URI suffix containing user value in resource %q (service %q, version %q): %q", resourceName, p.service, p.apiVersion, *uriSuffix)) continue } } @@ -301,9 +301,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model continue } - // Determine the base name of the operation, whilst attempting to trim a leading service name since that is redundant - operationName := "" - + // Determine prefixes to trim from the operation name prefixesToTrim := []string{ normalize.CleanName(p.service), normalize.Singularize(normalize.CleanName(p.service)), @@ -314,6 +312,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } + // Trim prefixes from the resourceName to get a shortResourceName, which we'll use to build the operationName and to match for known verbs shortResourceName := resourceName for _, prefixToTrim := range prefixesToTrim { if verb, ok := normalize.Verbs.Match(shortResourceName); ok { @@ -324,11 +323,16 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } - operationName = shortResourceName + // Set the name of the operation using the determined shortResourceName (i.e. resourceName with trimmed prefixes) + operationName := shortResourceName if len(operationName) == 0 { + // If trimming prefixes truncated the operationName, then use the full resourceName operationName = resourceName } + // Remove duplicate words in the operationName + operationName = normalize.DeDuplicateName(operationName) + // Now qualify the operation name based on the type of operation. Additionally, if the operation name // matches a known verb, move that verb to the beginning and use it instead of a standard verb. // For example, a Create operation for "Application" should get an operation name of "CreateApplication", @@ -354,7 +358,7 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } case parser.OperationTypeCreateUpdate: - operationName = fmt.Sprintf("CreateUpdate%s", normalize.Singularize(operationName)) + operationName = fmt.Sprintf("Set%s", normalize.Singularize(operationName)) case parser.OperationTypeUpdate: operationName = fmt.Sprintf("Update%s", normalize.Singularize(operationName)) @@ -561,12 +565,13 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } } - // Loop through resources and trim the leading word if it matches the category _and_ there are more words after it + // Clean up categories for _, resource := range resources { if resource.Service == "" || resource.Category == "" { continue } + // Trim the leading word if it matches the category _and_ there are more words after it serviceSingularized := normalize.Singularize(resource.Service) if strings.HasPrefix(resource.Category, serviceSingularized) { trimmedCategory := strings.TrimPrefix(resource.Category, serviceSingularized) @@ -574,6 +579,9 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model resource.Category = trimmedCategory } } + + // Remove duplicate words in the category + resource.Category = normalize.DeDuplicateName(resource.Category) } return diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index 92175668474..34023773b40 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -63,10 +63,10 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } var requestObject *sdkModels.SDKObjectDefinition - requestObjectIsCommonType := true if operation.RequestModel != nil { schemaName := *operation.RequestModel + requestObjectIsCommonType := true if !p.models.Found(schemaName) { return nil, fmt.Errorf("request model %q was not found for operation: %s", schemaName, operation.Name) @@ -224,11 +224,11 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } var responseObject *sdkModels.SDKObjectDefinition - responseObjectIsCommonType := true for _, response := range operation.Responses { if response.Type != nil && *response.Type == parser.DataTypeReference && response.ReferenceName != nil { schemaName := *response.ReferenceName + responseObjectIsCommonType := true if !p.constants.Found(schemaName) && !p.models.Found(schemaName) { return nil, fmt.Errorf("response constant or model %q was not found for operation: %s", schemaName, operation.Name) @@ -298,6 +298,7 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } } + // Append any service-local models that were discovered for _, model := range serviceModels { sdkModel, err := model.DataApiSdkModel(p.models, p.constants) if err != nil { From f6c4a6dbfe15297cac4a21df2979b7c2cc7e534f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 9 Sep 2024 20:40:10 +0100 Subject: [PATCH 103/117] importer-msgraph-metadata: extend synchronizationsecret workaround to applications --- .../components/workarounds/workaround_synchronizationsecrets.go | 1 + 1 file changed, 1 insertion(+) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go index 2a93d4276af..fb76a33c514 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go @@ -23,6 +23,7 @@ func (workaroundSynchronizationSecrets) Name() string { func (workaroundSynchronizationSecrets) Process(apiVersion, serviceName string, resources parser.Resources, resourceIds parser.ResourceIds) error { serviceNamesToPaths := map[string]string{ + "applications": "/applications/{application-id}/synchronization/secrets", "servicePrincipals": "/servicePrincipals/{servicePrincipal-id}/synchronization/secrets", } From 96389748d23e4417ad99f85cfb0c3135f7fa7592 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:31:52 +0100 Subject: [PATCH 104/117] data-api: support importer-specified example values for userspecified resource ID segments --- .../repository/internal/models/resource_ids.go | 7 +++++++ .../internal/transforms/resource_id_segment.go | 11 +++++++---- tools/data-api-sdk/v1/models/source_data_types.go | 11 +++++++++++ 3 files changed, 25 insertions(+), 4 deletions(-) diff --git a/tools/data-api-repository/repository/internal/models/resource_ids.go b/tools/data-api-repository/repository/internal/models/resource_ids.go index 61fbeb4558d..62f9ab0f88e 100644 --- a/tools/data-api-repository/repository/internal/models/resource_ids.go +++ b/tools/data-api-repository/repository/internal/models/resource_ids.go @@ -27,6 +27,13 @@ type ResourceIdSegment struct { // Type is set to ConstantResourceIdSegmentType. ConstantName *string `json:"constantName,omitempty"` + // ExampleValue provides an example of a valid value for this ResourceIDSegment. + // When this is absent in the API definition, an example value is automatically populated in + // the SDK model when the API definitions are loaded, so this isn't mandatory, but does + // override the generated value. This is only used for UserSpecified segment types, and is + // ignored for Constant segment types. + ExampleValue string `json:"exampleValue"` + // Name specifies the name for this ResourceId segment, which should be both unique and // type safe and unique - as this is used as both the name of a Field. Name string `json:"name"` diff --git a/tools/data-api-repository/repository/internal/transforms/resource_id_segment.go b/tools/data-api-repository/repository/internal/transforms/resource_id_segment.go index 2c4503ac69f..8f4329f52e6 100644 --- a/tools/data-api-repository/repository/internal/transforms/resource_id_segment.go +++ b/tools/data-api-repository/repository/internal/transforms/resource_id_segment.go @@ -62,8 +62,10 @@ func mapResourceIdSegmentFromRepository(input repositoryModels.ResourceIdSegment } if input.Type == repositoryModels.UserSpecifiedResourceIdSegmentType { - // TODO: store and load through the example value - exampleValue := fmt.Sprintf("%sValue", strings.TrimSuffix(input.Name, "Name")) + exampleValue := input.ExampleValue + if exampleValue == "" { + exampleValue = fmt.Sprintf("%sValue", strings.TrimSuffix(input.Name, "Name")) + } segment := sdkModels.NewUserSpecifiedResourceIDSegment(input.Name, exampleValue) return &segment, nil } @@ -129,8 +131,9 @@ func mapResourceIdSegmentToRepository(input sdkModels.ResourceIDSegment) (*repos if input.Type == sdkModels.UserSpecifiedResourceIDSegmentType { return &repositoryModels.ResourceIdSegment{ - Type: repositoryModels.UserSpecifiedResourceIdSegmentType, - Name: input.Name, + Type: repositoryModels.UserSpecifiedResourceIdSegmentType, + Name: input.Name, + ExampleValue: input.ExampleValue, }, nil } diff --git a/tools/data-api-sdk/v1/models/source_data_types.go b/tools/data-api-sdk/v1/models/source_data_types.go index 41aa3061e5b..78e62229d86 100644 --- a/tools/data-api-sdk/v1/models/source_data_types.go +++ b/tools/data-api-sdk/v1/models/source_data_types.go @@ -17,3 +17,14 @@ const ( func SourceDataTypeIsDataPlane(sourceDataType SourceDataType) bool { return sourceDataType != ResourceManagerSourceDataType } + +func SourceDataTypeName(sourceDataType SourceDataType) string { + switch sourceDataType { + case MicrosoftGraphSourceDataType: + return "Microsoft Graph" + case ResourceManagerSourceDataType: + return "Azure Resource Manager" + } + + return "" +} From 25807be27e93d2decff57be0a74f7f7766c675ba Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:34:34 +0100 Subject: [PATCH 105/117] generator-go-sdk: fix up struct tag for nullable model fields --- .../internal/generator/templater_models.go | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 0fcf91b2aab..6f37a4658d7 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -295,13 +295,16 @@ func (c modelsTemplater) methods(data GeneratorData) (*string, error) { func (c modelsTemplater) structLineForField(fieldName, fieldType string, fieldDetails models.SDKField, data GeneratorData) (*string, error) { jsonDetails := fieldDetails.JsonName - if c.fieldIsOptional(data, fieldDetails) || fieldDetails.ReadOnly { - if !strings.HasPrefix(fieldType, "nullable.") { + if strings.HasPrefix(fieldType, "nullable.") { + // nullable types should have the omitempty tag option and not be pointers + jsonDetails += ",omitempty" + } else { + if c.fieldIsOptional(data, fieldDetails) || fieldDetails.ReadOnly { + fieldType = fmt.Sprintf("*%s", fieldType) + jsonDetails += ",omitempty" + } else if fieldDetails.ObjectDefinition.Nullable { fieldType = fmt.Sprintf("*%s", fieldType) } - jsonDetails += ",omitempty" - } else if fieldDetails.ObjectDefinition.Nullable && !strings.HasPrefix(fieldType, "nullable.") { - fieldType = fmt.Sprintf("*%s", fieldType) } line := fmt.Sprintf("\t%s %s `json:\"%s\"`", fieldName, fieldType, jsonDetails) From f1fbe2433c0d6930e90b9f43062051731348cb87 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:36:01 +0100 Subject: [PATCH 106/117] generator-go-sdk: update readme generator to be sourceDataType-aware --- .../internal/generator/templater_readme.go | 21 ++++++---- .../generator/templater_readme_test.go | 42 +++++++++---------- 2 files changed, 35 insertions(+), 28 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_readme.go b/tools/generator-go-sdk/internal/generator/templater_readme.go index 4e060b98e85..a54614e83b0 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme.go @@ -20,7 +20,7 @@ type readmeTemplater struct { func (r readmeTemplater) template(data GeneratorData) (*string, error) { summary := r.packageSummary(data) - clientInit := r.clientInitialization(data.packageName, data.serviceClientName) + clientInit := r.clientInitialization(data.sourceType, data.packageName, data.serviceClientName) examples, err := r.exampleUsages(data) if err != nil { return nil, fmt.Errorf("building examples: %+v", err) @@ -53,27 +53,34 @@ func (r readmeTemplater) packageSummary(data GeneratorData) string { return fmt.Sprintf(` ## 'github.com/hashicorp/go-azure-sdk/%[1]s/%[2]s/%[3]s/%[4]s' Documentation -The '%[4]s' SDK allows for interaction with the Azure Resource Manager Service '%[2]s' (API Version '%[3]s'). +The '%[4]s' SDK allows for interaction with %[5]s '%[2]s' (API Version '%[3]s'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). ### Import Path '''go -%[5]s +%[6]s ''' -`, data.sourceType, data.servicePackageName, data.apiVersion, data.packageName, strings.Join(importLines, "\n")) +`, data.sourceType, data.servicePackageName, data.apiVersion, data.packageName, models.SourceDataTypeName(data.sourceType), strings.Join(importLines, "\n")) } -func (r readmeTemplater) clientInitialization(packageName, clientName string) string { +func (r readmeTemplater) clientInitialization(sourceType models.SourceDataType, packageName, clientName string) string { + var baseUri string + switch sourceType { + case models.MicrosoftGraphSourceDataType: + baseUri = "https://graph.microsoft.com" + case models.ResourceManagerSourceDataType: + baseUri = "https://management.azure.com" + } return fmt.Sprintf(` ### Client Initialization '''go -client := %[1]s.New%[2]sWithBaseURI("https://management.azure.com") +client := %[1]s.New%[2]sWithBaseURI("%[3]s") client.Client.Authorizer = authorizer ''' -`, packageName, clientName) +`, packageName, clientName, baseUri) } func (r readmeTemplater) exampleUsages(data GeneratorData) (*string, error) { diff --git a/tools/generator-go-sdk/internal/generator/templater_readme_test.go b/tools/generator-go-sdk/internal/generator/templater_readme_test.go index e493a7f318a..b28778e8c2d 100644 --- a/tools/generator-go-sdk/internal/generator/templater_readme_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_readme_test.go @@ -16,7 +16,7 @@ func TestReadmeTemplater_NoOperations(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -53,7 +53,7 @@ func TestReadmeTemplater_GetOperationWithResourceID(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -119,7 +119,7 @@ func TestReadmeTemplater_GetOperationWithResourceIDUsingACommonID(t *testing.T) expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -187,7 +187,7 @@ func TestReadmeTemplater_GetOperationWithResourceIDAndPayload(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -267,7 +267,7 @@ func TestReadmeTemplater_GetOperationWithResourceIDAndPayloadAndOptions(t *testi expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -355,7 +355,7 @@ func TestReadmeTemplater_GetOperationWithResourceIDAndOptions(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -429,7 +429,7 @@ func TestReadmeTemplater_GetOperationWithoutResourceID(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -487,7 +487,7 @@ func TestReadmeTemplater_GetOperationWithoutResourceIDWithOptions(t *testing.T) expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -553,7 +553,7 @@ func TestReadmeTemplater_ListOperationWithResourceID(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -621,7 +621,7 @@ func TestReadmeTemplater_ListOperationWithResourceIDUsingACommonID(t *testing.T) expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -691,7 +691,7 @@ func TestReadmeTemplater_ListOperationWithResourceIDAndPayload(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -773,7 +773,7 @@ func TestReadmeTemplater_ListOperationWithResourceIDAndPayloadAndOptions(t *test expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -863,7 +863,7 @@ func TestReadmeTemplater_ListOperationWithResourceIDAndOptions(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -946,7 +946,7 @@ func TestReadmeTemplater_ListOperationWithoutResourceID(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1006,7 +1006,7 @@ func TestReadmeTemplater_LongRunningOperationWithResourceID(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1069,7 +1069,7 @@ func TestReadmeTemplater_LongRunningOperationWithResourceIDUsingACommonID(t *tes expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1134,7 +1134,7 @@ func TestReadmeTemplater_LongRunningOperationWithResourceIDAndPayload(t *testing expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1211,7 +1211,7 @@ func TestReadmeTemplater_LongRunningOperationWithResourceIDAndPayloadAndOptions( expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1296,7 +1296,7 @@ func TestReadmeTemplater_LongRunningOperationWithResourceIDAndOptions(t *testing expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1367,7 +1367,7 @@ func TestReadmeTemplater_LongRunningOperationWithoutResourceID(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). @@ -1422,7 +1422,7 @@ func TestReadmeTemplater_MultipleOperations(t *testing.T) { expected := strings.ReplaceAll(` ## 'github.com/hashicorp/go-azure-sdk/resource-manager/compute/2022-02-01/disks' Documentation -The 'disks' SDK allows for interaction with the Azure Resource Manager Service 'compute' (API Version '2022-02-01'). +The 'disks' SDK allows for interaction with Azure Resource Manager 'compute' (API Version '2022-02-01'). This readme covers example usages, but further information on [using this SDK can be found in the project root](https://github.com/hashicorp/go-azure-sdk/tree/main/docs). From ba9c327cc26b18200dfbf8ae41b835937e82b55f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:36:51 +0100 Subject: [PATCH 107/117] generator-go-sdk: ensure all ancestor fields are unmarshalled for discriminated/child models --- .../internal/generator/templater_methods.go | 140 +++++++++--------- .../templater_methods_discriminators_test.go | 2 +- .../internal/generator/templater_models.go | 102 ++++++++----- .../templater_models_discriminators_test.go | 8 + 4 files changed, 150 insertions(+), 102 deletions(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 0972c76a5ec..106ff1f2350 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -617,8 +617,6 @@ func (c methodsPandoraTemplater) marshalerTemplate() (*string, error) { } func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*string, error) { - var output string - if c.operation.LongRunning { // Long Running operations shouldn't be attempted to be unmarshalled until the LRO is completed // in the event this needs to be unmarshalled early - the Response Object being exposed means that @@ -627,7 +625,7 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin } if c.operation.ResponseObject == nil { - return &output, nil + return pointer.To(""), nil } golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) @@ -636,8 +634,6 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin } typeName := *golangTypeName - discriminatedTypeParentName := "" - var model *models.SDKModel modelPackage := "" modelName := typeName @@ -645,48 +641,23 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin modelPackage = s[0] modelName = s[1] } + + // Note that `model` may be nil in the case of standard types (e.g. string, []byte etc) if m, ok := data.models[modelName]; ok { model = &m } else if m, ok = data.commonTypes.Models[modelName]; ok { model = &m } - if model != nil { - // it's either a parent model - if model.FieldNameContainingDiscriminatedValue != nil { - discriminatedTypeParentName = modelName - } - // or an implementation referencing a parent - if model.ParentTypeName != nil { - discriminatedTypeParentName = *model.ParentTypeName - } - - if model.DiscriminatedValue != nil { - // in this instance this would be a discriminated implementation present in the response object - // as such we should use that directly, rather than calling the parents unmarshal function - discriminatedTypeParentName = "" - } - } - + // Paginated response if c.operation.FieldContainingPaginationDetails != nil { - unmarshaler := fmt.Sprintf("Unmarshal%sImplementation", modelName) - if modelPackage != "" { - unmarshaler = fmt.Sprintf("%s.Unmarshal%sImplementation", modelPackage, modelName) - } - - output = fmt.Sprintf(` - var values struct { - Values *[]%[1]s %[2]s - } - if err = resp.Unmarshal(&values); err != nil { - return - } - - result.Model = values.Values -`, typeName, "`json:\"value\"`") + if model != nil && model.IsDiscriminatedParentType() { + unmarshaler := fmt.Sprintf("Unmarshal%sImplementation", modelName) + if modelPackage != "" { + unmarshaler = fmt.Sprintf("%s.Unmarshal%sImplementation", modelPackage, modelName) + } - if discriminatedTypeParentName != "" { - output = fmt.Sprintf(` + output := fmt.Sprintf(` var values struct { Values *[]json.RawMessage %[3]s } @@ -707,20 +678,33 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin } result.Model = &temp `, typeName, unmarshaler, "`json:\"value\"`") + + return &output, nil } + output := fmt.Sprintf(` + var values struct { + Values *[]%[1]s %[2]s + } + if err = resp.Unmarshal(&values); err != nil { + return + } + + result.Model = values.Values +`, typeName, "`json:\"value\"`") + return &output, nil } - // when this is a Discriminated Type (either the Parent or the Implementation, call the `unmarshal` func + // When this is a Discriminated Type (either the Parent or the Implementation, call the `unmarshal` func // for the relevant Parent - if discriminatedTypeParentName != "" { - unmarshaler := fmt.Sprintf("Unmarshal%sImplementation", discriminatedTypeParentName) + if model != nil && model.IsDiscriminatedParentType() { + unmarshaler := fmt.Sprintf("Unmarshal%sImplementation", modelName) if modelPackage != "" { - unmarshaler = fmt.Sprintf("%s.Unmarshal%sImplementation", modelPackage, discriminatedTypeParentName) + unmarshaler = fmt.Sprintf("%s.Unmarshal%sImplementation", modelPackage, modelName) } - output = fmt.Sprintf(` + output := fmt.Sprintf(` var respObj json.RawMessage if err = resp.Unmarshal(&respObj); err != nil { return @@ -729,37 +713,40 @@ func (c methodsPandoraTemplater) unmarshalerTemplate(data GeneratorData) (*strin if err != nil { return } - result.Model = &model + result.Model = model `, unmarshaler) - } else { - responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) - if err != nil { - return nil, fmt.Errorf("determing golang type name for response object: %+v", err) - } - if responseModelType != nil { - if c.operation.FieldContainingPaginationDetails != nil { - output = fmt.Sprintf(` - var model []%s`, *responseModelType) - } else { - output = fmt.Sprintf(` - var model %s`, *responseModelType) - } - output = fmt.Sprintf(`%s - result.Model = &model`, output) + + return &output, nil + } + + responseModelType, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) + if err != nil { + return nil, fmt.Errorf("determing golang type name for response object: %+v", err) + } + + output := "\n" + if responseModelType != nil { + if c.operation.FieldContainingPaginationDetails != nil { + output += fmt.Sprintf(` var model []%s`, *responseModelType) + } else { + output = fmt.Sprintf(` var model %s`, *responseModelType) } - output = fmt.Sprintf(`%s + output += ` + result.Model = &model` + } + + output += ` if err = resp.Unmarshal(result.Model); err != nil { return } -`, output) - } +` return &output, nil } func (c methodsPandoraTemplater) responseStructTemplate(data GeneratorData) (*string, error) { - model := "" + modelLine := "" typeName := "" if c.operation.ResponseObject != nil { golangTypeName, err := helpers.GolangTypeForSDKObjectDefinition(*c.operation.ResponseObject, nil, data.commonTypesPackageName) @@ -768,10 +755,27 @@ func (c methodsPandoraTemplater) responseStructTemplate(data GeneratorData) (*st } typeName = *golangTypeName - if c.operation.FieldContainingPaginationDetails != nil { - model = fmt.Sprintf("Model *[]%s", typeName) + var model *models.SDKModel + modelName := typeName + if s := strings.SplitN(modelName, ".", 2); len(s) == 2 { + modelName = s[1] + } + if m, ok := data.models[modelName]; ok { + model = &m + } else if m, ok = data.commonTypes.Models[modelName]; ok { + model = &m + } + + if model != nil && model.IsParent { + if c.operation.FieldContainingPaginationDetails != nil { + modelLine = fmt.Sprintf("Model *[]%s", typeName) + } else { + modelLine = fmt.Sprintf("Model %s", typeName) + } + } else if c.operation.FieldContainingPaginationDetails != nil { + modelLine = fmt.Sprintf("Model *[]%s", typeName) } else { - model = fmt.Sprintf("Model *%s", typeName) + modelLine = fmt.Sprintf("Model *%s", typeName) } } @@ -805,7 +809,7 @@ type %[1]s struct { } %[4]s -`, responseStructName, model, lro, paginationCode) +`, responseStructName, modelLine, lro, paginationCode) return &output, nil } diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go index f8c11b921a9..4d6104ab11f 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go @@ -220,7 +220,7 @@ func (c pandaClient) Get(ctx context.Context ) (result GetOperationResponse, err if err != nil { return } - result.Model = &model + result.Model = model return }` diff --git a/tools/generator-go-sdk/internal/generator/templater_models.go b/tools/generator-go-sdk/internal/generator/templater_models.go index 6f37a4658d7..bc232d8f6a0 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models.go +++ b/tools/generator-go-sdk/internal/generator/templater_models.go @@ -641,7 +641,28 @@ func (c modelsTemplater) codeForUnmarshalParentFunction(data GeneratorData) (*st if len(modelsImplementingThisClass) == 0 && featureflags.SkipDiscriminatedParentTypes() == false { return nil, fmt.Errorf("model %q is a discriminated parent type with no implementations", c.name) } - jsonFieldName := c.model.Fields[*c.model.FieldNameContainingDiscriminatedValue].JsonName + + // Discover discriminated field, which may be defined on an ancestor model + var discriminatedValueField models.SDKField + var ok bool + if discriminatedValueField, ok = c.model.Fields[*c.model.FieldNameContainingDiscriminatedValue]; !ok && c.model.ParentTypeName != nil { + ancestorTypeNames := []string{*c.model.ParentTypeName} + if c.model.FieldNameContainingDiscriminatedValue != nil { + _, foundAncestorTypeNames, err := c.findModelAncestry(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + if err != nil { + return nil, err + } + ancestorTypeNames = *foundAncestorTypeNames + + // Look for the discriminated value field in all ancestors + for _, ancestorTypeName := range ancestorTypeNames { + if discriminatedValueField, ok = data.models[ancestorTypeName].Fields[*c.model.FieldNameContainingDiscriminatedValue]; ok { + break + } + } + } + } + // NOTE: unmarshaling null returns an empty map, which'll mean the `ok` fails // the 'type' field being omitted will also mean that `ok` is false lines = append(lines, fmt.Sprintf(` @@ -659,7 +680,7 @@ func Unmarshal%[1]sImplementation(input []byte) (%[1]s, error) { if !ok { return nil, nil } -`, c.name, jsonFieldName)) +`, c.name, discriminatedValueField.JsonName)) sort.Strings(modelsImplementingThisClass) for _, implementationName := range modelsImplementingThisClass { @@ -708,45 +729,66 @@ func (c modelsTemplater) codeForUnmarshalStructFunction(data GeneratorData) (*st lines := make([]string, 0) - // fields either require unmarshaling or can be explicitly assigned, determine which + // Determine which fields can be directly assigned and which must be explicitly unmarshalled fieldsRequiringAssignment := make([]string, 0) - fieldsRequiringUnmarshalling := make([]string, 0) + fieldsRequiringUnmarshalling := make(map[string]models.SDKField) for fieldName, fieldDetails := range c.model.Fields { + // Check if the model field references a model interface, which will require explicit unmarshalling topLevelObject := helpers.InnerMostSDKObjectDefinition(fieldDetails.ObjectDefinition) if topLevelObject.Type == models.ReferenceSDKObjectDefinitionType { model, ok := data.models[*topLevelObject.ReferenceName] if ok && model.IsDiscriminatedParentType() { - fieldsRequiringUnmarshalling = append(fieldsRequiringUnmarshalling, fieldName) + fieldsRequiringUnmarshalling[fieldName] = fieldDetails continue } } fieldsRequiringAssignment = append(fieldsRequiringAssignment, fieldName) } + + // Enumerate fields from ancestor models, determine which fields to assign directory or explicitly unmarshal if c.model.ParentTypeName != nil { - parent, ok := data.models[*c.model.ParentTypeName] - if !ok { - return nil, fmt.Errorf("Parent Model %q (for Model %q) was not found", *c.model.ParentTypeName, c.name) + ancestorTypeNames := []string{*c.model.ParentTypeName} + if c.model.FieldNameContainingDiscriminatedValue != nil { + _, foundAncestorTypeNames, err := c.findModelAncestry(data, *c.model.ParentTypeName, *c.model.FieldNameContainingDiscriminatedValue) + if err != nil { + return nil, err + } + ancestorTypeNames = *foundAncestorTypeNames } - for fieldName, fieldDetails := range parent.Fields { - // also double-check if the parent has any fields matching the same conditions + + ancestorFields := make(map[string]models.SDKField) + + // We want to include fields from all ancestors + for _, ancestorTypeName := range ancestorTypeNames { + parent, ok := data.models[ancestorTypeName] + if !ok { + return nil, fmt.Errorf("couldn't find Ancestor Model %q for Model %q", ancestorTypeName, c.name) + } + for fieldName, fieldDetails := range parent.Fields { + if _, ok := ancestorFields[fieldName]; ok { + // Skip fields already present from a closer ancestor + continue + } + ancestorFields[fieldName] = fieldDetails + } + } + + for fieldName, fieldDetails := range ancestorFields { + // Check if the ancestor field references a model interface, which requires explicit unmarshalling topLevelObject := helpers.InnerMostSDKObjectDefinition(fieldDetails.ObjectDefinition) if topLevelObject.Type == models.ReferenceSDKObjectDefinitionType { model, ok := data.models[*topLevelObject.ReferenceName] if ok && model.IsDiscriminatedParentType() { - fieldsRequiringUnmarshalling = append(fieldsRequiringUnmarshalling, fieldName) + fieldsRequiringUnmarshalling[fieldName] = fieldDetails continue } } - // however specifically for the parent we don't want to assign the `type` field since we don't output it - // so check the implementation model to determine which field the `type` is in, and assign if they - // don't match - // - // at this point since we know there's a parent-implementation relationship, there's no need to nil-check - if *c.model.FieldNameContainingDiscriminatedValue != fieldName { - fieldsRequiringAssignment = append(fieldsRequiringAssignment, fieldName) - } + // Note: previously we skipped unmarshalling the `type` field for discriminated models, we now intentionally + // populate this field to allow consumers the option of inspecting it. Note that we do not marshal the `type` + // field, so it cannot be set by consumers. + fieldsRequiringAssignment = append(fieldsRequiringAssignment, fieldName) } } @@ -780,21 +822,15 @@ func (s *%[1]s) UnmarshalJSON(bytes []byte) error {`, structName)) } `, structName)) - sort.Strings(fieldsRequiringUnmarshalling) - for _, fieldName := range fieldsRequiringUnmarshalling { - fieldDetails, ok := c.model.Fields[fieldName] + fieldNamesRequiringUnmarshalling := make([]string, 0, len(fieldsRequiringUnmarshalling)) + for fieldName := range fieldsRequiringUnmarshalling { + fieldNamesRequiringUnmarshalling = append(fieldNamesRequiringUnmarshalling, fieldName) + } + sort.Strings(fieldNamesRequiringUnmarshalling) + for _, fieldName := range fieldNamesRequiringUnmarshalling { + fieldDetails, ok := fieldsRequiringUnmarshalling[fieldName] if !ok { - if c.model.ParentTypeName == nil { - return nil, fmt.Errorf("field %q was not found on Model %q which has no Parent", fieldName, c.name) - } - parent, ok := data.models[*c.model.ParentTypeName] - if !ok { - return nil, fmt.Errorf("parent model %q was not found", *c.model.ParentTypeName) - } - fieldDetails, ok = parent.Fields[fieldName] - if !ok { - return nil, fmt.Errorf("field %q was not found on Model %q or Parent %q", fieldName, c.name, *c.model.ParentTypeName) - } + return nil, fmt.Errorf("internal-error: field %q for model %q was not found in `fieldsRequiringUnmarshalling` map", fieldName, c.name) } topLevelObjectDef := helpers.InnerMostSDKObjectDefinition(fieldDetails.ObjectDefinition) diff --git a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go index 4daf25c2f9a..1828f372e10 100644 --- a/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_models_discriminators_test.go @@ -524,6 +524,14 @@ func (s FirstImplementation) MarshalJSON() ([]byte, error) { var _ json.Unmarshaler = &FirstImplementation{} func (s *FirstImplementation) UnmarshalJSON(bytes []byte) error { + type alias FirstImplementation + var decoded alias + if err := json.Unmarshal(bytes, &decoded); err != nil { + return fmt.Errorf("unmarshaling into FirstImplementation: %+v", err) + } + + s.Type = decoded.Type + var temp map[string]json.RawMessage if err := json.Unmarshal(bytes, &temp); err != nil { return fmt.Errorf("unmarshaling FirstImplementation into map[string]json.RawMessage: %+v", err) From c45f7304e912bf31c522a9e0e8424c867c60488e Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:38:20 +0100 Subject: [PATCH 108/117] importer-msgraph-metadata: set a reasonable example value for user specified resource ID segments --- .../importer-msgraph-metadata/components/parser/resourceids.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index c0733caa4dc..533582cba3a 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -84,7 +84,7 @@ func (r ResourceId) DataApiSdkResourceId() (*sdkModels.ResourceID, error) { case SegmentAction, SegmentCast, SegmentFunction, SegmentLabel, SegmentODataReference: sdkSegments = append(sdkSegments, sdkModels.NewStaticValueResourceIDSegment(segment.Value, segment.Value)) case SegmentUserValue: - sdkSegments = append(sdkSegments, sdkModels.NewUserSpecifiedResourceIDSegment(normalize.CleanNameCamel(*segment.field), segment.Value)) + sdkSegments = append(sdkSegments, sdkModels.NewUserSpecifiedResourceIDSegment(normalize.CleanNameCamel(*segment.field), normalize.CleanNameCamel(segment.Value))) default: return nil, fmt.Errorf("unknown segment type %q at index %d for resource ID: %q", segment.Type, i, r.Name) } From 6f698c04fbae2191f4e0073fd585677135ce4bfa Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:39:29 +0100 Subject: [PATCH 109/117] importer-msgraph-metadata: fix early exit bug in SynchronizationSecrets workaround --- .../components/workarounds/workaround_synchronizationsecrets.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go index fb76a33c514..f58913c6216 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_synchronizationsecrets.go @@ -29,7 +29,7 @@ func (workaroundSynchronizationSecrets) Process(apiVersion, serviceName string, for serviceNameToMatch, path := range serviceNamesToPaths { if serviceNameToMatch != serviceName { - return nil + continue } resourceName := fmt.Sprintf("%sSynchronizationSecret", normalize.Singularize(normalize.CleanName(serviceName))) From 9260ca581d022263c0f67cfedc0aa0dbe32036fb Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:40:23 +0100 Subject: [PATCH 110/117] importer-msgraph-metadata: disable deduplication for categories, once again causes clobbering --- .../internal/pipeline/task_parse_resources.go | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go index 67ddc77b107..bdcf6ea2325 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_parse_resources.go @@ -581,7 +581,9 @@ func (p pipelineForService) parseResources(resourceIds parser.ResourceIds, model } // Remove duplicate words in the category - resource.Category = normalize.DeDuplicateName(resource.Category) + // TODO: Figure out whether we can do this safely; causes clobbering in identityGovernance, e.g. + // EntitlementManagementAccessPackageResourceRoleScopeClient vs EntitlementManagementAccessPackageAccessPackageResourceRoleScopeClient + //resource.Category = normalize.DeDuplicateName(resource.Category) } return From 8e09d5cb9e304b929b62f5641a94b1980211ba25 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 10:40:58 +0100 Subject: [PATCH 111/117] importer-msgraph-metadata: add unit test for `TruncateToLastSegmentOfTypeBeforeSegment` method --- .../components/parser/resourceids.go | 2 +- .../components/parser/resourceids_test.go | 298 ++++++++++++++++++ 2 files changed, 299 insertions(+), 1 deletion(-) create mode 100644 tools/importer-msgraph-metadata/components/parser/resourceids_test.go diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 533582cba3a..92dd1fe2f85 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -312,7 +312,7 @@ func (r ResourceId) LastSegmentOfTypeBeforeSegment(types []ResourceIdSegmentType return nil } -// TruncateToLastSegmentOfTypeBeforeSegment returns a new ResourceId, truncating this ResourceId to the last segment of +// TruncateToLastSegmentOfTypeBeforeSegment returns a new ResourceId, truncated to the last segment of // the specified type from the ResourceId that precedes the provided segment index func (r ResourceId) TruncateToLastSegmentOfTypeBeforeSegment(types []ResourceIdSegmentType, i int) *ResourceId { ret := r diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids_test.go b/tools/importer-msgraph-metadata/components/parser/resourceids_test.go new file mode 100644 index 00000000000..171f023f1a1 --- /dev/null +++ b/tools/importer-msgraph-metadata/components/parser/resourceids_test.go @@ -0,0 +1,298 @@ +package parser + +import ( + "reflect" + "testing" + + "github.com/davecgh/go-spew/spew" + "github.com/hashicorp/go-azure-helpers/lang/pointer" +) + +func TestNewResourceID(t *testing.T) { + testCases := []struct { + input string + tags []string + expected ResourceId + }{ + { + input: "/applications", + tags: []string{"applications.application"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "applications", + plural: true, + }, + }, + }, + }, + { + input: "/applications/{application-id}", + tags: []string{"applications.application"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "applications", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{applicationId}", + field: pointer.To("ApplicationId"), + }, + }, + }, + }, + { + input: "/groups/{group-id}/owners", + tags: []string{"groups.directoryObject"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "groups", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{groupId}", + field: pointer.To("GroupId"), + }, + { + Type: SegmentLabel, + Value: "owners", + plural: true, + }, + }, + }, + }, + { + input: "/groups/{group-id}/owners/$count", + tags: []string{"groups.directoryObject"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "groups", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{groupId}", + field: pointer.To("GroupId"), + }, + { + Type: SegmentLabel, + Value: "owners", + plural: true, + }, + { + Type: SegmentODataReference, + Value: "$count", + }, + }, + }, + }, + { + input: "/groups/{group-id}/owners/$ref", + tags: []string{"groups.directoryObject"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "groups", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{groupId}", + field: pointer.To("GroupId"), + }, + { + Type: SegmentLabel, + Value: "owners", + plural: true, + }, + { + Type: SegmentODataReference, + Value: "$ref", + }, + }, + }, + }, + { + input: "/applications/{application-id}/microsoft.graph.setVerifiedPublisher", + tags: []string{"applications.Actions"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "applications", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{applicationId}", + field: pointer.To("ApplicationId"), + }, + { + Type: SegmentAction, + Value: "setVerifiedPublisher", + }, + }, + }, + }, + { + input: "/groups/microsoft.graph.delta()", + tags: []string{"groups.Functions"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "groups", + plural: true, + }, + { + Type: SegmentFunction, + Value: "microsoft.graph.delta()", + }, + }, + }, + }, + { + input: "/directory/administrativeUnits/{administrativeUnit-id}/members/{directoryObject-id}/microsoft.graph.servicePrincipal", + tags: []string{"directory.administrativeUnit"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "directory", + }, + { + Type: SegmentLabel, + Value: "administrativeUnits", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{administrativeUnitId}", + field: pointer.To("AdministrativeUnitId"), + }, + { + Type: SegmentLabel, + Value: "members", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{directoryObjectId}", + field: pointer.To("DirectoryObjectId"), + }, + { + Type: SegmentCast, + Value: "microsoft.graph.servicePrincipal", + }, + }, + }, + }, + { + input: "/security/threatIntelligence/vulnerabilities/{vulnerability-id}/components/$count", + tags: []string{"security.threatIntelligence"}, + expected: ResourceId{ + Segments: []ResourceIdSegment{ + { + Type: SegmentLabel, + Value: "security", + }, + { + Type: SegmentLabel, + Value: "threatIntelligence", + }, + { + Type: SegmentLabel, + Value: "vulnerabilities", + plural: true, + }, + { + Type: SegmentUserValue, + Value: "{vulnerabilityId}", + field: pointer.To("VulnerabilityId"), + }, + { + Type: SegmentLabel, + Value: "components", + plural: true, + }, + { + Type: SegmentODataReference, + Value: "$count", + }, + }, + }, + }, + } + + for _, c := range testCases { + id := NewResourceId(c.input, c.tags) + if !reflect.DeepEqual(id, c.expected) { + t.Errorf(`expected: +%s + +received: +%s`, spew.Sdump(c.expected), spew.Sdump(id)) + } + } +} + +func TestTruncateToLastSegmentOfTypeBeforeSegment(t *testing.T) { + testCases := []struct { + input string + tags []string + segmentTypes []ResourceIdSegmentType + index int + expected string + }{ + { + input: "/applications/{application-id}/owners", + tags: []string{"applications.DirectoryObject"}, + segmentTypes: []ResourceIdSegmentType{SegmentUserValue}, + index: -1, + expected: "/applications/{applicationId}", + }, + { + input: "/directory/administrativeUnits/{administrativeUnit-id}/members/{directoryObject-id}/microsoft.graph.servicePrincipal", + tags: []string{"directory.administrativeUnit"}, + segmentTypes: []ResourceIdSegmentType{SegmentUserValue}, + index: 4, + expected: "/directory/administrativeUnits/{administrativeUnit-id}", + }, + { + input: "/security/threatIntelligence/vulnerabilities/{vulnerability-id}/components/$count", + tags: []string{"security.threatIntelligence"}, + segmentTypes: []ResourceIdSegmentType{SegmentLabel}, + index: 4, + expected: "/security/threatIntelligence/vulnerabilities", + }, + } + + for _, c := range testCases { + original := NewResourceId(c.input, c.tags) + output := original.TruncateToLastSegmentOfTypeBeforeSegment(c.segmentTypes, c.index) + + if output == nil { + t.Error("received nil ResourceId") + } + + expected := NewResourceId(c.expected, c.tags) + + if !reflect.DeepEqual(*output, expected) { + t.Errorf(`expected: +%s + +received: +%s`, spew.Sdump(expected), spew.Sdump(output)) + } + } +} From 51de4a5c40ba3050bfef0038f1bf6a02b7c9c856 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 11:05:08 +0100 Subject: [PATCH 112/117] importer-msgraph-metadata: add unit test for `FullyQualifiedResourceName` method --- .../components/parser/resourceids_test.go | 86 +++++++++++++++++++ 1 file changed, 86 insertions(+) diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids_test.go b/tools/importer-msgraph-metadata/components/parser/resourceids_test.go index 171f023f1a1..63902910a0c 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids_test.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids_test.go @@ -246,6 +246,92 @@ received: } } +func TestFullyQualifiedResourceName(t *testing.T) { + testCases := []struct { + input string + tags []string + expected string + }{ + // Top-level resource + { + input: "/applications/{application-id}", + tags: []string{"applications.Application"}, + expected: "Application", + }, + + // Top-level resource with namespacing + { + input: "/roleManagement/directory/roleAssignmentScheduleInstances/{unifiedRoleAssignmentScheduleInstance-id}", + tags: []string{"roleManagement.rbacApplication"}, + expected: "RoleManagementDirectoryRoleAssignmentScheduleInstance", + }, + + // Child resource + { + input: "/applications/{application-id}/owners", + tags: []string{"applications.DirectoryObject"}, + expected: "ApplicationOwner", + }, + + // Child resource with namespacing + { + input: "/servicePrincipals/{servicePrincipal-id}/synchronization/jobs/{synchronizationJob-id}", + tags: []string{"security.threatIntelligence"}, + expected: "ServicePrincipalSynchronizationJob", + }, + + // Child resource with parent namespacing + { + input: "/security/threatIntelligence/vulnerabilities/{vulnerability-id}/components", + tags: []string{"security.threatIntelligence"}, + expected: "SecurityThreatIntelligenceVulnerabilityComponent", + }, + + // Child resource with name overlap should maintain namespacing to avoid clobbering + { + input: "/identityGovernance/entitlementManagement/accessPackages/{accessPackage-id}/accessPackageResourceRoleScopes", + tags: []string{"identityGovernance.entitlementManagement"}, + expected: "IdentityGovernanceEntitlementManagementAccessPackageAccessPackageResourceRoleScope", + }, + + // Verb moves to start of name + { + input: "/applications/{application-id}/microsoft.graph.setVerifiedPublisher", + tags: []string{"applications.Actions"}, + expected: "SetApplicationVerifiedPublisher", + }, + + // OData reference should be pluralized + { + input: "/directory/administrativeUnits/{administrativeUnit-id}/members/$ref", + tags: []string{"directory.administrativeUnit"}, + expected: "DirectoryAdministrativeUnitMemberRefs", + }, + + // OData count should pluralize the preceding label + { + input: "/servicePrincipals/{servicePrincipal-id}/createdObjects/$count", + tags: []string{"servicePrincipals.directoryObject"}, + expected: "ServicePrincipalCreatedObjectsCount", + }, + } + + for _, c := range testCases { + id := NewResourceId(c.input, c.tags) + output, ok := id.FullyQualifiedResourceName(nil) + + if !ok { + t.Errorf("received false") + } + + if output == nil { + t.Error("received nil fqrn") + } else if *output != c.expected { + t.Errorf("expected: %q, received: %q", c.expected, *output) + } + } +} + func TestTruncateToLastSegmentOfTypeBeforeSegment(t *testing.T) { testCases := []struct { input string From 63aa605ff63e420710833e15986fc99b662f4bbc Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Thu, 12 Sep 2024 14:53:08 +0100 Subject: [PATCH 113/117] importer-msgraph-metadata: support OData Metadata parameter --- .../components/parser/resources.go | 1 + .../internal/pipeline/task_translate_service.go | 14 +++++++++----- 2 files changed, 10 insertions(+), 5 deletions(-) diff --git a/tools/importer-msgraph-metadata/components/parser/resources.go b/tools/importer-msgraph-metadata/components/parser/resources.go index 1cfe9da5863..e40d482e464 100644 --- a/tools/importer-msgraph-metadata/components/parser/resources.go +++ b/tools/importer-msgraph-metadata/components/parser/resources.go @@ -143,6 +143,7 @@ type Resources map[string]*Resource // ServiceHasValidResources returns true when resources are found for the provided serviceName that have usable operations // defined (specifically any operations that do not require a response model, or that have a response model for any response) +// TODO: maybe remove this, we are not using this as it's preferable to error out rather than skip over func (r Resources) ServiceHasValidResources(serviceName string) bool { for _, resource := range r { if resource.Category == "" { diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index 34023773b40..d67aa9537f1 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -90,7 +90,15 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } } - options := make(map[string]sdkModels.SDKOperationOption) + options := map[string]sdkModels.SDKOperationOption{ + "Metadata": { + ODataFieldName: pointer.To("Metadata"), + ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ + ReferenceName: pointer.To("odata.Metadata"), + Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, + }, + }, + } if operation.RequestHeaders != nil { for _, header := range *operation.RequestHeaders { @@ -98,7 +106,6 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv options[normalize.CleanName(header.Name)] = sdkModels.SDKOperationOption{ ODataFieldName: &header.Name, ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ - NestedItem: nil, ReferenceName: pointer.To("odata.ConsistencyLevel"), Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, }, @@ -139,7 +146,6 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv options["Expand"] = sdkModels.SDKOperationOption{ ODataFieldName: pointer.To("Expand"), ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ - NestedItem: nil, ReferenceName: pointer.To("odata.Expand"), Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, }, @@ -149,7 +155,6 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv options["Format"] = sdkModels.SDKOperationOption{ ODataFieldName: pointer.To("Format"), ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ - NestedItem: nil, ReferenceName: pointer.To("odata.Format"), Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, }, @@ -159,7 +164,6 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv options["OrderBy"] = sdkModels.SDKOperationOption{ ODataFieldName: pointer.To("OrderBy"), ObjectDefinition: sdkModels.SDKOperationOptionObjectDefinition{ - NestedItem: nil, ReferenceName: pointer.To("odata.OrderBy"), Type: sdkModels.ReferenceSDKOperationOptionObjectDefinitionType, }, From b65699edfa85ecbc2158dd17dd6c8928de4ccdcd Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 13 Sep 2024 13:09:11 +0100 Subject: [PATCH 114/117] generator-go-sdk: fix bug in parent model check --- tools/generator-go-sdk/internal/generator/templater_methods.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods.go b/tools/generator-go-sdk/internal/generator/templater_methods.go index 106ff1f2350..60669b2f5c7 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods.go @@ -766,7 +766,7 @@ func (c methodsPandoraTemplater) responseStructTemplate(data GeneratorData) (*st model = &m } - if model != nil && model.IsParent { + if model != nil && model.IsDiscriminatedParentType() { if c.operation.FieldContainingPaginationDetails != nil { modelLine = fmt.Sprintf("Model *[]%s", typeName) } else { From b49419054c0cc5f7ccec2b7bf34cf47742635082 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Fri, 13 Sep 2024 13:24:39 +0100 Subject: [PATCH 115/117] generator-go-sdk: update test --- .../internal/generator/templater_methods_discriminators_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go index 4d6104ab11f..42a03b6d260 100644 --- a/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go +++ b/tools/generator-go-sdk/internal/generator/templater_methods_discriminators_test.go @@ -183,7 +183,7 @@ import ( type GetOperationResponse struct { HttpResponse *http.Response OData *odata.OData - Model *FizzyDrink + Model FizzyDrink } // Get ... From b3a0bd97802cfea09abd1e1668ddd6dc23f9a31e Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 16 Sep 2024 14:42:48 +0100 Subject: [PATCH 116/117] config: adding `beta` API for `identityGovernance` in `microsoft-graph` --- config/microsoft-graph.hcl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/microsoft-graph.hcl b/config/microsoft-graph.hcl index 5e9ffbb51d6..2d4db8bbf8a 100644 --- a/config/microsoft-graph.hcl +++ b/config/microsoft-graph.hcl @@ -68,7 +68,7 @@ service "identity" { service "identityGovernance" { name = "IdentityGovernance" - available = ["stable"] + available = ["stable", "beta"] } service "invitations" { From 743c42adcb82192ded3474ef0af2645f3c205cc3 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 17 Sep 2024 19:12:48 +0100 Subject: [PATCH 117/117] importer-msgraph-metadata: add more doc comments, remove resource blacklist since this is now covered by workarounds --- .../components/blacklisted/resource.go | 74 -------- .../components/parser/resourceids.go | 2 + .../components/parser/resources.go | 158 ++++++++++-------- .../components/parser/types.go | 137 ++++++++++----- .../workaround_conditionalaccesspolicy.go | 8 +- .../workarounds/workaround_odata_bind.go | 1 + ...rkaround_repeating_resource_id_segments.go | 3 + .../components/workarounds/workarounds.go | 7 + .../internal/pipeline/importer.go | 7 +- .../pipeline/task_translate_service.go | 5 - 10 files changed, 202 insertions(+), 200 deletions(-) delete mode 100644 tools/importer-msgraph-metadata/components/blacklisted/resource.go diff --git a/tools/importer-msgraph-metadata/components/blacklisted/resource.go b/tools/importer-msgraph-metadata/components/blacklisted/resource.go deleted file mode 100644 index 672de85a53d..00000000000 --- a/tools/importer-msgraph-metadata/components/blacklisted/resource.go +++ /dev/null @@ -1,74 +0,0 @@ -package blacklisted - -import ( - "fmt" - "strings" - - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" -) - -func Resource(resource *parser.Resource) bool { - if resource.Service == "Groups" { - - // Has IDs with repeating segments, which are not supported - if resource.Category == "SiteSite" { - return true - } - - // GroupSiteTermStore resources have repeating ID segments which are not supported at this time - if strings.Contains(resource.Category, "TermStore") { - return true - } - - // Onenote resources have repeating ID segments which are not supported at this time - if strings.Contains(resource.Category, "Onenote") { - return true - } - - } - - if resource.Service == "IdentityGovernance" { - - // These contain IDs with repeating segments, which are not supported at this time - prefixes := []string{ - "EntitlementManagementAccessPackageResourceRoleScopeRoleResourceScopeResourceRole", - "EntitlementManagementAccessPackageResourceRoleScopeScopeResourceRoleResourceScope", - "EntitlementManagementCatalogResourceRoleResourceScopeResourceRole", - "EntitlementManagementCatalogResourceScopeResourceRoleResourceScope", - "EntitlementManagementResourceRequestCatalogResourceRoleResourceScopeResource", - "EntitlementManagementResourceRequestCatalogResourceScopeResourceRoleResourceScope", - "EntitlementManagementResourceRequestResourceRoleResourceScope", - "EntitlementManagementResourceRequestResourceScopeResourceRole", - "EntitlementManagementResourceRoleScopeRoleResourceScopeResourceRole", - "EntitlementManagementResourceRoleScopeScopeResourceRoleResourceScope", - } - for _, prefix := range prefixes { - if strings.HasPrefix(resource.Category, prefix) || strings.HasPrefix(resource.Category, fmt.Sprintf("%s%s", resource.Service, prefix)) { - return true - } - } - - } - - if resource.Service == "Me" || resource.Service == "Users" { - - // Onenote resources have repeating ID segments which are not supported at this time - if strings.Contains(resource.Category, "Onenote") { - return true - } - - // These contain IDs with repeating segments, which are not supported at this time - prefixes := []string{ - "PendingAccessReviewInstanceDecisionInstanceStageDecision", - "PendingAccessReviewInstanceStageDecisionInstanceDecision", - } - for _, prefix := range prefixes { - if strings.HasPrefix(resource.Category, prefix) || strings.HasPrefix(resource.Category, fmt.Sprintf("%s%s", resource.Service, prefix)) { - return true - } - } - - } - - return false -} diff --git a/tools/importer-msgraph-metadata/components/parser/resourceids.go b/tools/importer-msgraph-metadata/components/parser/resourceids.go index 92dd1fe2f85..0ca775d3eab 100644 --- a/tools/importer-msgraph-metadata/components/parser/resourceids.go +++ b/tools/importer-msgraph-metadata/components/parser/resourceids.go @@ -76,6 +76,8 @@ func (r ResourceId) ID() string { return "/" + strings.Join(segments, "/") } +// DataApiSdkResourceId converts the internal ResourceId representation to a Data API SDK ResourceID, so it can be +// persisted to the Data API Definitions. func (r ResourceId) DataApiSdkResourceId() (*sdkModels.ResourceID, error) { sdkSegments := make([]sdkModels.ResourceIDSegment, 0, len(r.Segments)) diff --git a/tools/importer-msgraph-metadata/components/parser/resources.go b/tools/importer-msgraph-metadata/components/parser/resources.go index e40d482e464..d6dd418dfea 100644 --- a/tools/importer-msgraph-metadata/components/parser/resources.go +++ b/tools/importer-msgraph-metadata/components/parser/resources.go @@ -10,32 +10,99 @@ import ( sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" ) +type Resources map[string]*Resource + type Resource struct { - Name string - Category string - Version string - Service string - Paths []ResourceId + // The name of this resource + Name string + + // The category for this resource, used to group resources together in the same SDK package + Category string + + // The API version for this resource + Version string + + // The name of the service associated with this resource + Service string + + // All known paths for this resource, used for category matching between different resources + Paths []ResourceId + + // Supported operations for this resource Operations []Operation } type Operation struct { - Name string - Description string - Type OperationType - Method string - ResourceId *ResourceId - UriSuffix *string + // The full name of this operation, which should be unique across resources (at least in the same category) so + // prevent clobbering when resources/operations are grouped into an SDK package + Name string + + // Optional description which can be added to the generated SDK model as a comment + Description string + + // The type of this operation, initially determined from the HTTP method + Type OperationType + + // The HTTP method for this operation + Method string + + // The resource ID that comprises the first part of the URI for this operation + ResourceId *ResourceId + + // The remainder of the URI after the resource ID + UriSuffix *string + + // The content-type of the request body RequestContentType *string - RequestModel *string - RequestHeaders *Headers - RequestParams *Params - RequestType *DataType - Responses Responses - PaginationField *string - Tags []string + + // The model that describes the request body + RequestModel *string + + // Any user-specified HTTP headers supported for the request + RequestHeaders *Headers + + // Any user-specified query string parameters support for the request + RequestParams *Params + + // The internal data type for the request, used when the content type is JSON or XML, + // and the request is not described by a model + RequestType *DataType + + // The expected *success* responses for this operation + Responses Responses + + // When the content type is JSON or XML, and this is a List operation, the name of the field + // that specifies a URL to retrieve the next page of results + PaginationField *string + + // OpenAPI3 tags for this operation, used to reconcile operations to services + Tags []string +} + +type Responses []Response + +type Response struct { + // The HTTP status code associated with this expected response + Status int + + // The expected content type for this resource + ContentType *string + + // Specifies a referenced model or constant for the response, noting that this + // should be the full type name from the spec prior to normalizing + ReferenceName *string + + // The internal data type for the response, used when the content type is JSON or XML, + // and the response is not described by a model + Type *DataType + + // Model and Constant are used internally for ad-hoc response models + Model *Model + Constant *Constant } +type Headers []Header + type Header struct { Name string Type *DataType @@ -53,7 +120,7 @@ func (h Header) DataApiSdkObjectDefinition() (*sdkModels.SDKOperationOptionObjec }, nil } -type Headers []Header +type Params []Param type Param struct { Name string @@ -89,28 +156,6 @@ func (p Param) DataApiSdkObjectDefinition() (*sdkModels.SDKOperationOptionObject }, nil } -type Params []Param - -type Response struct { - Status int - ContentType *string - ReferenceName *string - Type *DataType - Model *Model - Constant *Constant -} - -type Responses []Response - -func (rs Responses) FindModelName() *string { - for _, r := range rs { - if r.ReferenceName != nil { - return r.ReferenceName - } - } - return nil -} - type OperationType uint8 const ( @@ -138,34 +183,3 @@ func NewOperationType(method string) OperationType { } return OperationTypeUnknown } - -type Resources map[string]*Resource - -// ServiceHasValidResources returns true when resources are found for the provided serviceName that have usable operations -// defined (specifically any operations that do not require a response model, or that have a response model for any response) -// TODO: maybe remove this, we are not using this as it's preferable to error out rather than skip over -func (r Resources) ServiceHasValidResources(serviceName string) bool { - for _, resource := range r { - if resource.Category == "" { - // These are logged earlier in the pipeline - continue - } - - for _, operation := range resource.Operations { - if operation.Type == OperationTypeList || operation.Type == OperationTypeRead { - // Determine whether to skip operation with missing response model - if operation.Type != OperationTypeDelete { - if responseModel := operation.Responses.FindModelName(); responseModel == nil { - if operation.ResourceId == nil || len(operation.ResourceId.Segments) == 0 || operation.ResourceId.Segments[len(operation.ResourceId.Segments)-1].Value != "$ref" { - continue - } - } - } - } - - return true - } - } - - return false -} diff --git a/tools/importer-msgraph-metadata/components/parser/types.go b/tools/importer-msgraph-metadata/components/parser/types.go index 85fda525303..1dd061a7c14 100644 --- a/tools/importer-msgraph-metadata/components/parser/types.go +++ b/tools/importer-msgraph-metadata/components/parser/types.go @@ -14,9 +14,9 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) -/* =================== - openapi3 cheatsheet - =================== +/* ======================= + kin-openapi3 cheatsheet + ======================= Schemas is a map[string]*SchemaRef SchemaRef is a struct{Ref, Value} where Ref is a string, Value is a *Schema The Ref string (after trimming) indicates a Schemas map key to follow/inherit @@ -69,19 +69,39 @@ func (m Models) Found(schemaName string) bool { } type Constant struct { - Name string + // The name of this constant from the spec (not normalized) + Name string + + // Whether this constant is a common type Common bool - Enum []string - Type *DataType + + // The accepted values for this constant + Enum []string + + // The data type for this constant (currently only supports strings) + Type *DataType } type Model struct { - Name string - Fields map[string]*ModelField - Common bool - Parent bool - TypeField *string - TypeValue *string + // The type name of this model from the spec (not normalized) + Name string + + // Fields that comprise this model + Fields map[string]*ModelField + + // Whether this model is a common type + Common bool + + // Whether this model has known child models + Parent bool + + // For parent models, the field name containing the discriminated type value + TypeField *string + + // For child models, the type value that specifies this model + TypeValue *string + + // For child models, the name of the parent model ParentModel *string } @@ -93,6 +113,8 @@ func (m *Model) AppendDefaultFields() { } } +// DataApiSdkModel converts the internal ModelField representation to a Data API SDKModel, so it can be persisted to the Data +// API Definitions. It's necessary to provide Models and Constants so that references (both fields and model ancestry) can be resolved. func (m *Model) DataApiSdkModel(models Models, constants Constants) (*sdkModels.SDKModel, error) { sdkFields := make(map[string]sdkModels.SDKField) for jsonName, field := range m.Fields { @@ -147,41 +169,44 @@ func (m *Model) DataApiSdkModel(models Models, constants Constants) (*sdkModels. }, nil } -func defaultModelFields() map[string]*ModelField { - // Add an explicit ODataId and ODataType field to each model, since it is inconsistently defined in the API specs. - // This won't be valid for every model, but it's impossible to tell which models support them, and it's effectively - // harmless to leave these in so long as they have the `omitempty` struct tag in the generated SDK. - return map[string]*ModelField{ - "@odata.id": { - Name: "ODataId", - Description: "The OData ID of this entity", - Type: pointer.To(DataTypeString), - Default: "", - }, - "@odata.type": { - Name: "ODataType", - Description: "The OData Type of this entity", - Type: pointer.To(DataTypeString), - Default: "", - }, - } -} - type ModelField struct { - Name string - Type *DataType - Description string - Default interface{} - Required bool - ReadOnly bool - WriteOnly bool - Nullable bool - AllowEmptyValue bool + // The name of this field + Name string + + // The internal type for this field + Type *DataType + + // The internal type for items, when this field type is DataTypeArray + ItemType *DataType + + // Optional description which can be added to the generated SDK model as a comment + Description string + + // The default value for this field + Default any + + // Whether the field is required + Required bool + + // Read-only fields should be omitted during marshalling in the generated SDK + ReadOnly bool + + // Whether the field value can be a JSON null + Nullable bool + + // Whether this field contains the discriminated type for a child model DiscriminatedValue bool - ItemType *DataType - ReferenceName *string + + // The name of a referenced model or constant, noting that this should be the full type name from the spec prior to normalizing + ReferenceName *string + + // This is parsed from the spec but otherwise currently unused + WriteOnly bool + AllowEmptyValue bool } +// DataApiSdkObjectDefinition converts the internal ModelField representation to a Data API SDKObjectDefinition, so it can be +// persisted to the Data API Definitions. It's necessary to provide Models and Constants so that references can be resolved. func (f ModelField) DataApiSdkObjectDefinition(models Models, constants Constants) (*sdkModels.SDKObjectDefinition, error) { if f.Type == nil { return nil, fmt.Errorf("field %q has no Type", f.Name) @@ -604,7 +629,7 @@ func modelFieldFromSchemaRef(jsonField string, fieldSchema *openapi3.SchemaRef) field.ItemType = FieldType(items.Value.Type, items.Value.Format, field.ReferenceName != nil) } - // Detect nullable, read-only and required fields from the description + // Detect nullable, read-only and required fields from the description, which appears to be reliably auto-generated. if (strings.HasPrefix(fieldSchema.Value.Description, "Nullable.") || strings.Contains(fieldSchema.Value.Description, " Nullable.")) && !strings.Contains(strings.ToLower(fieldSchema.Value.Description), "not nullable.") { field.Nullable = true } @@ -622,6 +647,30 @@ func modelFieldFromSchemaRef(jsonField string, fieldSchema *openapi3.SchemaRef) return &field, nil } +// defaultModelFields adds an explicit ODataId and ODataType field to each model, since it is inconsistently defined in +// the API specs. This won't be valid for every model, but it's impossible to tell which models support them, and it's +// effectively harmless to leave these in so long as they have the `omitempty` struct tag in the generated SDK. +func defaultModelFields() map[string]*ModelField { + return map[string]*ModelField{ + "@odata.id": { + Name: "ODataId", + Description: "The OData ID of this entity", + Type: pointer.To(DataTypeString), + Default: "", + Required: false, + Nullable: false, + }, + "@odata.type": { + Name: "ODataType", + Description: "The OData Type of this entity", + Type: pointer.To(DataTypeString), + Default: "", + Required: false, + Nullable: false, + }, + } +} + // parseEnum returns a slice of sanitized enum values (which are always strings) func parseEnum(input []interface{}) []string { out := make([]string, 0) diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go index fea8279da61..127eed29ebc 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_conditionalaccesspolicy.go @@ -25,7 +25,7 @@ func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parse return fmt.Errorf("`ConditionalAccessPolicy` model not found") } - // grantControls and sessionControls must be null to unset them, so make them nullable + required + // `grantControls` and `sessionControls` must be null to unset them, so make them nullable + required if _, ok = model.Fields["grantControls"]; !ok { return fmt.Errorf("`GrantControls` field not found") } @@ -42,7 +42,7 @@ func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parse return fmt.Errorf("`ConditionalAccessConditionSet` model not found") } - // devices, locations, platforms must each be null to unset them, so make them nullable + required + // `devices`, `locations`, `platforms` must each be null to unset them, so make them nullable + required if _, ok = model.Fields["devices"]; !ok { return fmt.Errorf("`Devices` field not found") } @@ -88,7 +88,7 @@ func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parse return fmt.Errorf("`ConditionalAccessSessionControls` model not found") } - // cloudAppSecurityPolicy must be null to unset it, so make it nullable + required + // `cloudAppSecurityPolicy` must be null to unset it, so make it nullable + required if _, ok = model.Fields["cloudAppSecurity"]; !ok { return fmt.Errorf("`CloudAppSecurity` field not found") } @@ -100,7 +100,7 @@ func (workaroundConditionalAccessPolicy) Process(apiVersion string, models parse return fmt.Errorf("`ConditionalAccessUsers` model not found") } - // excludeGuestsOrExternalUsers / includeGuestsOrExternalUsers must be null to unset them, so make them nullable + required + // `excludeGuestsOrExternalUsers` / `includeGuestsOrExternalUsers` must be null to unset them, so make them nullable + required if _, ok = model.Fields["excludeGuestsOrExternalUsers"]; !ok { return fmt.Errorf("`ExcludeGuestsOrExternalUsers` field not found") } diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go index f8ed087bc4a..f3218be009a 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_odata_bind.go @@ -17,6 +17,7 @@ var _ dataWorkaround = workaroundODataBind{} // workaroundODataBind inserts an `@odata.bind` field where a field or collection refers to a DirectoryObject. The // OpenAPI spec unfortunately does not document relationships between entities. +// For example usage, see https://learn.microsoft.com/en-us/graph/api/group-post-groups?view=graph-rest-1.0&tabs=http#example-2-create-a-group-with-owners-and-members type workaroundODataBind struct{} func (workaroundODataBind) Name() string { diff --git a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go index b7e4136e0a3..3bf495b896d 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workaround_repeating_resource_id_segments.go @@ -11,6 +11,9 @@ import ( var _ dataWorkaround = workaroundRepeatingResourceIdSegments{} // workaroundRepeatingResourceIdSegments removes incompatible resource IDs due to repeating segments which are not supported at this time. +// After removing unsupported resource IDs, when a corresponding resource is being parsed, it will not match against any resource IDs, and +// the resulting URI suffix will contain the entire URL. If that resulting URI suffix contains any user-specified segments, the resource +// will also be skipped. type workaroundRepeatingResourceIdSegments struct{} func (workaroundRepeatingResourceIdSegments) Name() string { diff --git a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go index 305d5061cd6..7bf11829f6c 100644 --- a/tools/importer-msgraph-metadata/components/workarounds/workarounds.go +++ b/tools/importer-msgraph-metadata/components/workarounds/workarounds.go @@ -10,14 +10,21 @@ import ( "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/internal/logging" ) +// workarounds are general data workarounds that operate on models, constants and resource IDs. They can make any changes +// to this parsed data - note that each of these are passed in as maps, so changes propagate to the underlying object. var workarounds = []dataWorkaround{ + // Process general workarounds first workaroundODataBind{}, workaroundRepeatingResourceIdSegments{}, + // Model-specific workarounds workaroundApplication{}, workaroundConditionalAccessPolicy{}, } +// serviceWorkarounds make post-parsing changes to individual services and are able to make any changes to resources +// within that service and/or to resource IDs (which are shared across all services). Note that each of these are passed +// in as maps, so changes propagate to the underlying object. var serviceWorkarounds = []serviceWorkaround{ workaroundSynchronizationSecrets{}, } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/importer.go b/tools/importer-msgraph-metadata/internal/pipeline/importer.go index 5eaeef31a63..42d4e3bd283 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/importer.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/importer.go @@ -79,6 +79,7 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha return err } + // Parse all configured services for _, service := range config.Services { for _, version := range service.Available { if version == apiVersion { @@ -94,6 +95,7 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha return fmt.Errorf("unknown service was configured for API version %s: %#v", version, service) } + // Check that service is configured for import if len(input.Services) > 0 { skip := true @@ -133,17 +135,20 @@ func runImportForVersion(input RunInput, apiVersion, openApiFile, metadataGitSha } p.resourceIds = usedResourceIds + // Translate common types for this API version commonTypesForApiVersion, err := p.translateCommonTypesToDataApiSdkTypes() if err != nil { return err } + // Translate and persist all services to Data API Definitions for service := range p.resources { if err = p.ForService(service).PersistDefinitions(*commonTypesForApiVersion); err != nil { return err } } + // Persist all common types to Data API Definitions if err = p.PersistCommonTypesDefinitions(*commonTypesForApiVersion); err != nil { return err } @@ -161,7 +166,7 @@ func (p pipelineForService) RunImport() error { return nil } - // Apply workarounds + // Apply workarounds for this service if err = workarounds.ApplyWorkaroundsForService(p.apiVersion, p.service, resources, p.resourceIds); err != nil { return err } diff --git a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go index d67aa9537f1..502e0f84999 100644 --- a/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go +++ b/tools/importer-msgraph-metadata/internal/pipeline/task_translate_service.go @@ -9,7 +9,6 @@ import ( "github.com/hashicorp/go-azure-helpers/lang/pointer" sdkModels "github.com/hashicorp/pandora/tools/data-api-sdk/v1/models" - "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/blacklisted" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/normalize" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/parser" "github.com/hashicorp/pandora/tools/importer-msgraph-metadata/components/versions" @@ -26,10 +25,6 @@ func (p pipelineForService) translateServiceToDataApiSdkTypes() (*sdkModels.Serv } for _, resource := range p.resources[p.service] { - if blacklisted.Resource(resource) { - continue - } - // First scaffold the version and SDK resources (categories) if _, ok := sdkService.APIVersions[resource.Version]; !ok { sdkService.APIVersions[resource.Version] = sdkModels.APIVersion{