From 7bccda3aed24e02d5b6a8339b96da39aea844b3f Mon Sep 17 00:00:00 2001 From: Ariel Silverman Date: Mon, 23 Oct 2023 09:01:34 -0700 Subject: [PATCH] Implement provider declaration aliasing (#12127) # Overview Implementation of https://github.com/Azure/bicep/issues/11598 ###### Microsoft Reviewers: [Open in CodeFlow](https://microsoft.github.io/open-pr/?codeflow=https://github.com/Azure/bicep/pull/12127) --------- Co-authored-by: asilverman --- .../BuildCommandTests.cs | 56 +++++++++--- src/Bicep.Cli/Commands/PublishCommand.cs | 2 +- .../ExtensibilityTests.cs | 4 +- .../ImportTests.cs | 20 ++--- .../RegistryTests.cs | 10 +-- src/Bicep.Core.Samples/DataSetsExtensions.cs | 15 ++-- .../moduleWithAzImport.bicep | 2 +- .../BicepTestConstants.cs | 1 + .../ConfigurationManagerTests.cs | 40 +++++++++ .../Configuration/RootConfigurationTests.cs | 1 + .../Diagnostics/ErrorBuilderTests.cs | 6 ++ ...OutputsShouldNotContainSecretsRuleTests.cs | 3 + .../UseRecentApiVersionRuleTests.cs | 1 + .../OciArtifactModuleReferenceTests.cs | 18 ++-- .../Registry/ArtifactDispatcherTests.cs | 20 ++--- .../Registry/OciModuleRegistryTests.cs | 81 ++++++++--------- .../Syntax/ImportSpecificationTest.cs | 41 +++++++++ .../Utils/OciModuleRegistryHelper.cs | 23 ++--- .../AnalyzersConfigurationExtensions.cs | 2 + .../Configuration/ConfigurationManager.cs | 1 + .../ProviderAliasesConfiguration.cs | 88 +++++++++++++++++++ .../Configuration/RootConfiguration.cs | 12 ++- src/Bicep.Core/Configuration/bicepconfig.json | 8 ++ .../Diagnostics/DiagnosticBuilder.cs | 27 +++++- .../CompileTimeImports/ImportClosureInfo.cs | 4 +- src/Bicep.Core/Emit/ScopeHelper.cs | 2 +- src/Bicep.Core/Modules/OciModuleReference.cs | 45 ---------- .../Navigation/IArtifactReferenceSyntax.cs | 3 + src/Bicep.Core/Registry/ArtifactRegistry.cs | 2 +- src/Bicep.Core/Registry/ArtifactType.cs | 11 +++ .../Registry/IArtifactDispatcher.cs | 16 ++-- ...actory.cs => IArtifactReferenceFactory.cs} | 6 +- src/Bicep.Core/Registry/IArtifactRegistry.cs | 19 ++-- .../Registry/LocalModuleRegistry.cs | 6 +- src/Bicep.Core/Registry/ModuleDispatcher.cs | 82 +++++++++-------- .../Registry/ModuleDispatcherExtensions.cs | 2 +- .../Registry/Oci/OciArtifactReference.cs | 47 +++++++--- .../Registry/OciArtifactRegistry.cs | 46 +++++----- .../Registry/TemplateSpecModuleRegistry.cs | 6 +- src/Bicep.Core/Semantics/Compilation.cs | 6 +- src/Bicep.Core/Semantics/ImportedSymbol.cs | 4 +- .../Semantics/WildcardImportSymbol.cs | 4 +- .../CompileTimeImportDeclarationSyntax.cs | 3 + src/Bicep.Core/Syntax/ImportSpecification.cs | 79 ++++++++++++----- .../Syntax/ModuleDeclarationSyntax.cs | 3 + .../Syntax/ProviderDeclarationSyntax.cs | 5 +- .../Syntax/TestDeclarationSyntax.cs | 3 + .../Syntax/UsingDeclarationSyntax.cs | 3 + .../Workspaces/SourceFileGroupingBuilder.cs | 36 ++++---- .../HoverTests.cs | 2 +- .../Registry/ModuleRestoreSchedulerTests.cs | 12 +-- .../BicepRegistryCacheRequestHandlerTests.cs | 33 +++---- .../Handlers/BicepDefinitionHandler.cs | 18 ++-- .../BicepForceModulesRestoreCommandHandler.cs | 2 +- .../Handlers/BicepHoverHandler.cs | 2 +- .../BicepRegistryCacheRequestHandler.cs | 4 +- .../schemas/bicepconfig.schema.json | 82 ++++++++++++++--- 57 files changed, 720 insertions(+), 360 deletions(-) create mode 100644 src/Bicep.Core.UnitTests/Syntax/ImportSpecificationTest.cs create mode 100644 src/Bicep.Core/Configuration/ProviderAliasesConfiguration.cs delete mode 100644 src/Bicep.Core/Modules/OciModuleReference.cs create mode 100644 src/Bicep.Core/Registry/ArtifactType.cs rename src/Bicep.Core/Registry/{IModuleReferenceFactory.cs => IArtifactReferenceFactory.cs} (51%) diff --git a/src/Bicep.Cli.IntegrationTests/BuildCommandTests.cs b/src/Bicep.Cli.IntegrationTests/BuildCommandTests.cs index 749b8f6f9b3..5f7de852725 100644 --- a/src/Bicep.Cli.IntegrationTests/BuildCommandTests.cs +++ b/src/Bicep.Cli.IntegrationTests/BuildCommandTests.cs @@ -103,15 +103,21 @@ public async Task Build_Valid_SingleFile_WithTemplateSpecReference_ShouldSucceed actualLocation: compiledFilePath); } - [TestMethod] - public async Task Provider_Artifacts_Restore_From_Registry_ShouldSucceed() + [DataTestMethod] + [DataRow("br:mcr.microsoft.com/bicep/providers/az", true)] + [DataRow("br/public:az", true)] + [DataRow("br/contoso:az", true)] + // Negative + [DataRow("az", false)] + public async Task Build_Valid_SingleFile_WithProviderDeclarationStatement(string providerDeclarationSyntax, bool shouldSucceed) { // SETUP // 1. create a mock registry client - var registryUri = new Uri($"https://{LanguageConstants.BicepPublicMcrRegistry}"); - var repository = $"bicep/providers/az"; - var (clientFactory, blobClients) = DataSetsExtensions.CreateMockRegistryClients(false, (registryUri, repository)); - var myClient = blobClients[(registryUri, repository)]; + var mcrUri = new Uri($"https://{LanguageConstants.BicepPublicMcrRegistry}"); + var registyUris = new[] { mcrUri }; + var repository = "bicep/providers/az"; + var (clientFactory, blobClients) = DataSetsExtensions.CreateMockRegistryClients(false, (mcrUri, repository)); + var myClient = blobClients[(mcrUri, repository)]; // 2. upload a manifest and its blob layer var manifestStr = $$""" @@ -142,14 +148,30 @@ public async Task Provider_Artifacts_Restore_From_Registry_ShouldSucceed() await myClient.UploadBlobAsync(new MemoryStream()); // 3. create a main.bicep and save it to a output directory - var bicepFile = """ -import 'az@2.0.0' + var bicepFile = $""" +import '{providerDeclarationSyntax}@2.0.0' """; var tempDirectory = FileHelper.GetUniqueTestOutputPath(TestContext); Directory.CreateDirectory(tempDirectory); var bicepFilePath = Path.Combine(tempDirectory, "main.bicep"); File.WriteAllText(bicepFilePath, bicepFile); + + var bicepConfigFile = $$""" +{ + "providerAliases" : { + "br": { + "contoso": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + } +} +"""; + var bicepConfigPath = Path.Combine(tempDirectory, "bicepconfig.json"); + File.WriteAllText(bicepConfigPath, bicepConfigFile); + // 4. create a settings object with the mock registry client and relevant features enabled var settings = new InvocationSettings(new(TestContext, RegistryEnabled: true, ExtensibilityEnabled: true, DynamicTypeLoading: true), clientFactory.Object, Repository.Create().Object); @@ -161,14 +183,20 @@ public async Task Provider_Artifacts_Restore_From_Registry_ShouldSucceed() // 6. assert 'bicep build' completed successfully using (new AssertionScope()) { - result.Should().Be(0); + result.Should().Be(shouldSucceed ? 0 : 1); output.Should().BeEmpty(); - AssertNoErrors(error); + if (shouldSucceed) + { + AssertNoErrors(error); + } + } + if (shouldSucceed) + { + // 7. assert the provider files were restored to the cache directory + Directory.Exists(settings.FeatureOverrides.CacheRootDirectory).Should().BeTrue(); + var providerDir = Path.Combine(settings.FeatureOverrides.CacheRootDirectory!, ModuleReferenceSchemes.Oci, LanguageConstants.BicepPublicMcrRegistry, "bicep$providers$az", "2.0.0$"); + Directory.EnumerateFiles(providerDir).ToList().Select(Path.GetFileName).Should().BeEquivalentTo(new List { "types.tgz", "lock", "manifest", "metadata" }); } - // 7. assert the provider files were restored to the cache directory - Directory.Exists(settings.FeatureOverrides.CacheRootDirectory).Should().BeTrue(); - var providerDir = Path.Combine(settings.FeatureOverrides.CacheRootDirectory!, ModuleReferenceSchemes.Oci, LanguageConstants.BicepPublicMcrRegistry, "bicep$providers$az", "2.0.0$"); - Directory.EnumerateFiles(providerDir).ToList().Select(Path.GetFileName).Should().BeEquivalentTo(new List { "types.tgz", "lock", "manifest", "metadata" }); } [DataTestMethod] diff --git a/src/Bicep.Cli/Commands/PublishCommand.cs b/src/Bicep.Cli/Commands/PublishCommand.cs index 7b8efad7cdc..600a6c31b97 100644 --- a/src/Bicep.Cli/Commands/PublishCommand.cs +++ b/src/Bicep.Cli/Commands/PublishCommand.cs @@ -98,7 +98,7 @@ private async Task PublishModuleAsync(ArtifactReference target, Stream compiledA private ArtifactReference ValidateReference(string targetModuleReference, Uri targetModuleUri) { - if (!this.moduleDispatcher.TryGetModuleReference(targetModuleReference, targetModuleUri).IsSuccess(out var moduleReference, out var failureBuilder)) + if (!this.moduleDispatcher.TryGetArtifactReference(ArtifactType.Module, targetModuleReference, targetModuleUri).IsSuccess(out var moduleReference, out var failureBuilder)) { // TODO: We should probably clean up the dispatcher contract so this sort of thing isn't necessary (unless we change how target module is set in this command) var message = failureBuilder(DiagnosticBuilder.ForDocumentStart()).Message; diff --git a/src/Bicep.Core.IntegrationTests/ExtensibilityTests.cs b/src/Bicep.Core.IntegrationTests/ExtensibilityTests.cs index a697fc97003..79c9ad35fd1 100644 --- a/src/Bicep.Core.IntegrationTests/ExtensibilityTests.cs +++ b/src/Bicep.Core.IntegrationTests/ExtensibilityTests.cs @@ -624,7 +624,7 @@ param connectionString string public void Az_namespace_can_be_used_without_configuration() { var result = CompilationHelper.Compile(Services, @" -import 'az@1.0.0' +import 'br/public:az@1.0.0' "); result.Should().GenerateATemplate(); @@ -635,7 +635,7 @@ public void Az_namespace_can_be_used_without_configuration() public void Az_namespace_errors_with_configuration() { var result = CompilationHelper.Compile(Services, @" -import 'az@1.0.0' with {} +import 'br/public:az@1.0.0' with {} "); result.Should().NotGenerateATemplate(); diff --git a/src/Bicep.Core.IntegrationTests/ImportTests.cs b/src/Bicep.Core.IntegrationTests/ImportTests.cs index 8ff7a0bb9de..e475dfc93ee 100644 --- a/src/Bicep.Core.IntegrationTests/ImportTests.cs +++ b/src/Bicep.Core.IntegrationTests/ImportTests.cs @@ -54,7 +54,7 @@ public TestNamespaceProvider(Dictionary> bui public void Imports_are_disabled_unless_feature_is_enabled() { var result = CompilationHelper.Compile(@" -import 'az@1.0.0' +import 'br/public:az@1.0.0' "); result.Should().HaveDiagnostics(new[] { ("BCP203", DiagnosticLevel.Error, "Using import statements requires enabling EXPERIMENTAL feature \"Extensibility\"."), @@ -74,7 +74,7 @@ public void Import_statement_parse_diagnostics_are_guiding() }); result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' blahblah +import 'br/public:az@1.0.0' blahblah "); result.Should().HaveDiagnostics(new[] { ("BCP305", DiagnosticLevel.Error, "Expected the \"with\" keyword, \"as\" keyword, or a new line character at this location."), @@ -108,7 +108,7 @@ public void Import_statement_parse_diagnostics_are_guiding() }); result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' as +import 'br/public:az@1.0.0' as "); result.Should().HaveDiagnostics(new[] { ("BCP202", DiagnosticLevel.Error, "Expected an import alias name at this location."), @@ -130,7 +130,7 @@ public void Imports_return_error_with_unrecognized_namespace() public void Import_configuration_is_blocked_by_default() { var result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' with { +import 'br/public:az@1.0.0' with { foo: 'bar' } "); @@ -143,7 +143,7 @@ public void Import_configuration_is_blocked_by_default() public void Using_import_statements_frees_up_the_namespace_symbol() { var result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' as newAz +import 'br/public:az@1.0.0' as newAz var az = 'Fake AZ!' var myRg = newAz.resourceGroup() @@ -159,7 +159,7 @@ public void Using_import_statements_frees_up_the_namespace_symbol() public void You_can_swap_imported_namespaces_if_you_really_really_want_to() { var result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' as sys +import 'br/public:az@1.0.0' as sys import 'sys@1.0.0' as az var myRg = sys.resourceGroup() @@ -176,7 +176,7 @@ public void You_can_swap_imported_namespaces_if_you_really_really_want_to() public void Overwriting_single_built_in_namespace_with_import_is_prohibited() { var result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' as sys +import 'br/public:az@1.0.0' as sys var myRg = sys.resourceGroup() @@ -190,8 +190,8 @@ public void Overwriting_single_built_in_namespace_with_import_is_prohibited() public void Singleton_imports_cannot_be_used_multiple_times() { var result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' as az1 -import 'az@1.0.0' as az2 +import 'br/public:az@1.0.0' as az1 +import 'br/public:az@1.0.0' as az2 import 'sys@1.0.0' as sys1 import 'sys@1.0.0' as sys2 @@ -209,7 +209,7 @@ public void Singleton_imports_cannot_be_used_multiple_times() public void Import_names_must_not_conflict_with_other_symbols() { var result = CompilationHelper.Compile(ServicesWithImports, @" -import 'az@1.0.0' +import 'br/public:az@1.0.0' import 'kubernetes@1.0.0' with { kubeConfig: '' namespace: '' diff --git a/src/Bicep.Core.IntegrationTests/RegistryTests.cs b/src/Bicep.Core.IntegrationTests/RegistryTests.cs index 05b90cae2f2..45cf2259d7d 100644 --- a/src/Bicep.Core.IntegrationTests/RegistryTests.cs +++ b/src/Bicep.Core.IntegrationTests/RegistryTests.cs @@ -249,7 +249,7 @@ public async Task ModuleRestoreWithStuckFileLockShouldFailAfterTimeout(IEnumerab dispatcher.GetArtifactRestoreStatus(moduleReference, out _).Should().Be(ArtifactRestoreStatus.Unknown); } - dispatcher.TryGetLocalModuleEntryPointUri(moduleReferences[0]).IsSuccess(out var moduleFileUri).Should().BeTrue(); + dispatcher.TryGetLocalArtifactEntryPointUri(moduleReferences[0]).IsSuccess(out var moduleFileUri).Should().BeTrue(); moduleFileUri.Should().NotBeNull(); var moduleFilePath = moduleFileUri!.LocalPath; @@ -318,7 +318,7 @@ public async Task ForceModuleRestoreWithStuckFileLockShouldFailAfterTimeout(IEnu dispatcher.GetArtifactRestoreStatus(moduleReference, out _).Should().Be(ArtifactRestoreStatus.Unknown); } - dispatcher.TryGetLocalModuleEntryPointUri(moduleReferences[0]).IsSuccess(out var moduleFileUri).Should().BeTrue(); + dispatcher.TryGetLocalArtifactEntryPointUri(moduleReferences[0]).IsSuccess(out var moduleFileUri).Should().BeTrue(); moduleFileUri.Should().NotBeNull(); var moduleFilePath = moduleFileUri!.LocalPath; @@ -334,7 +334,7 @@ public async Task ForceModuleRestoreWithStuckFileLockShouldFailAfterTimeout(IEnu // let's try to restore a module while holding a lock using (@lock) { - (await dispatcher.RestoreModules(moduleReferences, forceModulesRestore: true)).Should().BeTrue(); + (await dispatcher.RestoreModules(moduleReferences, forceRestore: true)).Should().BeTrue(); } // REF: FileLockTests.cs/FileLockShouldNotThrowIfLockFileIsDeleted() @@ -395,14 +395,14 @@ public async Task ForceModuleRestoreShouldRestoreAllModules(IEnumerable CreateMockRegistryClients(Im { var target = publishInfo.Metadata.Target; - if (!dispatcher.TryGetModuleReference(target, RandomFileUri()).IsSuccess(out var @ref) || @ref is not OciModuleReference targetReference) + if (!dispatcher.TryGetArtifactReference(ArtifactType.Module, target, RandomFileUri()).IsSuccess(out var @ref) || @ref is not OciArtifactReference targetReference) { throw new InvalidOperationException($"Module '{moduleName}' has an invalid target reference '{target}'. Specify a reference to an OCI artifact."); } - Uri registryUri = new Uri($"https://{targetReference.Registry}"); + Uri registryUri = new($"https://{targetReference.Registry}"); clients.Add((registryUri, targetReference.Repository)); } @@ -94,9 +95,9 @@ public static (Mock factoryMock, ImmutableDicti .AddSingleton(featureProviderFactory) ).Construct(); - foreach (var client in clients) + foreach (var (registryUri, repository) in clients) { - clientsBuilder.TryAdd((client.registryUri, client.repository), new MockRegistryBlobClient()); + clientsBuilder.TryAdd((registryUri, repository), new MockRegistryBlobClient()); } var repoToClient = clientsBuilder.ToImmutable(); @@ -133,7 +134,7 @@ public static (Mock factoryMock, ImmutableDicti public static ITemplateSpecRepositoryFactory CreateEmptyTemplateSpecRepositoryFactory(bool enablePublishSource = false) => CreateMockTemplateSpecRepositoryFactory(ImmutableDictionary.Empty, enablePublishSource); - public static ITemplateSpecRepositoryFactory CreateMockTemplateSpecRepositoryFactory(this DataSet dataSet, TestContext testContext, bool enablePublishSource = false) + public static ITemplateSpecRepositoryFactory CreateMockTemplateSpecRepositoryFactory(this DataSet dataSet, TestContext _, bool enablePublishSource = false) => CreateMockTemplateSpecRepositoryFactory(dataSet.TemplateSpecs, enablePublishSource); public static ITemplateSpecRepositoryFactory CreateMockTemplateSpecRepositoryFactory(ImmutableDictionary templateSpecs, bool enablePublishSource = false) @@ -148,7 +149,7 @@ public static ITemplateSpecRepositoryFactory CreateMockTemplateSpecRepositoryFac foreach (var (moduleName, templateSpecInfo) in templateSpecs) { - if (!dispatcher.TryGetModuleReference(templateSpecInfo.Metadata.Target, RandomFileUri()).IsSuccess(out var @ref) || @ref is not TemplateSpecModuleReference reference) + if (!dispatcher.TryGetArtifactReference(ArtifactType.Module, templateSpecInfo.Metadata.Target, RandomFileUri()).IsSuccess(out var @ref) || @ref is not TemplateSpecModuleReference reference) { throw new InvalidOperationException($"Module '{moduleName}' has an invalid target reference '{templateSpecInfo.Metadata.Target}'. Specify a reference to a template spec."); } @@ -190,7 +191,7 @@ public static async Task PublishModuleToRegistryAsync(IContainerRegistryClientFa .AddSingleton(featureProviderFactory) ).Construct(); - var targetReference = dispatcher.TryGetModuleReference(target, RandomFileUri()).IsSuccess(out var @ref) ? @ref + var targetReference = dispatcher.TryGetArtifactReference(ArtifactType.Module, target, RandomFileUri()).IsSuccess(out var @ref) ? @ref : throw new InvalidOperationException($"Module '{moduleName}' has an invalid target reference '{target}'. Specify a reference to an OCI artifact."); var result = CompilationHelper.Compile(moduleSource); diff --git a/src/Bicep.Core.Samples/Files/baselines/Modules_CRLF/child/folder with separate config/moduleWithAzImport.bicep b/src/Bicep.Core.Samples/Files/baselines/Modules_CRLF/child/folder with separate config/moduleWithAzImport.bicep index 1025c4a3467..f6ecace83a7 100644 --- a/src/Bicep.Core.Samples/Files/baselines/Modules_CRLF/child/folder with separate config/moduleWithAzImport.bicep +++ b/src/Bicep.Core.Samples/Files/baselines/Modules_CRLF/child/folder with separate config/moduleWithAzImport.bicep @@ -1,3 +1,3 @@ -import 'az@1.0.0' +import 'br/public:az@1.0.0' output str string = 'foo' diff --git a/src/Bicep.Core.UnitTests/BicepTestConstants.cs b/src/Bicep.Core.UnitTests/BicepTestConstants.cs index f6b0fc0fb6e..a5aca255f84 100644 --- a/src/Bicep.Core.UnitTests/BicepTestConstants.cs +++ b/src/Bicep.Core.UnitTests/BicepTestConstants.cs @@ -87,6 +87,7 @@ public static RootConfiguration CreateMockConfiguration(Dictionary(), + ["providerAliases"] = new Dictionary(), ["analyzers"] = new Dictionary(), ["experimentalFeaturesEnabled"] = new Dictionary(), ["formatting"] = new Dictionary(), diff --git a/src/Bicep.Core.UnitTests/Configuration/ConfigurationManagerTests.cs b/src/Bicep.Core.UnitTests/Configuration/ConfigurationManagerTests.cs index b1ec18b64a8..3587d7d2f82 100644 --- a/src/Bicep.Core.UnitTests/Configuration/ConfigurationManagerTests.cs +++ b/src/Bicep.Core.UnitTests/Configuration/ConfigurationManagerTests.cs @@ -64,6 +64,14 @@ public void GetBuiltInConfiguration_NoParameter_ReturnsBuiltInConfigurationWithA } } }, + "providerAliases": { + "br": { + "public": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + }, "analyzers": { "core": { "verbose": false, @@ -168,6 +176,14 @@ public void GetBuiltInConfiguration_DisableAllAnalyzers_ReturnsBuiltInConfigurat } } }, + "providerAliases": { + "br": { + "public": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + }, "analyzers": {}, "experimentalFeaturesEnabled": { "symbolicNameCodegen": false, @@ -234,6 +250,14 @@ public void GetBuiltInConfiguration_DisableAnalyzers_ReturnsBuiltInConfiguration } } }, + "providerAliases": { + "br": { + "public": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + }, "analyzers": { "core": { "verbose": false, @@ -483,6 +507,14 @@ public void GetConfiguration_ValidCustomConfiguration_OverridesBuiltInConfigurat } } }, + "providerAliases": { + "br": { + "public": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + }, "analyzers": { "core": { "enabled": false, @@ -573,6 +605,14 @@ public void GetConfiguration_ValidCustomConfiguration_OverridesBuiltInConfigurat } } }, + "providerAliases": { + "br": { + "public": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + }, "analyzers": { "core": { "verbose": false, diff --git a/src/Bicep.Core.UnitTests/Configuration/RootConfigurationTests.cs b/src/Bicep.Core.UnitTests/Configuration/RootConfigurationTests.cs index f886eff26d4..6d187e31389 100644 --- a/src/Bicep.Core.UnitTests/Configuration/RootConfigurationTests.cs +++ b/src/Bicep.Core.UnitTests/Configuration/RootConfigurationTests.cs @@ -21,6 +21,7 @@ public void RootConfiguration_LeadingTildeInCacheRootDirectory_ExpandPath(string var configuration = new RootConfiguration( BicepTestConstants.BuiltInConfiguration.Cloud, BicepTestConstants.BuiltInConfiguration.ModuleAliases, + BicepTestConstants.BuiltInConfiguration.ProviderAliases, BicepTestConstants.BuiltInConfiguration.Analyzers, cacheRootDirectory, BicepTestConstants.BuiltInConfiguration.ExperimentalFeaturesEnabled, diff --git a/src/Bicep.Core.UnitTests/Diagnostics/ErrorBuilderTests.cs b/src/Bicep.Core.UnitTests/Diagnostics/ErrorBuilderTests.cs index 23a68c083db..366ec9f2812 100644 --- a/src/Bicep.Core.UnitTests/Diagnostics/ErrorBuilderTests.cs +++ b/src/Bicep.Core.UnitTests/Diagnostics/ErrorBuilderTests.cs @@ -7,6 +7,7 @@ using System.Reflection; using Bicep.Core.Diagnostics; using Bicep.Core.Parsing; +using Bicep.Core.Registry; using Bicep.Core.Resources; using Bicep.Core.Semantics; using Bicep.Core.Semantics.Metadata; @@ -192,6 +193,11 @@ private static object CreateMockParameter(ParameterInfo parameter, int index) return BicepSourceFileKind.BicepFile; } + if (parameter.ParameterType == typeof(ArtifactType)) + { + return ArtifactType.Module; + } + throw new AssertFailedException($"Unable to generate mock parameter value of type '{parameter.ParameterType}' for the diagnostic builder method."); } diff --git a/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/OutputsShouldNotContainSecretsRuleTests.cs b/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/OutputsShouldNotContainSecretsRuleTests.cs index b1db43d0edd..65c0fc2bea9 100644 --- a/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/OutputsShouldNotContainSecretsRuleTests.cs +++ b/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/OutputsShouldNotContainSecretsRuleTests.cs @@ -137,6 +137,7 @@ public void If_OutputReferencesSecureParamProperty_ShouldFail(string text, param CompileAndTest(text, OnCompileErrors.IncludeErrors, expectedMessages, config => new( config.Cloud, config.ModuleAliases, + config.ProviderAliases, config.Analyzers, config.CacheRootDirectory, config.ExperimentalFeaturesEnabled with { UserDefinedTypes = true }, @@ -228,6 +229,7 @@ public void If_OutputReferencesParamWithSecureProperty_ShouldFail(string text, p CompileAndTest(text, OnCompileErrors.IncludeErrors, expectedMessages, config => new( config.Cloud, config.ModuleAliases, + config.ProviderAliases, config.Analyzers, config.CacheRootDirectory, config.ExperimentalFeaturesEnabled with { UserDefinedTypes = true }, @@ -279,6 +281,7 @@ public void If_OutputReferencesNonSecureParamProperty_ShouldPass(string text, pa CompileAndTest(text, OnCompileErrors.IncludeErrors, expectedMessages, config => new( config.Cloud, config.ModuleAliases, + config.ProviderAliases, config.Analyzers, config.CacheRootDirectory, config.ExperimentalFeaturesEnabled with { UserDefinedTypes = true }, diff --git a/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/UseRecentApiVersionRuleTests.cs b/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/UseRecentApiVersionRuleTests.cs index 67182c3bab7..ee5c3807819 100644 --- a/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/UseRecentApiVersionRuleTests.cs +++ b/src/Bicep.Core.UnitTests/Diagnostics/LinterRuleTests/UseRecentApiVersionRuleTests.cs @@ -109,6 +109,7 @@ private static RootConfiguration CreateConfigurationWithFakeToday(RootConfigurat return new RootConfiguration( original.Cloud, original.ModuleAliases, + original.ProviderAliases, new AnalyzersConfiguration( JsonElementFactory.CreateElement(@" { diff --git a/src/Bicep.Core.UnitTests/Modules/OciArtifactModuleReferenceTests.cs b/src/Bicep.Core.UnitTests/Modules/OciArtifactModuleReferenceTests.cs index 08f2bb17df0..894ade5cfab 100644 --- a/src/Bicep.Core.UnitTests/Modules/OciArtifactModuleReferenceTests.cs +++ b/src/Bicep.Core.UnitTests/Modules/OciArtifactModuleReferenceTests.cs @@ -9,6 +9,8 @@ using Bicep.Core.Configuration; using Bicep.Core.FileSystem; using Bicep.Core.Modules; +using Bicep.Core.Registry; +using Bicep.Core.Registry.Oci; using Bicep.Core.UnitTests.Assertions; using FluentAssertions; using FluentAssertions.Execution; @@ -103,7 +105,7 @@ public void ValidReferenceShouldBeUriParseable(ValidCase @case) [DataTestMethod] public void InvalidReferencesShouldProduceExpectedError(string value, string expectedCode, string expectedError) { - OciModuleReference.TryParse(null, value, BicepTestConstants.BuiltInConfigurationWithAllAnalyzersDisabled, RandomFileUri()).IsSuccess(out var @ref, out var failureBuilder).Should().BeFalse(); + OciArtifactReference.TryParse(ArtifactType.Module, null, value, BicepTestConstants.BuiltInConfigurationWithAllAnalyzersDisabled, RandomFileUri()).IsSuccess(out var @ref, out var failureBuilder).Should().BeFalse(); @ref.Should().BeNull(); failureBuilder!.Should().NotBeNull(); @@ -145,7 +147,7 @@ public void MismatchedReferencesShouldNotBeEqual(string package1, string package [DataRow("foo bar ÄÄÄ")] public void TryParse_InvalidAliasName_ReturnsFalseAndSetsErrorDiagnostic(string aliasName) { - OciModuleReference.TryParse(aliasName, "", BicepTestConstants.BuiltInConfiguration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeFalse(); + OciArtifactReference.TryParse(ArtifactType.Module, aliasName, "", BicepTestConstants.BuiltInConfiguration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeFalse(); reference.Should().BeNull(); errorBuilder!.Should().HaveCode("BCP211"); @@ -159,7 +161,7 @@ public void TryParse_AliasNotInConfiguration_ReturnsFalseAndSetsErrorDiagnostic( { var configuration = BicepTestConstants.CreateMockConfiguration(configurationPath: configurationPath); - OciModuleReference.TryParse(aliasName, referenceValue, configuration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeFalse(); + OciArtifactReference.TryParse(ArtifactType.Module, aliasName, referenceValue, configuration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeFalse(); reference.Should().BeNull(); errorBuilder!.Should().NotBeNull(); @@ -171,7 +173,7 @@ public void TryParse_AliasNotInConfiguration_ReturnsFalseAndSetsErrorDiagnostic( [DynamicData(nameof(GetInvalidAliasData), DynamicDataSourceType.Method)] public void TryParse_InvalidAlias_ReturnsFalseAndSetsErrorDiagnostic(string aliasName, string referenceValue, RootConfiguration configuration, string expectedCode, string expectedMessage) { - OciModuleReference.TryParse(aliasName, referenceValue, configuration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeFalse(); + OciArtifactReference.TryParse(ArtifactType.Module, aliasName, referenceValue, configuration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeFalse(); reference.Should().BeNull(); errorBuilder!.Should().NotBeNull(); @@ -183,22 +185,22 @@ public void TryParse_InvalidAlias_ReturnsFalseAndSetsErrorDiagnostic(string alia [DynamicData(nameof(GetValidAliasData), DynamicDataSourceType.Method)] public void TryGetModuleReference_ValidAlias_ReplacesReferenceValue(string aliasName, string referenceValue, string fullyQualifiedReferenceValue, RootConfiguration configuration) { - OciModuleReference.TryParse(aliasName, referenceValue, configuration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeTrue(); + OciArtifactReference.TryParse(ArtifactType.Module, aliasName, referenceValue, configuration, RandomFileUri()).IsSuccess(out var reference, out var errorBuilder).Should().BeTrue(); reference.Should().NotBeNull(); reference!.FullyQualifiedReference.Should().Be(fullyQualifiedReferenceValue); } - private static OciModuleReference Parse(string package) + private static OciArtifactReference Parse(string package) { - OciModuleReference.TryParse(null, package, BicepTestConstants.BuiltInConfigurationWithAllAnalyzersDisabled, RandomFileUri()).IsSuccess(out var parsed, out var failureBuilder).Should().BeTrue(); + OciArtifactReference.TryParse(ArtifactType.Module, null, package, BicepTestConstants.BuiltInConfigurationWithAllAnalyzersDisabled, RandomFileUri()).IsSuccess(out var parsed, out var failureBuilder).Should().BeTrue(); failureBuilder!.Should().BeNull(); parsed.Should().NotBeNull(); return parsed!; } - private static (OciModuleReference, OciModuleReference) ParsePair(string first, string second) => (Parse(first), Parse(second)); + private static (OciArtifactReference, OciArtifactReference) ParsePair(string first, string second) => (Parse(first), Parse(second)); private static IEnumerable GetValidCases() { diff --git a/src/Bicep.Core.UnitTests/Registry/ArtifactDispatcherTests.cs b/src/Bicep.Core.UnitTests/Registry/ArtifactDispatcherTests.cs index 7f5c8e05547..ac8a2fd544f 100644 --- a/src/Bicep.Core.UnitTests/Registry/ArtifactDispatcherTests.cs +++ b/src/Bicep.Core.UnitTests/Registry/ArtifactDispatcherTests.cs @@ -41,7 +41,7 @@ public void NoRegistries_ValidateModuleReference_ShouldReturnError() var dispatcher = CreateDispatcher(BicepTestConstants.ConfigurationManager); var configuration = BicepTestConstants.BuiltInConfiguration; - dispatcher.TryGetModuleReference(module, RandomFileUri()).IsSuccess(out var reference, out var failureBuilder).Should().BeFalse(); + dispatcher.TryGetArtifactReference(module, RandomFileUri()).IsSuccess(out var reference, out var failureBuilder).Should().BeFalse(); reference.Should().BeNull(); failureBuilder!.Should().NotBeNull(); @@ -52,7 +52,7 @@ public void NoRegistries_ValidateModuleReference_ShouldReturnError() } var localModule = CreateModule("test.bicep"); - dispatcher.TryGetModuleReference(localModule, RandomFileUri()).IsSuccess(out var localModuleReference, out var localModuleFailureBuilder).Should().BeFalse(); + dispatcher.TryGetArtifactReference(localModule, RandomFileUri()).IsSuccess(out var localModuleReference, out var localModuleFailureBuilder).Should().BeFalse(); localModuleReference.Should().BeNull(); failureBuilder!.Should().NotBeNull(); using (new AssertionScope()) @@ -89,20 +89,20 @@ public async Task MockRegistries_ModuleLifecycle() var validRefUri = RandomFileUri(); ArtifactReference? validRef = new MockModuleReference("validRef", validRefUri); - mock.Setup(m => m.TryParseArtifactReference(null, "validRef")).Returns(ResultHelper.Create(validRef, @null)); + mock.Setup(m => m.TryParseArtifactReference(ArtifactType.Module, null, "validRef")).Returns(ResultHelper.Create(validRef, @null)); var validRefUri2 = RandomFileUri(); ArtifactReference? validRef2 = new MockModuleReference("validRef2", validRefUri2); - mock.Setup(m => m.TryParseArtifactReference(null, "validRef2")).Returns(ResultHelper.Create(validRef2, @null)); + mock.Setup(m => m.TryParseArtifactReference(ArtifactType.Module, null, "validRef2")).Returns(ResultHelper.Create(validRef2, @null)); var validRefUri3 = RandomFileUri(); ArtifactReference? validRef3 = new MockModuleReference("validRef3", validRefUri3); - mock.Setup(m => m.TryParseArtifactReference(null, "validRef3")).Returns(ResultHelper.Create(validRef3, @null)); + mock.Setup(m => m.TryParseArtifactReference(ArtifactType.Module, null, "validRef3")).Returns(ResultHelper.Create(validRef3, @null)); var badRefUri = RandomFileUri(); ArtifactReference? nullRef = null; ErrorBuilderDelegate? badRefError = x => new ErrorDiagnostic(x.TextSpan, "BCPMock", "Bad ref error"); - mock.Setup(m => m.TryParseArtifactReference(null, "badRef")).Returns(ResultHelper.Create(nullRef, badRefError)); + mock.Setup(m => m.TryParseArtifactReference(ArtifactType.Module, null, "badRef")).Returns(ResultHelper.Create(nullRef, badRefError)); mock.Setup(m => m.IsArtifactRestoreRequired(validRef)).Returns(true); mock.Setup(m => m.IsArtifactRestoreRequired(validRef2)).Returns(false); @@ -126,11 +126,11 @@ public async Task MockRegistries_ModuleLifecycle() var goodModule3 = CreateModule("mock:validRef3"); var badModule = CreateModule("mock:badRef"); - dispatcher.TryGetModuleReference(goodModule, validRefUri).IsSuccess(out var @ref, out var goodValidationBuilder).Should().BeTrue(); + dispatcher.TryGetArtifactReference(goodModule, validRefUri).IsSuccess(out var @ref, out var goodValidationBuilder).Should().BeTrue(); @ref.Should().Be(validRef); goodValidationBuilder!.Should().BeNull(); - dispatcher.TryGetModuleReference(badModule, badRefUri).IsSuccess(out @ref, out var badValidationBuilder).Should().BeFalse(); + dispatcher.TryGetArtifactReference(badModule, badRefUri).IsSuccess(out @ref, out var badValidationBuilder).Should().BeFalse(); @ref.Should().BeNull(); badValidationBuilder!.Should().NotBeNull(); badValidationBuilder!.Should().HaveCode("BCPMock"); @@ -147,11 +147,11 @@ public async Task MockRegistries_ModuleLifecycle() goodAvailabilityBuilder3!.Should().HaveCode("BCP190"); goodAvailabilityBuilder3!.Should().HaveMessage("The module with reference \"mock:validRef3\" has not been restored."); - dispatcher.TryGetLocalModuleEntryPointUri(validRef).IsSuccess(out var @uri, out var entryPointBuilder).Should().BeTrue(); + dispatcher.TryGetLocalArtifactEntryPointUri(validRef).IsSuccess(out var @uri, out var entryPointBuilder).Should().BeTrue(); @uri.Should().Be(new Uri("untitled://validRef")); entryPointBuilder!.Should().BeNull(); - dispatcher.TryGetLocalModuleEntryPointUri(validRef3).IsSuccess(out @uri, out var entryPointBuilder3).Should().BeTrue(); + dispatcher.TryGetLocalArtifactEntryPointUri(validRef3).IsSuccess(out @uri, out var entryPointBuilder3).Should().BeTrue(); @uri.Should().Be(new Uri("untitled://validRef3")); entryPointBuilder3!.Should().BeNull(); diff --git a/src/Bicep.Core.UnitTests/Registry/OciModuleRegistryTests.cs b/src/Bicep.Core.UnitTests/Registry/OciModuleRegistryTests.cs index 24d90ba7f87..c82d7ec9bcd 100644 --- a/src/Bicep.Core.UnitTests/Registry/OciModuleRegistryTests.cs +++ b/src/Bicep.Core.UnitTests/Registry/OciModuleRegistryTests.cs @@ -9,6 +9,7 @@ using Bicep.Core.Features; using Bicep.Core.Modules; using Bicep.Core.Registry; +using Bicep.Core.Registry.Oci; using Bicep.Core.SourceCode; using Bicep.Core.UnitTests.Assertions; using Bicep.Core.UnitTests.Mock; @@ -44,14 +45,14 @@ public void TestInitialize() [DataTestMethod] public void GetDocumentationUri_WithInvalidManifestContents_ShouldReturnNull(string manifestFileContents) { - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().BeNull(); } @@ -59,14 +60,14 @@ public void GetDocumentationUri_WithInvalidManifestContents_ShouldReturnNull(str [TestMethod] public void GetDocumentationUri_WithNonExistentManifestFile_ShouldReturnNull() { - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", null, "test.azurecr.io", "bicep/modules/storage", digest: "sha:12345"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().BeNull(); } @@ -92,14 +93,14 @@ public void GetDocumentationUri_WithManifestFileAndNoAnnotations_ShouldReturnNul } ] }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().BeNull(); } @@ -130,14 +131,14 @@ public void GetDocumentationUri_WithAnnotationsInManifestFileAndInvalidDocumenta ""org.opencontainers.image.documentation"": """ + documentationUri + @""" } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().BeNull(); } @@ -165,17 +166,17 @@ public async Task GetDocumentationUri_WithAnnotationsInManifestFile_ButEmpty_Sho ""annotations"": { } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var documentation = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var documentation = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); documentation.Should().BeNull(); - var description = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var description = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); description.Should().BeNull(); } @@ -204,17 +205,17 @@ public async Task GetDocumentationUri_WithAnnotationsInManifestFile_ButOnlyHasOt ""org.opencontainers.image.notdescription"": """ + "description" + @""" } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var documentation = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var documentation = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); documentation.Should().BeNull(); - var description = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var description = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); description.Should().BeNull(); } @@ -243,14 +244,14 @@ public void GetDocumentationUri_WithValidDocumentationUriInManifestFile_ShouldRe ""org.opencontainers.image.documentation"": """ + documentationUri + @""" } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().NotBeNull(); result.Should().BeEquivalentTo(documentationUri); @@ -289,14 +290,14 @@ public void GetDocumentationUri_WithMcrModuleReferenceAndNoDocumentationUriInMan ] } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( bicepFileContents, manifestFileContents, "mcr.microsoft.com", "bicep/app/dapr-containerapps-environment/bicep/core", tag: "1.0.1"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().NotBeNull(); result.Should().BeEquivalentTo("https://github.com/Azure/bicep-registry-modules/tree/app/dapr-containerapps-environment/bicep/core/1.0.1/modules/app/dapr-containerapps-environment/bicep/core/README.md"); @@ -312,14 +313,14 @@ public void GetDocumentationUri_WithMcrModuleReferenceAndNoDocumentationUriInMan [DataTestMethod] public void GetDescription_WithInvalidManifestContents_ShouldReturnNull(string manifestFileContents) { - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var result = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); result.Should().BeNull(); } @@ -327,14 +328,14 @@ public void GetDescription_WithInvalidManifestContents_ShouldReturnNull(string m [TestMethod] public async Task GetDescription_WithNonExistentManifestFile_ShouldReturnNull() { - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", null, "test.azurecr.io", "bicep/modules/storage", digest: "sha:12345"); - var result = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var result = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); result.Should().BeNull(); } @@ -360,14 +361,14 @@ public async Task GetDescription_WithManifestFileAndNoAnnotations_ShouldReturnNu } ] }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var result = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); result.Should().BeNull(); } @@ -395,14 +396,14 @@ public async Task GetDescription_WithManifestFileAndJustDocumentationUri_ShouldR } ] }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var result = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); result.Should().BeNull(); } @@ -432,14 +433,14 @@ public async Task GetDescription_WithValidDescriptionInManifestFile_ShouldReturn ""org.opencontainers.image.description"": """ + description + @""" } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var result = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); result.Should().NotBeNull(); result.Should().BeEquivalentTo(description); @@ -471,14 +472,14 @@ public async Task GetDescription_WithAnnotationsInManifestFileAndInvalidDescript ""org.opencontainers.image.description"": """ + description + @""" } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var result = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var result = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); result.Should().BeNull(); } @@ -510,19 +511,19 @@ public async Task GetDescription_WithValidDescriptionAndDocumentationUriInManife ""org.opencontainers.image.description"": """ + description + @""" } }"; - (OciArtifactRegistry OciArtifactRegistry, OciModuleReference OciModuleReference) = CreateModuleRegistryWithCachedModuleReference( + (OciArtifactRegistry OciArtifactRegistry, OciArtifactReference OciArtifactReference) = CreateModuleRegistryWithCachedModuleReference( "output myOutput string = 'hello!'", manifestFileContents, "test.azurecr.io", "bicep/modules/storage", "sha:12345"); - var actualDocumentationUri = OciArtifactRegistry.TryGetDocumentationUri(OciModuleReference); + var actualDocumentationUri = OciArtifactRegistry.TryGetDocumentationUri(OciArtifactReference); actualDocumentationUri.Should().NotBeNull(); actualDocumentationUri.Should().BeEquivalentTo(documentationUri); - var actualDescription = await OciArtifactRegistry.TryGetDescription(OciModuleReference); + var actualDescription = await OciArtifactRegistry.TryGetDescription(OciArtifactReference); actualDescription.Should().NotBeNull(); actualDescription.Should().BeEquivalentTo(description.Replace("\\", "")); // unencode json @@ -694,9 +695,9 @@ public async Task RestoreModuleWithSources_ShouldRestoreSourceToDisk(bool publis #region Helpers - private async Task RestoreModule(OciArtifactRegistry ociRegistry, OciModuleReference moduleReference) + private async Task RestoreModule(OciArtifactRegistry ociRegistry, OciArtifactReference reference) { - var (_, failureBuilder) = (await ociRegistry.RestoreArtifacts(new[] { moduleReference })).SingleOrDefault(); + var (_, failureBuilder) = (await ociRegistry.RestoreArtifacts(new[] { reference })).SingleOrDefault(); if (failureBuilder is { }) { var builder = new DiagnosticBuilderInternal(new Core.Parsing.TextSpan()); @@ -708,9 +709,9 @@ private async Task RestoreModule(OciArtifactRegistry ociRegistry, OciModuleRefer } } - private OciModuleReference CreateModuleReference(string registry, string repository, string? tag, string? digest) + private OciArtifactReference CreateModuleReference(string registry, string repository, string? tag, string? digest) { - OciModuleReference.TryParse(null, $"{registry}/{repository}:{tag}", BicepTestConstants.BuiltInConfiguration, new Uri("file:///main.bicep")).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParse(ArtifactType.Module, null, $"{registry}/{repository}:{tag}", BicepTestConstants.BuiltInConfiguration, new Uri("file:///main.bicep")).IsSuccess(out var moduleReference).Should().BeTrue(); return moduleReference!; } @@ -738,7 +739,7 @@ bool publishSourceFeatureEnabled // Creates a new (real) OciArtifactRegistry and sets it up so that it has an on-disk cached module // reference as if it had been restored from the registry. This can be used to test scenarios // where a module has already been restored. - private (OciArtifactRegistry, OciModuleReference) CreateModuleRegistryWithCachedModuleReference( + private (OciArtifactRegistry, OciArtifactReference) CreateModuleRegistryWithCachedModuleReference( string parentBicepFileContents, // The bicep file which references the module string? manifestFileContents, string registry, @@ -748,7 +749,7 @@ bool publishSourceFeatureEnabled { var (OciArtifactRegistry, _, parentModuleUri) = CreateModuleRegistryAndBicepFile(parentBicepFileContents, true); - OciModuleReference? ociModuleReference = OciArtifactRegistryHelper.CreateModuleReferenceMock( + OciArtifactReference? OciArtifactReference = OciArtifactRegistryHelper.CreateModuleReferenceMock( registry, repository, parentModuleUri, @@ -767,7 +768,7 @@ bool publishSourceFeatureEnabled tag); } - return (OciArtifactRegistry, ociModuleReference); + return (OciArtifactRegistry, OciArtifactReference); } private IFeatureProvider GetFeatures(string cacheRootDirectory, bool publishSourceEnabled) diff --git a/src/Bicep.Core.UnitTests/Syntax/ImportSpecificationTest.cs b/src/Bicep.Core.UnitTests/Syntax/ImportSpecificationTest.cs new file mode 100644 index 00000000000..fad81fd1d8c --- /dev/null +++ b/src/Bicep.Core.UnitTests/Syntax/ImportSpecificationTest.cs @@ -0,0 +1,41 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using Bicep.Core; +using Bicep.Core.Syntax; +using FluentAssertions; +using Microsoft.VisualStudio.TestTools.UnitTesting; + +[TestClass] +public class ImportSpecificationTests +{ + [DataTestMethod] + [DataRow("br/public:az@1.0.0", true, "az")] + [DataRow("br:mcr.microsoft.com/bicep/providers/az@1.0.0", true, "az")] + [DataRow("kubernetes@1.0.0", true, "kubernetes")] + [DataRow("sys@1.0.0", true, "sys")] + [DataRow("br:mcr.microsoft.com/bicep/providers/az@1.0.0-beta", true, "az")] // valid prerelease version + [DataRow("br:mcr.microsoft.com/bicep/providers/az@1.0.0+build.123", true, "az")] // valid build metadata + [DataRow("br:mcr.microsoft.com/bicep/providers/az@1.0.0-beta+build.123", true, "az")] // valid prerelease version and build metadata + // Negative cases + [DataRow("az@1.0.0", false)] // 'az@1.0.0' is not a valid provider declaration statement + [DataRow("br/public:az", false)] // is not a valid provider declaration statement, it lacks a version + [DataRow("br:mcr.microsoft.com/bicep/providers/az@", false)] // is not a valid provider declaration statement, partially constructed + [DataRow("br:mcr.microsoft.com/bicep/providers/az@1.0.0-beta+build.123+extra", false)] // invalid syntax, multiple build metadata sections + [DataRow("br:mcr.microsoft.com/bicep/providers/az@1.0.0-beta+build.123+extra.456", false)] // invalid syntax, build metadata section contains a dot + [DataRow("br:mcr.microsoft.com/bicep/providers/kubernetes@1.0.0", false)] // valid descriptor, but not az provider + [DataRow("br/public:kubernetes@1.0.0", false)] // valid descriptor, but not az provider + public void TestCreateFromStringSyntax(string input, bool isValidDeclaration, string expectedName = LanguageConstants.ErrorName) + { + // Arrange + var syntax = SyntaxFactory.CreateStringLiteral(input); + // Act + var got = ImportSpecification.From(syntax); + // Assert + if (isValidDeclaration) + { + got.Name.Should().Be(expectedName); + } + got.IsValid.Should().Be(isValidDeclaration); + } +} \ No newline at end of file diff --git a/src/Bicep.Core.UnitTests/Utils/OciModuleRegistryHelper.cs b/src/Bicep.Core.UnitTests/Utils/OciModuleRegistryHelper.cs index 0c9a12f3f0e..6de00f40bae 100644 --- a/src/Bicep.Core.UnitTests/Utils/OciModuleRegistryHelper.cs +++ b/src/Bicep.Core.UnitTests/Utils/OciModuleRegistryHelper.cs @@ -24,26 +24,13 @@ namespace Bicep.Core.UnitTests.Utils { public static class OciArtifactRegistryHelper { - public static OciModuleReference CreateModuleReferenceMock( - string registry, - string repository, - Uri parentModuleUri, - string? digest, - string? tag) - { - var artifactReferenceMock = StrictMock.Of(); - artifactReferenceMock.SetupGet(m => m.Registry).Returns(registry); - artifactReferenceMock.SetupGet(m => m.Repository).Returns(repository); - artifactReferenceMock.SetupGet(m => m.Digest).Returns(digest); - artifactReferenceMock.SetupGet(m => m.Tag).Returns(tag); - artifactReferenceMock.SetupGet(m => m.ArtifactId).Returns($"{registry}/{repository}:{tag ?? digest}"); - - return new OciModuleReference(artifactReferenceMock.Object, parentModuleUri); - } + public static OciArtifactReference CreateModuleReferenceMock(string registry, string repository, Uri parentModuleUri, string? digest, string? tag) + => new(ArtifactType.Module, registry, repository, tag, digest, parentModuleUri); + - public static OciModuleReference CreateModuleReference(string registry, string repository, string? tag, string? digest) + public static OciArtifactReference CreateModuleReference(string registry, string repository, string? tag, string? digest) { - OciModuleReference.TryParse(null, $"{registry}/{repository}:{tag}", BicepTestConstants.BuiltInConfiguration, new Uri("file:///main.bicep")).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParse(ArtifactType.Module, null, $"{registry}/{repository}:{tag}", BicepTestConstants.BuiltInConfiguration, new Uri("file:///main.bicep")).IsSuccess(out var moduleReference).Should().BeTrue(); return moduleReference!; } diff --git a/src/Bicep.Core/Configuration/AnalyzersConfigurationExtensions.cs b/src/Bicep.Core/Configuration/AnalyzersConfigurationExtensions.cs index 147b91068dd..9c9b5089896 100644 --- a/src/Bicep.Core/Configuration/AnalyzersConfigurationExtensions.cs +++ b/src/Bicep.Core/Configuration/AnalyzersConfigurationExtensions.cs @@ -26,6 +26,7 @@ public static RootConfiguration WithAllAnalyzersDisabled(this RootConfiguration return new RootConfiguration( analyzersConfiguration.Cloud, analyzersConfiguration.ModuleAliases, + analyzersConfiguration.ProviderAliases, analyzersConfiguration.Analyzers.WithAllAnalyzersDisabled(), analyzersConfiguration.CacheRootDirectory, analyzersConfiguration.ExperimentalFeaturesEnabled, @@ -39,6 +40,7 @@ public static RootConfiguration WithAnalyzersDisabled(this RootConfiguration ana return new RootConfiguration( analyzersConfiguration.Cloud, analyzersConfiguration.ModuleAliases, + analyzersConfiguration.ProviderAliases, analyzersConfiguration.Analyzers.WithAnalyzersDisabled(analyzerCodesToDisable), analyzersConfiguration.CacheRootDirectory, analyzersConfiguration.ExperimentalFeaturesEnabled, diff --git a/src/Bicep.Core/Configuration/ConfigurationManager.cs b/src/Bicep.Core/Configuration/ConfigurationManager.cs index 09f8dac5ce5..f2ca5a9ad64 100644 --- a/src/Bicep.Core/Configuration/ConfigurationManager.cs +++ b/src/Bicep.Core/Configuration/ConfigurationManager.cs @@ -99,6 +99,7 @@ private static RootConfiguration WithLoadDiagnostics(RootConfiguration configura return new( configuration.Cloud, configuration.ModuleAliases, + configuration.ProviderAliases, configuration.Analyzers, configuration.CacheRootDirectory, configuration.ExperimentalFeaturesEnabled, diff --git a/src/Bicep.Core/Configuration/ProviderAliasesConfiguration.cs b/src/Bicep.Core/Configuration/ProviderAliasesConfiguration.cs new file mode 100644 index 00000000000..5e1a75d0e83 --- /dev/null +++ b/src/Bicep.Core/Configuration/ProviderAliasesConfiguration.cs @@ -0,0 +1,88 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Collections.Immutable; +using System.Diagnostics.CodeAnalysis; +using System.Linq; +using System.Text; +using System.Text.Json; +using System.Text.Json.Serialization; +using System.Text.RegularExpressions; +using System.Threading.Tasks; +using Bicep.Core.Diagnostics; +using Bicep.Core.Extensions; +using static Bicep.Core.Diagnostics.DiagnosticBuilder; + +namespace Bicep.Core.Configuration +{ + public record ProviderAliases + { + [JsonPropertyName("br")] + public ImmutableSortedDictionary OciArtifactProviderAliases { get; init; } = ImmutableSortedDictionary.Empty; + } + + public record OciArtifactProviderAlias + { + public string? Registry { get; init; } + public string? ProviderPath { get; init; } + public override string ToString() => this.ProviderPath is not null + ? $"{Registry}/{ProviderPath}" + : $"{Registry}"; + } + + public partial class ProviderAliasesConfiguration : ConfigurationSection + { + private readonly string? configurationPath; + + private ProviderAliasesConfiguration(ProviderAliases data, string? configurationPath) + : base(data) + { + this.configurationPath = configurationPath; + } + public static ProviderAliasesConfiguration Bind(JsonElement element, string? configurationPath) => new(element.ToNonNullObject(), configurationPath); + + public ImmutableSortedDictionary GetOciArtifactProviderAliases() + { + return this.Data.OciArtifactProviderAliases; + } + + public ResultWithDiagnostic TryGetOciArtifactProviderAlias(string aliasName) + { + if (!ValidateAliasName(aliasName, out var errorBuilder)) + { + return new(errorBuilder); + } + + if (!this.Data.OciArtifactProviderAliases.TryGetValue(aliasName, out var alias)) + { + return new(x => x.OciArtifactProviderAliasNameDoesNotExistInConfiguration(aliasName, this.configurationPath)); + } + + if (alias.Registry is null) + { + return new(x => x.InvalidOciArtifactProviderAliasRegistryNullOrUndefined(aliasName, this.configurationPath)); + } + + return new(alias); + } + + private static bool ValidateAliasName(string aliasName, [NotNullWhen(false)] out ErrorBuilderDelegate? errorBuilder) + { + if (!ProviderAliasNameRegex().IsMatch(aliasName)) + { + errorBuilder = x => x.InvalidProviderAliasName(aliasName); + return false; + } + + errorBuilder = null; + return true; + } + + [GeneratedRegex("^[a-zA-Z0-9-_]+$", RegexOptions.CultureInvariant)] + private static partial Regex ProviderAliasNameRegex(); + + + } +} diff --git a/src/Bicep.Core/Configuration/RootConfiguration.cs b/src/Bicep.Core/Configuration/RootConfiguration.cs index f0e22de5474..be56d098d4f 100644 --- a/src/Bicep.Core/Configuration/RootConfiguration.cs +++ b/src/Bicep.Core/Configuration/RootConfiguration.cs @@ -17,6 +17,8 @@ public class RootConfiguration private const string ModuleAliasesKey = "moduleAliases"; + private const string ProviderAliasesKey = "providerAliases"; + private const string AnalyzersKey = "analyzers"; private const string CacheRootDirectoryKey = "cacheRootDirectory"; @@ -28,6 +30,7 @@ public class RootConfiguration public RootConfiguration( CloudConfiguration cloud, ModuleAliasesConfiguration moduleAliases, + ProviderAliasesConfiguration providerAliases, AnalyzersConfiguration analyzers, string? cacheRootDirectory, ExperimentalFeaturesEnabled experimentalFeaturesEnabled, @@ -37,6 +40,7 @@ public RootConfiguration( { this.Cloud = cloud; this.ModuleAliases = moduleAliases; + this.ProviderAliases = providerAliases; this.Analyzers = analyzers; this.CacheRootDirectory = ExpandCacheRootDirectory(cacheRootDirectory); this.ExperimentalFeaturesEnabled = experimentalFeaturesEnabled; @@ -49,18 +53,21 @@ public static RootConfiguration Bind(JsonElement element, string? configurationP { var cloud = CloudConfiguration.Bind(element.GetProperty(CloudKey), configurationPath); var moduleAliases = ModuleAliasesConfiguration.Bind(element.GetProperty(ModuleAliasesKey), configurationPath); + var providerAliases = ProviderAliasesConfiguration.Bind(element.GetProperty(ProviderAliasesKey), configurationPath); var analyzers = new AnalyzersConfiguration(element.GetProperty(AnalyzersKey)); var cacheRootDirectory = element.TryGetProperty(CacheRootDirectoryKey, out var e) ? e.GetString() : default; var experimentalFeaturesEnabled = ExperimentalFeaturesEnabled.Bind(element.GetProperty(ExperimentalFeaturesEnabledKey)); var formatting = FormattingConfiguration.Bind(element.GetProperty(FormattingKey)); - return new(cloud, moduleAliases, analyzers, cacheRootDirectory, experimentalFeaturesEnabled, formatting, configurationPath, diagnosticBuilders); + return new(cloud, moduleAliases, providerAliases, analyzers, cacheRootDirectory, experimentalFeaturesEnabled, formatting, configurationPath, diagnosticBuilders); } public CloudConfiguration Cloud { get; } public ModuleAliasesConfiguration ModuleAliases { get; } + public ProviderAliasesConfiguration ProviderAliases { get; } + public AnalyzersConfiguration Analyzers { get; } public string? CacheRootDirectory { get; } @@ -88,6 +95,9 @@ public string ToUtf8Json() writer.WritePropertyName(ModuleAliasesKey); this.ModuleAliases.WriteTo(writer); + writer.WritePropertyName(ProviderAliasesKey); + this.ProviderAliases.WriteTo(writer); + writer.WritePropertyName(AnalyzersKey); this.Analyzers.WriteTo(writer); diff --git a/src/Bicep.Core/Configuration/bicepconfig.json b/src/Bicep.Core/Configuration/bicepconfig.json index afb71fb49de..c143ffb8922 100644 --- a/src/Bicep.Core/Configuration/bicepconfig.json +++ b/src/Bicep.Core/Configuration/bicepconfig.json @@ -32,6 +32,14 @@ } } }, + "providerAliases": { + "br": { + "public": { + "registry": "mcr.microsoft.com", + "providerPath": "bicep/providers" + } + } + }, "analyzers": { "core": { "verbose": false, diff --git a/src/Bicep.Core/Diagnostics/DiagnosticBuilder.cs b/src/Bicep.Core/Diagnostics/DiagnosticBuilder.cs index 9173270ba86..3e6670e3e4d 100644 --- a/src/Bicep.Core/Diagnostics/DiagnosticBuilder.cs +++ b/src/Bicep.Core/Diagnostics/DiagnosticBuilder.cs @@ -12,6 +12,7 @@ using Bicep.Core.Modules; using Bicep.Core.Navigation; using Bicep.Core.Parsing; +using Bicep.Core.Registry; using Bicep.Core.Resources; using Bicep.Core.Semantics; using Bicep.Core.Semantics.Metadata; @@ -991,7 +992,7 @@ public ErrorDiagnostic DirectAccessToCollectionNotSupported(IEnumerable? "BCP162", "Expected a loop item variable identifier or \"(\" at this location."); - public ErrorDiagnostic ScopeUnsupportedOnChildResource(string parentIdentifier) => new( + public ErrorDiagnostic ScopeUnsupportedOnChildResource() => new( TextSpan, "BCP164", $"A child resource's scope is computed based on the scope of its ancestor resource. This means that using the \"{LanguageConstants.ResourceScopePropertyName}\" property on a child resource is unsupported."); @@ -1761,7 +1762,7 @@ public ErrorDiagnostic IndexOutOfBounds(string typeName, long tupleLength, long var message = new StringBuilder("The provided index value of \"").Append(indexSought).Append("\" is not valid for type \"").Append(typeName).Append("\"."); if (tupleLength > 0) { - message.Append(" Indexes for this type must be between 0 and ").Append(tupleLength - 1).Append("."); + message.Append(" Indexes for this type must be between 0 and ").Append(tupleLength - 1).Append('.'); } return new(TextSpan, "BCP311", message.ToString()); @@ -2088,6 +2089,28 @@ public ErrorDiagnostic RuntimeValueNotAllowedInFunctionDeclaration(string? acces TextSpan, "BCP376", $"The \"{name}\" symbol cannot be imported because imports of kind {exportMetadataKind} are not supported in files of kind {sourceFileKind}."); + + public ErrorDiagnostic InvalidProviderAliasName(string aliasName) => new( + TextSpan, + "BCP377", + $"The provider alias name \"{aliasName}\" is invalid. Valid characters are alphanumeric, \"_\", or \"-\"."); + + public ErrorDiagnostic InvalidOciArtifactProviderAliasRegistryNullOrUndefined(string aliasName, string? configurationPath) => new( + TextSpan, + "BCP378", + $"The OCI artifact provider alias \"{aliasName}\" in the {BuildBicepConfigurationClause(configurationPath)} is invalid. The \"registry\" property cannot be null or undefined."); + + public ErrorDiagnostic OciArtifactProviderAliasNameDoesNotExistInConfiguration(string aliasName, string? configurationPath) => new( + TextSpan, + "BCP379", + $"The OCI artifact provider alias name \"{aliasName}\" does not exist in the {BuildBicepConfigurationClause(configurationPath)}."); + + public ErrorDiagnostic UnsupportedArtifactType(ArtifactType artifactType) => new( + TextSpan, + "BCP380", + $"Artifacts of type: \"{artifactType}\" are not supported." + ); + } public static DiagnosticBuilderInternal ForPosition(TextSpan span) diff --git a/src/Bicep.Core/Emit/CompileTimeImports/ImportClosureInfo.cs b/src/Bicep.Core/Emit/CompileTimeImports/ImportClosureInfo.cs index 94782d9d5ae..4a142024f82 100644 --- a/src/Bicep.Core/Emit/CompileTimeImports/ImportClosureInfo.cs +++ b/src/Bicep.Core/Emit/CompileTimeImports/ImportClosureInfo.cs @@ -200,7 +200,7 @@ TemplateSpecSemanticModel targetTemplateSpecModel private static ArtifactReference GetImportReference(ImportedSymbol symbol) { - if (symbol.TryGetModuleReference().IsSuccess(out var moduleReference)) + if (symbol.TryGetArtifactReference().IsSuccess(out var moduleReference)) { return moduleReference; } @@ -210,7 +210,7 @@ private static ArtifactReference GetImportReference(ImportedSymbol symbol) private static ArtifactReference GetImportReference(WildcardImportSymbol symbol) { - if (symbol.TryGetModuleReference().IsSuccess(out var moduleReference)) + if (symbol.TryGetArtifactReference().IsSuccess(out var moduleReference)) { return moduleReference; } diff --git a/src/Bicep.Core/Emit/ScopeHelper.cs b/src/Bicep.Core/Emit/ScopeHelper.cs index c27d0635b99..3418ff0120f 100644 --- a/src/Bicep.Core/Emit/ScopeHelper.cs +++ b/src/Bicep.Core/Emit/ScopeHelper.cs @@ -418,7 +418,7 @@ void logInvalidScopeDiagnostic(IPositionable positionable, ResourceScope supplie if (resource.TryGetScopeSyntax() is { } scopeSyntax) { // it doesn't make sense to have scope on a descendent resource; it should be inherited from the oldest ancestor. - diagnosticWriter.Write(scopeSyntax, x => x.ScopeUnsupportedOnChildResource(ancestors.Last().Resource.Symbol.Name)); + diagnosticWriter.Write(scopeSyntax, x => x.ScopeUnsupportedOnChildResource()); // TODO: format the ancestor name using the resource accessor (::) for nested resources scopeInfo[resource] = defaultScopeData; continue; diff --git a/src/Bicep.Core/Modules/OciModuleReference.cs b/src/Bicep.Core/Modules/OciModuleReference.cs deleted file mode 100644 index fb213843803..00000000000 --- a/src/Bicep.Core/Modules/OciModuleReference.cs +++ /dev/null @@ -1,45 +0,0 @@ -// Copyright (c) Microsoft Corporation. -// Licensed under the MIT License. - -using System; -using System.Diagnostics.CodeAnalysis; -using Bicep.Core.Configuration; -using Bicep.Core.Diagnostics; -using Bicep.Core.Registry; -using Bicep.Core.Registry.Oci; - - -namespace Bicep.Core.Modules -{ - /// - /// Represents a reference to an artifact in an OCI registry. - /// - public class OciModuleReference : ArtifactReference, IOciArtifactReference - { - private readonly IOciArtifactReference ociArtifactRef; - public OciModuleReference(IOciArtifactReference ociArtifactReference, Uri parentModuleUri) - : base(ModuleReferenceSchemes.Oci, parentModuleUri) - { - this.ociArtifactRef = ociArtifactReference; - } - public override string UnqualifiedReference => this.ArtifactId; - public override bool IsExternal => true; - public override bool Equals(object? obj) => obj is OciModuleReference other && this.ociArtifactRef.Equals(other.ociArtifactRef); - public override int GetHashCode() => this.ociArtifactRef.GetHashCode(); - public string Registry => this.ociArtifactRef.Registry; - public string Repository => this.ociArtifactRef.Repository; - public string? Tag => this.ociArtifactRef.Tag; - public string? Digest => this.ociArtifactRef.Digest; - public string ArtifactId => this.ociArtifactRef.ArtifactId; - - public static ResultWithDiagnostic TryParse(string? aliasName, string rawValue, RootConfiguration configuration, Uri parentModuleUri) - { - if (!OciArtifactReference.TryParse(aliasName, rawValue, configuration).IsSuccess(out var artifactReference, out var failureBuilder)) - { - return new(failureBuilder); - } - - return new(new OciModuleReference(artifactReference, parentModuleUri)); - } - } -} diff --git a/src/Bicep.Core/Navigation/IArtifactReferenceSyntax.cs b/src/Bicep.Core/Navigation/IArtifactReferenceSyntax.cs index cca885d9379..b16221efc5e 100644 --- a/src/Bicep.Core/Navigation/IArtifactReferenceSyntax.cs +++ b/src/Bicep.Core/Navigation/IArtifactReferenceSyntax.cs @@ -1,5 +1,6 @@ // Copyright (c) Microsoft Corporation. // Licensed under the MIT License. +using Bicep.Core.Registry; using Bicep.Core.Syntax; namespace Bicep.Core.Navigation; @@ -13,4 +14,6 @@ public interface IArtifactReferenceSyntax SyntaxBase SourceSyntax { get; } SyntaxBase? Path { get; } + + ArtifactType GetArtifactType(); } diff --git a/src/Bicep.Core/Registry/ArtifactRegistry.cs b/src/Bicep.Core/Registry/ArtifactRegistry.cs index 2c430f155f2..a63cb0b1476 100644 --- a/src/Bicep.Core/Registry/ArtifactRegistry.cs +++ b/src/Bicep.Core/Registry/ArtifactRegistry.cs @@ -29,7 +29,7 @@ public abstract class ArtifactRegistry : IArtifactRegistry where T : Artifact public abstract ResultWithDiagnostic TryGetLocalArtifactEntryPointUri(T reference); - public abstract ResultWithDiagnostic TryParseArtifactReference(string? aliasName, string reference); + public abstract ResultWithDiagnostic TryParseArtifactReference(ArtifactType artifactType, string? aliasName, string reference); public abstract string? TryGetDocumentationUri(T reference); diff --git a/src/Bicep.Core/Registry/ArtifactType.cs b/src/Bicep.Core/Registry/ArtifactType.cs new file mode 100644 index 00000000000..83af73d3295 --- /dev/null +++ b/src/Bicep.Core/Registry/ArtifactType.cs @@ -0,0 +1,11 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +namespace Bicep.Core.Registry +{ + public enum ArtifactType + { + Module, + Provider, + } +} diff --git a/src/Bicep.Core/Registry/IArtifactDispatcher.cs b/src/Bicep.Core/Registry/IArtifactDispatcher.cs index 306f752cb81..f27c69e992d 100644 --- a/src/Bicep.Core/Registry/IArtifactDispatcher.cs +++ b/src/Bicep.Core/Registry/IArtifactDispatcher.cs @@ -11,23 +11,23 @@ namespace Bicep.Core.Registry { - public interface IModuleDispatcher : IModuleReferenceFactory + public interface IModuleDispatcher : IArtifactReferenceFactory { - RegistryCapabilities GetRegistryCapabilities(ArtifactReference moduleReference); + RegistryCapabilities GetRegistryCapabilities(ArtifactReference reference); - ArtifactRestoreStatus GetArtifactRestoreStatus(ArtifactReference moduleReference, out DiagnosticBuilder.ErrorBuilderDelegate? errorDetailBuilder); + ArtifactRestoreStatus GetArtifactRestoreStatus(ArtifactReference reference, out DiagnosticBuilder.ErrorBuilderDelegate? errorDetailBuilder); - ResultWithDiagnostic TryGetLocalModuleEntryPointUri(ArtifactReference moduleReference); + ResultWithDiagnostic TryGetLocalArtifactEntryPointUri(ArtifactReference reference); - Task RestoreModules(IEnumerable moduleReferences, bool forceModulesRestore = false); + Task RestoreModules(IEnumerable references, bool forceRestore = false); - Task CheckModuleExists(ArtifactReference moduleReference); + Task CheckModuleExists(ArtifactReference reference); - Task PublishModule(ArtifactReference moduleReference, Stream compiledArmTemplate, Stream? bicepSources, string? documentationUri); + Task PublishModule(ArtifactReference reference, Stream compiledArmTemplate, Stream? bicepSources, string? documentationUri); void PruneRestoreStatuses(); // Retrieves the sources that have been restored along with the module into the cache (if available) - SourceArchive? TryGetModuleSources(ArtifactReference moduleReference); + SourceArchive? TryGetModuleSources(ArtifactReference reference); } } diff --git a/src/Bicep.Core/Registry/IModuleReferenceFactory.cs b/src/Bicep.Core/Registry/IArtifactReferenceFactory.cs similarity index 51% rename from src/Bicep.Core/Registry/IModuleReferenceFactory.cs rename to src/Bicep.Core/Registry/IArtifactReferenceFactory.cs index ca4e8f631fc..7dcb79f65ec 100644 --- a/src/Bicep.Core/Registry/IModuleReferenceFactory.cs +++ b/src/Bicep.Core/Registry/IArtifactReferenceFactory.cs @@ -9,11 +9,11 @@ namespace Bicep.Core.Registry; -public interface IModuleReferenceFactory +public interface IArtifactReferenceFactory { ImmutableArray AvailableSchemes(Uri parentModuleUri); - ResultWithDiagnostic TryGetModuleReference(string reference, Uri parentModuleUri); + ResultWithDiagnostic TryGetArtifactReference(ArtifactType artifactType, string reference, Uri parentModuleUri); - ResultWithDiagnostic TryGetModuleReference(IArtifactReferenceSyntax artifactDeclaration, Uri parentModuleUri); + ResultWithDiagnostic TryGetArtifactReference(IArtifactReferenceSyntax artifactDeclaration, Uri parentModuleUri); } diff --git a/src/Bicep.Core/Registry/IArtifactRegistry.cs b/src/Bicep.Core/Registry/IArtifactRegistry.cs index fd0c7a05b85..900e0d9fa7d 100644 --- a/src/Bicep.Core/Registry/IArtifactRegistry.cs +++ b/src/Bicep.Core/Registry/IArtifactRegistry.cs @@ -33,7 +33,8 @@ public interface IArtifactRegistry /// /// The alias name /// The unqualified artifact reference - ResultWithDiagnostic TryParseArtifactReference(string? aliasName, string reference); + /// The artifact type. Either "module" or "provider" + ResultWithDiagnostic TryParseArtifactReference(ArtifactType artifactType, string? aliasName, string reference); /// /// Returns true if the specified artifact is already cached in the local cache. @@ -71,28 +72,28 @@ public interface IArtifactRegistry /// /// Publishes the module at the specified path to the registry. /// - /// The module reference + /// The module reference /// The compiled module /// The source archive (binary stream of SourceArchive) - Task PublishArtifact(ArtifactReference moduleReference, Stream compiled, Stream? bicepSources, string? documentationUri, string? description); + Task PublishArtifact(ArtifactReference reference, Stream compiled, Stream? bicepSources, string? documentationUri, string? description); /// /// Returns documentationUri for the module. /// - /// The module reference - string? GetDocumentationUri(ArtifactReference moduleReference); + /// The module reference + string? GetDocumentationUri(ArtifactReference reference); /// /// Returns description for the module. /// - /// The module reference - Task TryGetDescription(ArtifactReference moduleReference); + /// The module reference + Task TryGetDescription(ArtifactReference reference); /// /// Returns the source code for the module, if available. /// - /// The module reference + /// The module reference /// A source archive - SourceArchive? TryGetSource(ArtifactReference moduleReference); + SourceArchive? TryGetSource(ArtifactReference reference); } } diff --git a/src/Bicep.Core/Registry/LocalModuleRegistry.cs b/src/Bicep.Core/Registry/LocalModuleRegistry.cs index 7d992a70915..547150f7b3b 100644 --- a/src/Bicep.Core/Registry/LocalModuleRegistry.cs +++ b/src/Bicep.Core/Registry/LocalModuleRegistry.cs @@ -31,8 +31,12 @@ public LocalModuleRegistry(IFileResolver fileResolver, Uri parentModuleUri, Bice public override RegistryCapabilities GetCapabilities(LocalModuleReference reference) => RegistryCapabilities.Default; - public override ResultWithDiagnostic TryParseArtifactReference(string? alias, string reference) + public override ResultWithDiagnostic TryParseArtifactReference(ArtifactType artifactType, string? alias, string reference) { + if (artifactType != ArtifactType.Module) + { + return new(x => x.UnsupportedArtifactType(artifactType)); + } if (!LocalModuleReference.TryParse(reference, parentModuleUri).IsSuccess(out var @ref, out var failureBuilder)) { return new(failureBuilder); diff --git a/src/Bicep.Core/Registry/ModuleDispatcher.cs b/src/Bicep.Core/Registry/ModuleDispatcher.cs index deeb5a08b59..a40dc8880cf 100644 --- a/src/Bicep.Core/Registry/ModuleDispatcher.cs +++ b/src/Bicep.Core/Registry/ModuleDispatcher.cs @@ -42,6 +42,10 @@ public ImmutableArray AvailableSchemes(Uri parentModuleUri) => Registries(parentModuleUri).Keys.OrderBy(s => s).ToImmutableArray(); public ResultWithDiagnostic TryGetModuleReference(string reference, Uri parentModuleUri) + => TryGetArtifactReference(ArtifactType.Module, reference, parentModuleUri); + + + public ResultWithDiagnostic TryGetArtifactReference(ArtifactType artifactType, string reference, Uri parentModuleUri) { var registries = Registries(parentModuleUri); var parts = reference.Split(':', 2, StringSplitOptions.None); @@ -51,7 +55,7 @@ public ResultWithDiagnostic TryGetModuleReference(string refe // local path reference if (registries.TryGetValue(ModuleReferenceSchemes.Local, out var localRegistry)) { - return localRegistry.TryParseArtifactReference(null, parts[0]); + return localRegistry.TryParseArtifactReference(artifactType, null, parts[0]); } return new(x => x.UnknownModuleReferenceScheme(ModuleReferenceSchemes.Local, this.AvailableSchemes(parentModuleUri))); @@ -73,7 +77,7 @@ public ResultWithDiagnostic TryGetModuleReference(string refe // the scheme is recognized var rawValue = parts[1]; - return registry.TryParseArtifactReference(aliasName, rawValue); + return registry.TryParseArtifactReference(artifactType, aliasName, rawValue); } // unknown scheme @@ -85,14 +89,14 @@ public ResultWithDiagnostic TryGetModuleReference(string refe } } - public ResultWithDiagnostic TryGetModuleReference(IArtifactReferenceSyntax moduleDeclaration, Uri parentModuleUri) + public ResultWithDiagnostic TryGetArtifactReference(IArtifactReferenceSyntax artifactDeclaration, Uri parentModuleUri) { - if (!SyntaxHelper.TryGetForeignTemplatePath(moduleDeclaration).IsSuccess(out var moduleReferenceString, out var failureBuilder)) + if (!SyntaxHelper.TryGetForeignTemplatePath(artifactDeclaration).IsSuccess(out var artifactReferenceString, out var failureBuilder)) { return new(failureBuilder); } - return this.TryGetModuleReference(moduleReferenceString, parentModuleUri); + return this.TryGetArtifactReference(artifactDeclaration.GetArtifactType(), artifactReferenceString, parentModuleUri); } public RegistryCapabilities GetRegistryCapabilities(ArtifactReference artifactReference) @@ -126,32 +130,32 @@ public ArtifactRestoreStatus GetArtifactRestoreStatus( return ArtifactRestoreStatus.Succeeded; } - public ResultWithDiagnostic TryGetLocalModuleEntryPointUri(ArtifactReference moduleReference) + public ResultWithDiagnostic TryGetLocalArtifactEntryPointUri(ArtifactReference artifactReference) { - var configuration = configurationManager.GetConfiguration(moduleReference.ParentModuleUri); - // has restore already failed for this module? - if (this.HasRestoreFailed(moduleReference, configuration, out var restoreFailureBuilder)) + var configuration = configurationManager.GetConfiguration(artifactReference.ParentModuleUri); + // has restore already failed for this artifact? + if (this.HasRestoreFailed(artifactReference, configuration, out var restoreFailureBuilder)) { return new(restoreFailureBuilder); } - var registry = this.GetRegistry(moduleReference); - return registry.TryGetLocalArtifactEntryPointUri(moduleReference); + var registry = this.GetRegistry(artifactReference); + return registry.TryGetLocalArtifactEntryPointUri(artifactReference); } - public async Task RestoreModules(IEnumerable moduleReferences, bool forceModulesRestore = false) + public async Task RestoreModules(IEnumerable references, bool forceRestore = false) { // WARNING: The various operations on ModuleReference objects here rely on the custom Equals() implementation and NOT on object identity - if (!forceModulesRestore && - moduleReferences.All(module => this.GetArtifactRestoreStatus(module, out _) == ArtifactRestoreStatus.Succeeded)) + if (!forceRestore && + references.All(module => this.GetArtifactRestoreStatus(module, out _) == ArtifactRestoreStatus.Succeeded)) { // all the modules have already been restored - no need to do anything return false; } // many module declarations can point to the same module - var uniqueReferences = moduleReferences.Distinct(); + var uniqueReferences = references.Distinct(); // split modules refs by registry var referencesByRegistry = uniqueReferences.ToLookup(@ref => Registries(@ref.ParentModuleUri)[@ref.Scheme]); @@ -160,7 +164,7 @@ public async Task RestoreModules(IEnumerable moduleRefe foreach (var registry in referencesByRegistry.Select(byRegistry => byRegistry.Key)) { // if we're asked to purge modules cache - if (forceModulesRestore) + if (forceRestore) { var forceModulesRestoreStatuses = await registry.InvalidateArtifactsCache(referencesByRegistry[registry]); @@ -183,18 +187,18 @@ public async Task RestoreModules(IEnumerable moduleRefe return true; } - public async Task PublishModule(ArtifactReference moduleReference, Stream compiledArmTemplate, Stream? bicepSources, string? documentationUri) + public async Task PublishModule(ArtifactReference reference, Stream compiledArmTemplate, Stream? bicepSources, string? documentationUri) { - var registry = this.GetRegistry(moduleReference); + var registry = this.GetRegistry(reference); var description = DescriptionHelper.TryGetFromArmTemplate(compiledArmTemplate); - await registry.PublishArtifact(moduleReference, compiledArmTemplate, bicepSources, documentationUri, description); + await registry.PublishArtifact(reference, compiledArmTemplate, bicepSources, documentationUri, description); } - public async Task CheckModuleExists(ArtifactReference moduleReference) + public async Task CheckModuleExists(ArtifactReference reference) { - var registry = this.GetRegistry(moduleReference); - return await registry.CheckArtifactExists(moduleReference); + var registry = this.GetRegistry(reference); + return await registry.CheckArtifactExists(reference); } public void PruneRestoreStatuses() @@ -212,18 +216,18 @@ public void PruneRestoreStatuses() } } - private IArtifactRegistry GetRegistry(ArtifactReference moduleReference) => - Registries(moduleReference.ParentModuleUri).TryGetValue(moduleReference.Scheme, out var registry) ? registry : throw new InvalidOperationException($"Unexpected moduleDeclaration reference scheme '{moduleReference.Scheme}'."); + private IArtifactRegistry GetRegistry(ArtifactReference reference) => + Registries(reference.ParentModuleUri).TryGetValue(reference.Scheme, out var registry) ? registry : throw new InvalidOperationException($"Unexpected artifactDeclaration reference scheme '{reference.Scheme}'."); - public SourceArchive? TryGetModuleSources(ArtifactReference moduleReference) + public SourceArchive? TryGetModuleSources(ArtifactReference reference) { - var registry = this.GetRegistry(moduleReference); - return registry.TryGetSource(moduleReference); + var registry = this.GetRegistry(reference); + return registry.TryGetSource(reference); } - private bool HasRestoreFailed(ArtifactReference moduleReference, RootConfiguration configuration, [NotNullWhen(true)] out DiagnosticBuilder.ErrorBuilderDelegate? failureBuilder) + private bool HasRestoreFailed(ArtifactReference reference, RootConfiguration configuration, [NotNullWhen(true)] out DiagnosticBuilder.ErrorBuilderDelegate? failureBuilder) { - if (this.restoreFailures.TryGetValue(new(configuration.Cloud, moduleReference), out var failureInfo) && !IsFailureInfoExpired(failureInfo, DateTime.UtcNow)) + if (this.restoreFailures.TryGetValue(new(configuration.Cloud, reference), out var failureInfo) && !IsFailureInfoExpired(failureInfo, DateTime.UtcNow)) { // the restore operation failed on the module previously // and the record of the failure has not yet expired @@ -237,7 +241,7 @@ private bool HasRestoreFailed(ArtifactReference moduleReference, RootConfigurati private static bool IsFailureInfoExpired(RestoreFailureInfo failureInfo, DateTime dateTime) => dateTime >= failureInfo.Expiration; - private void SetRestoreFailure(ArtifactReference moduleReference, RootConfiguration configuration, DiagnosticBuilder.ErrorBuilderDelegate failureBuilder) + private void SetRestoreFailure(ArtifactReference reference, RootConfiguration configuration, DiagnosticBuilder.ErrorBuilderDelegate failureBuilder) { // as the user is typing, the modules will keep getting recompiled // we can't keep retrying syntactically correct references to non-existent modules on every key press @@ -245,39 +249,39 @@ private void SetRestoreFailure(ArtifactReference moduleReference, RootConfigurat // we're not not doing sliding expiration because we want a retry to happen eventually // (we may consider adding an ability to immediately retry to the UX in the future as well) var expiration = DateTime.UtcNow.Add(FailureExpirationInterval); - this.restoreFailures.TryAdd(new(configuration.Cloud, moduleReference), new RestoreFailureInfo(moduleReference, failureBuilder, expiration)); + this.restoreFailures.TryAdd(new(configuration.Cloud, reference), new RestoreFailureInfo(reference, failureBuilder, expiration)); } private class RestoreFailureKey { private readonly CloudConfiguration configuration; - private readonly ArtifactReference moduleReference; + private readonly ArtifactReference reference; - public RestoreFailureKey(CloudConfiguration configuration, ArtifactReference moduleReference) + public RestoreFailureKey(CloudConfiguration configuration, ArtifactReference reference) { this.configuration = configuration; - this.moduleReference = moduleReference; + this.reference = reference; } public override bool Equals(object? obj) => obj is RestoreFailureKey other && this.configuration.Equals(other.configuration) && - this.moduleReference.Equals(other.moduleReference); + this.reference.Equals(other.reference); - public override int GetHashCode() => HashCode.Combine(this.configuration, this.moduleReference); + public override int GetHashCode() => HashCode.Combine(this.configuration, this.reference); } private class RestoreFailureInfo { - public RestoreFailureInfo(ArtifactReference moduleReference, DiagnosticBuilder.ErrorBuilderDelegate failureBuilder, DateTime expiration) + public RestoreFailureInfo(ArtifactReference reference, DiagnosticBuilder.ErrorBuilderDelegate failureBuilder, DateTime expiration) { - this.ModuleReference = moduleReference; + this.Reference = reference; this.FailureBuilder = failureBuilder; this.Expiration = expiration; } - public ArtifactReference ModuleReference { get; } + public ArtifactReference Reference { get; } public DiagnosticBuilder.ErrorBuilderDelegate FailureBuilder { get; } diff --git a/src/Bicep.Core/Registry/ModuleDispatcherExtensions.cs b/src/Bicep.Core/Registry/ModuleDispatcherExtensions.cs index f8d6a27b355..fb2c4a2f306 100644 --- a/src/Bicep.Core/Registry/ModuleDispatcherExtensions.cs +++ b/src/Bicep.Core/Registry/ModuleDispatcherExtensions.cs @@ -14,7 +14,7 @@ public static class ModuleDispatcherExtensions { public static IEnumerable GetValidModuleReferences(this IModuleDispatcher moduleDispatcher, IEnumerable artifacts) => artifacts - .Select(t => moduleDispatcher.TryGetModuleReference(t.DeclarationSyntax, t.ParentTemplateFile.FileUri).TryUnwrap()) + .Select(t => moduleDispatcher.TryGetArtifactReference(t.DeclarationSyntax, t.ParentTemplateFile.FileUri).TryUnwrap()) .WhereNotNull(); } } diff --git a/src/Bicep.Core/Registry/Oci/OciArtifactReference.cs b/src/Bicep.Core/Registry/Oci/OciArtifactReference.cs index ad940d7e0e6..9fb59a62480 100644 --- a/src/Bicep.Core/Registry/Oci/OciArtifactReference.cs +++ b/src/Bicep.Core/Registry/Oci/OciArtifactReference.cs @@ -12,10 +12,10 @@ namespace Bicep.Core.Registry.Oci { - - public class OciArtifactReference : IOciArtifactReference + public class OciArtifactReference : ArtifactReference, IOciArtifactReference { - private OciArtifactReference(string registry, string repository, string? tag, string? digest) + public OciArtifactReference(ArtifactType type, string registry, string repository, string? tag, string? digest, Uri parentModuleUri) : + base(OciArtifactReferenceFacts.Scheme, parentModuleUri) { switch (tag, digest) { @@ -29,7 +29,9 @@ private OciArtifactReference(string registry, string repository, string? tag, st this.Repository = repository; this.Tag = tag; this.Digest = digest; + this.Type = type; } + /// /// Gets the registry URI. /// @@ -57,9 +59,19 @@ private OciArtifactReference(string registry, string repository, string? tag, st ? $"{this.Registry}/{this.Repository}:{this.Tag}" : $"{this.Registry}/{this.Repository}@{this.Digest}"; - public string FullyQualifiedReference => ArtifactId; + /// + /// Gets the type of artifact reference. Either module or provider. + /// + public ArtifactType Type { get; } + + public override string UnqualifiedReference => ArtifactId; + + public override bool IsExternal => true; - public static ResultWithDiagnostic TryParse(string? aliasName, string rawValue, RootConfiguration configuration) + public static ResultWithDiagnostic TryParseModule(string? aliasName, string rawValue, RootConfiguration configuration, Uri parentModuleUri) + => TryParse(ArtifactType.Module, aliasName, rawValue, configuration, parentModuleUri); + + public static ResultWithDiagnostic TryParse(ArtifactType type, string? aliasName, string rawValue, RootConfiguration configuration, Uri parentModuleUri) { static string GetBadReference(string referenceValue) => $"{OciArtifactReferenceFacts.Scheme}:{referenceValue}"; @@ -67,12 +79,25 @@ public static ResultWithDiagnostic TryParse(string? aliasN if (aliasName is not null) { - if (!configuration.ModuleAliases.TryGetOciArtifactModuleAlias(aliasName).IsSuccess(out var alias, out var failureBuilder)) + switch (type) { - return new(failureBuilder); + case ArtifactType.Module: + if (!configuration.ModuleAliases.TryGetOciArtifactModuleAlias(aliasName).IsSuccess(out var moduleAlias, out var moduleFailureBuilder)) + { + return new(moduleFailureBuilder); + } + rawValue = $"{moduleAlias}/{rawValue}"; + break; + case ArtifactType.Provider: + if (!configuration.ProviderAliases.TryGetOciArtifactProviderAlias(aliasName).IsSuccess(out var providerAlias, out var providerFailureBuilder)) + { + return new(providerFailureBuilder); + } + rawValue = $"{providerAlias}/{rawValue}"; + break; + default: + return new(x => x.UnsupportedArtifactType(type)); } - - rawValue = $"{alias}/{rawValue}"; } // the set of valid OCI artifact refs is a subset of the set of valid URIs if you remove the scheme portion from each URI @@ -179,7 +204,7 @@ public static ResultWithDiagnostic TryParse(string? aliasN tag)); } - return new(new OciArtifactReference(registry, repository, tag: tag, digest: null)); + return new(new OciArtifactReference(type, registry, repository, tag, digest: null, parentModuleUri: parentModuleUri)); case '@': var digest = tagOrDigest; @@ -188,7 +213,7 @@ public static ResultWithDiagnostic TryParse(string? aliasN return new(x => x.InvalidOciArtifactReferenceInvalidDigest(aliasName, GetBadReference(rawValue), digest)); } - return new(new OciArtifactReference(registry, repository, tag: null, digest: digest)); + return new(new OciArtifactReference(type, registry, repository, tag: null, digest: digest, parentModuleUri: parentModuleUri)); default: throw new NotImplementedException($"Unexpected last segment delimiter character '{delimiter.Value}'."); diff --git a/src/Bicep.Core/Registry/OciArtifactRegistry.cs b/src/Bicep.Core/Registry/OciArtifactRegistry.cs index 45524104650..c2484657639 100644 --- a/src/Bicep.Core/Registry/OciArtifactRegistry.cs +++ b/src/Bicep.Core/Registry/OciArtifactRegistry.cs @@ -23,7 +23,7 @@ namespace Bicep.Core.Registry { - public sealed class OciArtifactRegistry : ExternalArtifactRegistry + public sealed class OciArtifactRegistry : ExternalArtifactRegistry { private readonly AzureContainerRegistryManager client; @@ -54,23 +54,23 @@ public OciArtifactRegistry( public string CacheRootDirectory => this.features.CacheRootDirectory; - public override RegistryCapabilities GetCapabilities(OciModuleReference reference) + public override RegistryCapabilities GetCapabilities(OciArtifactReference reference) { // cannot publish without tag return reference.Tag is null ? RegistryCapabilities.Default : RegistryCapabilities.Publish; } - public override ResultWithDiagnostic TryParseArtifactReference(string? aliasName, string reference) + public override ResultWithDiagnostic TryParseArtifactReference(ArtifactType artifactType, string? aliasName, string reference) { - if (!OciModuleReference.TryParse(aliasName, reference, configuration, parentModuleUri).IsSuccess(out var @ref, out var failureBuilder)) + if (!OciArtifactReference.TryParse(artifactType, aliasName, reference, configuration, parentModuleUri).IsSuccess(out var @ref, out var failureBuilder)) { return new(failureBuilder); } - return new(@ref); } - public override bool IsArtifactRestoreRequired(OciModuleReference reference) + + public override bool IsArtifactRestoreRequired(OciArtifactReference reference) { /* * this should be kept in sync with the WriteModuleContent() implementation @@ -93,7 +93,7 @@ public override bool IsArtifactRestoreRequired(OciModuleReference reference) !this.FileResolver.FileExists(this.GetModuleFileUri(reference, ModuleFileType.Metadata)); } - public override async Task CheckArtifactExists(OciModuleReference reference) + public override async Task CheckArtifactExists(OciArtifactReference reference) { try { @@ -130,13 +130,13 @@ public override async Task CheckArtifactExists(OciModuleReference referenc return true; } - public override ResultWithDiagnostic TryGetLocalArtifactEntryPointUri(OciModuleReference reference) + public override ResultWithDiagnostic TryGetLocalArtifactEntryPointUri(OciArtifactReference reference) { var localUri = this.GetModuleFileUri(reference, ModuleFileType.ModuleMain); return new(localUri); } - public override string? TryGetDocumentationUri(OciModuleReference ociArtifactModuleReference) + public override string? TryGetDocumentationUri(OciArtifactReference ociArtifactModuleReference) { var ociAnnotations = TryGetOciAnnotations(ociArtifactModuleReference); if (ociAnnotations is null || @@ -167,13 +167,13 @@ public static string GetPublicBicepModuleDocumentationUri(string publicModuleNam return $"https://github.com/Azure/bicep-registry-modules/tree/{publicModuleName}/{tag}/modules/{publicModuleName}/README.md"; } - public override Task TryGetDescription(OciModuleReference ociArtifactModuleReference) + public override Task TryGetDescription(OciArtifactReference ociArtifactModuleReference) { var ociAnnotations = TryGetOciAnnotations(ociArtifactModuleReference); return Task.FromResult(DescriptionHelper.TryGetFromOciManifestAnnotations(ociAnnotations)); } - private OciManifest GetCachedManifest(OciModuleReference ociArtifactModuleReference) + private OciManifest GetCachedManifest(OciArtifactReference ociArtifactModuleReference) { string manifestFilePath = this.GetModuleFilePath(ociArtifactModuleReference, ModuleFileType.Manifest); @@ -190,7 +190,7 @@ private OciManifest GetCachedManifest(OciModuleReference ociArtifactModuleRefere } } - private ImmutableDictionary? TryGetOciAnnotations(OciModuleReference ociArtifactModuleReference) + private ImmutableDictionary? TryGetOciAnnotations(OciArtifactReference ociArtifactModuleReference) { try { @@ -202,7 +202,7 @@ private OciManifest GetCachedManifest(OciModuleReference ociArtifactModuleRefere } } - public override async Task> RestoreArtifacts(IEnumerable references) + public override async Task> RestoreArtifacts(IEnumerable references) { var statuses = new Dictionary(); @@ -229,12 +229,12 @@ private OciManifest GetCachedManifest(OciModuleReference ociArtifactModuleRefere return statuses; } - public override async Task> InvalidateArtifactsCache(IEnumerable references) + public override async Task> InvalidateArtifactsCache(IEnumerable references) { return await base.InvalidateArtifactsCacheInternal(references); } - public override async Task PublishArtifact(OciModuleReference moduleReference, Stream compiledArmTemplate, Stream? bicepSources, string? documentationUri, string? description) + public override async Task PublishArtifact(OciArtifactReference reference, Stream compiledArmTemplate, Stream? bicepSources, string? documentationUri, string? description) { // This needs to be valid JSON, otherwise there may be compatibility issues. // NOTE: Bicep v0.20 and earlier will throw on this, so it's a breaking change. @@ -255,7 +255,7 @@ public override async Task PublishArtifact(OciModuleReference moduleReference, S { await this.client.PushArtifactAsync( configuration, - moduleReference, + reference, // Technically null should be fine for mediaType, but ACR guys recommend OciImageManifest for safer compatibility ManifestMediaType.OciImageManifest.ToString(), BicepModuleMediaTypes.BicepModuleArtifactType, @@ -278,7 +278,7 @@ await this.client.PushArtifactAsync( // media types are case-insensitive (they are lowercase by convention only) public static readonly StringComparison MediaTypeComparison = StringComparison.OrdinalIgnoreCase; - protected override void WriteArtifactContentToCache(OciModuleReference reference, OciArtifactResult result) + protected override void WriteArtifactContentToCache(OciArtifactReference reference, OciArtifactResult result) { /* * this should be kept in sync with the IsModuleRestoreRequired() implementation @@ -336,7 +336,7 @@ protected override void WriteArtifactContentToCache(OciModuleReference reference this.FileResolver.Write(this.GetModuleFileUri(reference, ModuleFileType.Metadata), metadataStream); } - protected override string GetArtifactDirectoryPath(OciModuleReference reference) + protected override string GetArtifactDirectoryPath(OciArtifactReference reference) { // cachePath is already set to %userprofile%\.bicep\br or ~/.bicep/br by default depending on OS // we need to split each component of the reference into a sub directory to fit within the max file name length limit on linux and mac @@ -377,9 +377,9 @@ protected override string GetArtifactDirectoryPath(OciModuleReference reference) return Path.Combine(this.cachePath, registry, repository, tagOrDigest); } - protected override Uri GetArtifactLockFileUri(OciModuleReference reference) => this.GetModuleFileUri(reference, ModuleFileType.Lock); + protected override Uri GetArtifactLockFileUri(OciArtifactReference reference) => this.GetModuleFileUri(reference, ModuleFileType.Lock); - private async Task<(OciArtifactResult?, string? errorMessage)> TryRestoreArtifactAsync(RootConfiguration configuration, OciModuleReference reference) + private async Task<(OciArtifactResult?, string? errorMessage)> TryRestoreArtifactAsync(RootConfiguration configuration, OciArtifactReference reference) { try { @@ -414,7 +414,7 @@ protected override string GetArtifactDirectoryPath(OciModuleReference reference) private static bool CheckAllInnerExceptionsAreRequestFailures(AggregateException exception) => exception.InnerExceptions.All(inner => inner is RequestFailedException); - private Uri GetModuleFileUri(OciModuleReference reference, ModuleFileType fileType) + private Uri GetModuleFileUri(OciArtifactReference reference, ModuleFileType fileType) { string localFilePath = this.GetModuleFilePath(reference, fileType); if (Uri.TryCreate(localFilePath, UriKind.Absolute, out var uri)) @@ -425,7 +425,7 @@ private Uri GetModuleFileUri(OciModuleReference reference, ModuleFileType fileTy throw new NotImplementedException($"Local module file path is malformed: \"{localFilePath}\""); } - private string GetModuleFilePath(OciModuleReference reference, ModuleFileType fileType) + private string GetModuleFilePath(OciArtifactReference reference, ModuleFileType fileType) { var fileName = fileType switch { @@ -441,7 +441,7 @@ private string GetModuleFilePath(OciModuleReference reference, ModuleFileType fi return Path.Combine(this.GetArtifactDirectoryPath(reference), fileName); } - public override SourceArchive? TryGetSource(OciModuleReference reference) + public override SourceArchive? TryGetSource(OciArtifactReference reference) { var zipPath = GetModuleFilePath(reference, ModuleFileType.Source); if (File.Exists(zipPath)) diff --git a/src/Bicep.Core/Registry/TemplateSpecModuleRegistry.cs b/src/Bicep.Core/Registry/TemplateSpecModuleRegistry.cs index 6e304b8f124..0ef777533e1 100644 --- a/src/Bicep.Core/Registry/TemplateSpecModuleRegistry.cs +++ b/src/Bicep.Core/Registry/TemplateSpecModuleRegistry.cs @@ -41,8 +41,12 @@ public TemplateSpecModuleRegistry(IFileResolver fileResolver, ITemplateSpecRepos public override RegistryCapabilities GetCapabilities(TemplateSpecModuleReference reference) => RegistryCapabilities.Default; - public override ResultWithDiagnostic TryParseArtifactReference(string? aliasName, string reference) + public override ResultWithDiagnostic TryParseArtifactReference(ArtifactType artifactType, string? aliasName, string reference) { + if (artifactType != ArtifactType.Module) + { + return new(x => x.UnsupportedArtifactType(artifactType)); + } if (!TemplateSpecModuleReference.TryParse(aliasName, reference, configuration, parentModuleUri).IsSuccess(out var @ref, out var failureBuilder)) { return new(failureBuilder); diff --git a/src/Bicep.Core/Semantics/Compilation.cs b/src/Bicep.Core/Semantics/Compilation.cs index 9ea6dc0b6c4..5ae24488e22 100644 --- a/src/Bicep.Core/Semantics/Compilation.cs +++ b/src/Bicep.Core/Semantics/Compilation.cs @@ -32,7 +32,7 @@ public Compilation( SourceFileGrouping sourceFileGrouping, IConfigurationManager configurationManager, IBicepAnalyzer linterAnalyzer, - IModuleReferenceFactory moduleReferenceFactory, + IArtifactReferenceFactory artifactReferenceFactory, ImmutableDictionary? modelLookup = null) { this.featureProviderFactory = featureProviderFactory; @@ -41,7 +41,7 @@ public Compilation( this.NamespaceProvider = namespaceProvider; this.configurationManager = configurationManager; this.linterAnalyzer = linterAnalyzer; - this.ModuleReferenceFactory = moduleReferenceFactory; + this.ArtifactReferenceFactory = artifactReferenceFactory; this.lazySemanticModelLookup = sourceFileGrouping.SourceFiles.ToImmutableDictionary( sourceFile => sourceFile, @@ -61,7 +61,7 @@ public Compilation( public INamespaceProvider NamespaceProvider { get; } - public IModuleReferenceFactory ModuleReferenceFactory { get; } + public IArtifactReferenceFactory ArtifactReferenceFactory { get; } public SemanticModel GetEntrypointSemanticModel() // entry point semantic models are guaranteed to cast successfully diff --git a/src/Bicep.Core/Semantics/ImportedSymbol.cs b/src/Bicep.Core/Semantics/ImportedSymbol.cs index 30e6de2d659..9af2eb1e9de 100644 --- a/src/Bicep.Core/Semantics/ImportedSymbol.cs +++ b/src/Bicep.Core/Semantics/ImportedSymbol.cs @@ -46,8 +46,8 @@ StringSyntax @string when @string.TryGetLiteralValue() is string literalValue => public string? TryGetDescription() => TryGetExportMetadata()?.Description; - public ResultWithDiagnostic TryGetModuleReference() - => Context.Compilation.ModuleReferenceFactory.TryGetModuleReference(EnclosingDeclaration, Context.SourceFile.FileUri); + public ResultWithDiagnostic TryGetArtifactReference() + => Context.Compilation.ArtifactReferenceFactory.TryGetArtifactReference(EnclosingDeclaration, Context.SourceFile.FileUri); public override void Accept(SymbolVisitor visitor) { diff --git a/src/Bicep.Core/Semantics/WildcardImportSymbol.cs b/src/Bicep.Core/Semantics/WildcardImportSymbol.cs index 117db717aab..567511caacd 100644 --- a/src/Bicep.Core/Semantics/WildcardImportSymbol.cs +++ b/src/Bicep.Core/Semantics/WildcardImportSymbol.cs @@ -38,8 +38,8 @@ public override void Accept(SymbolVisitor visitor) Context.Compilation) .TryUnwrap(); - public ResultWithDiagnostic TryGetModuleReference() - => Context.Compilation.ModuleReferenceFactory.TryGetModuleReference(EnclosingDeclaration, Context.SourceFile.FileUri); + public ResultWithDiagnostic TryGetArtifactReference() + => Context.Compilation.ArtifactReferenceFactory.TryGetArtifactReference(EnclosingDeclaration, Context.SourceFile.FileUri); public override IEnumerable GetDiagnostics() { diff --git a/src/Bicep.Core/Syntax/CompileTimeImportDeclarationSyntax.cs b/src/Bicep.Core/Syntax/CompileTimeImportDeclarationSyntax.cs index 3752198f348..3b7d0407972 100644 --- a/src/Bicep.Core/Syntax/CompileTimeImportDeclarationSyntax.cs +++ b/src/Bicep.Core/Syntax/CompileTimeImportDeclarationSyntax.cs @@ -4,6 +4,7 @@ using System.Linq; using Bicep.Core.Navigation; using Bicep.Core.Parsing; +using Bicep.Core.Registry; namespace Bicep.Core.Syntax; @@ -36,4 +37,6 @@ SyntaxBase IArtifactReferenceSyntax.SourceSyntax public override void Accept(ISyntaxVisitor visitor) => visitor.VisitCompileTimeImportDeclarationSyntax(this); public override TextSpan Span => TextSpan.Between(this.LeadingNodes.FirstOrDefault() ?? this.Keyword, this.FromClause); + + public ArtifactType GetArtifactType() => ArtifactType.Module; } diff --git a/src/Bicep.Core/Syntax/ImportSpecification.cs b/src/Bicep.Core/Syntax/ImportSpecification.cs index 9a2a1593aff..d49946a52ee 100644 --- a/src/Bicep.Core/Syntax/ImportSpecification.cs +++ b/src/Bicep.Core/Syntax/ImportSpecification.cs @@ -11,6 +11,7 @@ using System.Threading.Tasks; using Bicep.Core.Diagnostics; using Bicep.Core.Parsing; +using Bicep.Core.Semantics.Namespaces; namespace Bicep.Core.Syntax { @@ -20,22 +21,35 @@ public class ImportSpecification : ISymbolNameSource // language=regex private const string NamePattern = "[a-zA-Z][a-zA-Z0-9]+"; + private const string FromRegistryPattern = @"br[:\/]\S+"; + // Regex copied from https://semver.org/. // language=regex private const string SemanticVersionPattern = @"(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?"; - private static readonly Regex SpecificationPattern = new( + private static readonly Regex BuiltInSpecificationPattern = new( @$"^(?{NamePattern})@(?{SemanticVersionPattern})$", RegexOptions.ECMAScript | RegexOptions.Compiled); - private ImportSpecification(string name, string version, bool isValid, TextSpan span) + private static readonly Regex FromRegistrySpecificationPattern = new( + @$"^(?
{FromRegistryPattern})@(?{SemanticVersionPattern})$", + RegexOptions.ECMAScript | RegexOptions.Compiled); + + private static readonly Regex RepositoryNamePattern = new( + @"^\S*[:\/](?az)$", + RegexOptions.ECMAScript | RegexOptions.Compiled); + + private ImportSpecification(string unexpandedPath, string name, string version, bool isValid, TextSpan span) { + UnexpandedPath = unexpandedPath; Name = name; Version = version; IsValid = isValid; Span = span; } + private string UnexpandedPath { get; } + public string Name { get; } public string Version { get; } @@ -46,35 +60,58 @@ private ImportSpecification(string name, string version, bool isValid, TextSpan public static ImportSpecification From(SyntaxBase specificationSyntax) { - switch (specificationSyntax) + if (specificationSyntax is StringSyntax stringSyntax && stringSyntax.TryGetLiteralValue() is { } value) { - case StringSyntax stringSyntax when stringSyntax.TryGetLiteralValue() is { } value: - var (name, version, isValid) = Parse(value); - var span = isValid ? new TextSpan(stringSyntax.Span.Position + 1, name.Length) : stringSyntax.Span; - - return new ImportSpecification(name, version, isValid, span); + return CreateFromStringSyntax(stringSyntax, value); + } - case SkippedTriviaSyntax trivia: - return new ImportSpecification(trivia.TriviaName, trivia.TriviaName, false, trivia.Span); + return new ImportSpecification( + LanguageConstants.ErrorName, + LanguageConstants.ErrorName, + LanguageConstants.ErrorName, + false, + specificationSyntax.Span); + } - default: - return new ImportSpecification(LanguageConstants.ErrorName, LanguageConstants.ErrorName, false, specificationSyntax.Span); + public SyntaxBase ToPath() + { + if (!this.IsValid) + { + return new SkippedTriviaSyntax(this.Span, Enumerable.Empty()); } + return SyntaxFactory.CreateStringLiteral(this.UnexpandedPath); } - private static (string Name, string Version, bool IsValid) Parse(string value) + private static ImportSpecification CreateFromStringSyntax(StringSyntax stringSyntax, string value) { - var match = SpecificationPattern.Match(value); - - if (!match.Success) + if (BuiltInSpecificationPattern.Match(value) is { } builtInMatch && builtInMatch.Success) { - return (LanguageConstants.ErrorName, LanguageConstants.ErrorName, false); + var name = builtInMatch.Groups["name"].Value; + var span = new TextSpan(stringSyntax.Span.Position + 1, name.Length); + var version = builtInMatch.Groups["version"].Value; + // built-in providers (e.g. kubernetes@1.0.0 or sys@1.0.0) are allowed as long as the name is not 'az' + return new(name, name, version, name != AzNamespaceType.BuiltInName, span); } - var name = match.Groups["name"].Value; - var version = match.Groups["version"].Value; - - return (name, version, true); + if (FromRegistrySpecificationPattern.Match(value) is { } registryMatch && registryMatch.Success) + { + // NOTE(asilverman): The regex for the registry pattern is intentionally loose since it will be validated by the module resolver. + var address = registryMatch.Groups["address"].Value; + var version = registryMatch.Groups["version"].Value; + + var span = new TextSpan(stringSyntax.Span.Position + 1, address.Length); + // NOTE(asilverman): I normalize the artifact address to the way we represent module addresses, see https://github.com/Azure/bicep/issues/12202 + var unexpandedArtifactAddress = $"{address}:{version}"; + var name = RepositoryNamePattern.Match(address).Groups["name"].Value; + // NOTE(asilverman): Only a repo name of az is allowed for now. This shall be relaxed once we generalize dynamic type loading for other provider packages. + return new(unexpandedArtifactAddress, name, version, name == AzNamespaceType.BuiltInName, span); + } + return new( + LanguageConstants.ErrorName, + LanguageConstants.ErrorName, + LanguageConstants.ErrorName, + false, + stringSyntax.Span); } } } diff --git a/src/Bicep.Core/Syntax/ModuleDeclarationSyntax.cs b/src/Bicep.Core/Syntax/ModuleDeclarationSyntax.cs index 41b78721cd5..2c30e6ef3e9 100644 --- a/src/Bicep.Core/Syntax/ModuleDeclarationSyntax.cs +++ b/src/Bicep.Core/Syntax/ModuleDeclarationSyntax.cs @@ -6,6 +6,7 @@ using System.Linq; using Bicep.Core.Navigation; using Bicep.Core.Parsing; +using Bicep.Core.Registry; namespace Bicep.Core.Syntax { @@ -71,5 +72,7 @@ public ObjectSyntax GetBody() => this.TryGetBody() ?? throw new InvalidOperationException($"A valid module body is not available on this module due to errors. Use {nameof(TryGetBody)}() instead."); public bool HasCondition() => this.Value is IfConditionSyntax or ForSyntax { Body: IfConditionSyntax }; + + public ArtifactType GetArtifactType() => ArtifactType.Module; } } diff --git a/src/Bicep.Core/Syntax/ProviderDeclarationSyntax.cs b/src/Bicep.Core/Syntax/ProviderDeclarationSyntax.cs index 02cb101c6dc..a539c424357 100644 --- a/src/Bicep.Core/Syntax/ProviderDeclarationSyntax.cs +++ b/src/Bicep.Core/Syntax/ProviderDeclarationSyntax.cs @@ -6,6 +6,7 @@ using Bicep.Core.Features; using Bicep.Core.Navigation; using Bicep.Core.Parsing; +using Bicep.Core.Registry; using Bicep.Core.Registry.Oci; namespace Bicep.Core.Syntax @@ -48,6 +49,8 @@ public ProviderDeclarationSyntax(IEnumerable leadingNodes, Token key public override void Accept(ISyntaxVisitor visitor) => visitor.VisitProviderDeclarationSyntax(this); - public SyntaxBase Path => SyntaxFactory.CreateStringLiteral($@"{OciArtifactReferenceFacts.Scheme}:{FeatureProvider.ReadEnvVar("__EXPERIMENTAL_BICEP_REGISTRY_FQDN", LanguageConstants.BicepPublicMcrRegistry)}/bicep/providers/{this.Specification.Name}:{this.Specification.Version}"); + public ArtifactType GetArtifactType() => ArtifactType.Provider; + + public SyntaxBase Path => this.Specification.ToPath(); } } diff --git a/src/Bicep.Core/Syntax/TestDeclarationSyntax.cs b/src/Bicep.Core/Syntax/TestDeclarationSyntax.cs index 6ed59385c3a..c8fa5e520d0 100644 --- a/src/Bicep.Core/Syntax/TestDeclarationSyntax.cs +++ b/src/Bicep.Core/Syntax/TestDeclarationSyntax.cs @@ -5,6 +5,7 @@ using System.Linq; using Bicep.Core.Navigation; using Bicep.Core.Parsing; +using Bicep.Core.Registry; namespace Bicep.Core.Syntax { @@ -55,5 +56,7 @@ public TestDeclarationSyntax(IEnumerable leadingNodes, Token keyword public ObjectSyntax GetBody() => this.TryGetBody() ?? throw new InvalidOperationException($"A valid test body is not available on this test due to errors. Use {nameof(TryGetBody)}() instead."); + + public ArtifactType GetArtifactType() => ArtifactType.Module; } } diff --git a/src/Bicep.Core/Syntax/UsingDeclarationSyntax.cs b/src/Bicep.Core/Syntax/UsingDeclarationSyntax.cs index a4b0e2ef4ec..f32e91079be 100644 --- a/src/Bicep.Core/Syntax/UsingDeclarationSyntax.cs +++ b/src/Bicep.Core/Syntax/UsingDeclarationSyntax.cs @@ -3,6 +3,7 @@ using System.Linq; using Bicep.Core.Navigation; using Bicep.Core.Parsing; +using Bicep.Core.Registry; namespace Bicep.Core.Syntax { @@ -25,6 +26,8 @@ public UsingDeclarationSyntax(Token keyword, SyntaxBase path) public override void Accept(ISyntaxVisitor visitor) => visitor.VisitUsingDeclarationSyntax(this); + public ArtifactType GetArtifactType() => ArtifactType.Module; + public override TextSpan Span => TextSpan.Between(this.Keyword, this.Path); SyntaxBase IArtifactReferenceSyntax.SourceSyntax => Path; diff --git a/src/Bicep.Core/Workspaces/SourceFileGroupingBuilder.cs b/src/Bicep.Core/Workspaces/SourceFileGroupingBuilder.cs index 8624b08241b..68f23b36e82 100644 --- a/src/Bicep.Core/Workspaces/SourceFileGroupingBuilder.cs +++ b/src/Bicep.Core/Workspaces/SourceFileGroupingBuilder.cs @@ -21,13 +21,13 @@ namespace Bicep.Core.Workspaces public class SourceFileGroupingBuilder { private readonly IFileResolver fileResolver; - private readonly IModuleDispatcher moduleDispatcher; + private readonly IModuleDispatcher dispatcher; private readonly IReadOnlyWorkspace workspace; private readonly Dictionary> fileResultByUri; private readonly ConcurrentDictionary>> uriResultByArtifactReference; - private readonly bool forceModulesRestore; + private readonly bool forceRestore; private SourceFileGroupingBuilder( IFileResolver fileResolver, @@ -36,11 +36,11 @@ private SourceFileGroupingBuilder( bool forceModulesRestore = false) { this.fileResolver = fileResolver; - this.moduleDispatcher = moduleDispatcher; + this.dispatcher = moduleDispatcher; this.workspace = workspace; this.uriResultByArtifactReference = new(); this.fileResultByUri = new(); - this.forceModulesRestore = forceModulesRestore; + this.forceRestore = forceModulesRestore; } private SourceFileGroupingBuilder( @@ -48,14 +48,14 @@ private SourceFileGroupingBuilder( IModuleDispatcher moduleDispatcher, IReadOnlyWorkspace workspace, SourceFileGrouping current, - bool forceforceModulesRestore = false) + bool forceArtifactRestore = false) { this.fileResolver = fileResolver; - this.moduleDispatcher = moduleDispatcher; + this.dispatcher = moduleDispatcher; this.workspace = workspace; this.uriResultByArtifactReference = new(current.UriResultByArtifactReference.Select(kvp => KeyValuePair.Create(kvp.Key, kvp.Value.ToDictionary(p => p.Key, p => p.Value)))); this.fileResultByUri = current.FileResultByUri.Where(x => x.Value.TryUnwrap() is not null).ToDictionary(x => x.Key, x => x.Value); - this.forceModulesRestore = forceforceModulesRestore; + this.forceRestore = forceArtifactRestore; } public static SourceFileGrouping Build(IFileResolver fileResolver, IModuleDispatcher moduleDispatcher, IReadOnlyWorkspace workspace, Uri entryFileUri, IFeatureProviderFactory featuresFactory, bool forceModulesRestore = false) @@ -161,7 +161,7 @@ private void PopulateRecursive(BicepSourceFile file, IFeatureProviderFactory fea { continue; } - var (childModuleReference, uriResult) = GetModuleRestoreResult(file.FileUri, restorable); + var (childModuleReference, uriResult) = GetArtifactRestoreResult(file.FileUri, restorable); uriResultByArtifactReference.GetOrAdd(file, f => new())[restorable] = uriResult; @@ -181,40 +181,40 @@ private void PopulateRecursive(BicepSourceFile file, IFeatureProviderFactory fea } } - private (ArtifactReference? reference, Result result) GetModuleRestoreResult(Uri parentFileUri, IArtifactReferenceSyntax foreignTemplateReference) + private (ArtifactReference? reference, Result result) GetArtifactRestoreResult(Uri parentFileUri, IArtifactReferenceSyntax referenceSyntax) { - if (!moduleDispatcher.TryGetModuleReference(foreignTemplateReference, parentFileUri).IsSuccess(out var moduleReference, out var referenceResolutionError)) + if (!dispatcher.TryGetArtifactReference(referenceSyntax, parentFileUri).IsSuccess(out var artifactReference, out var referenceResolutionError)) { // module reference is not valid return (null, new(new UriResolutionError(referenceResolutionError, false))); } - if (!moduleDispatcher.TryGetLocalModuleEntryPointUri(moduleReference).IsSuccess(out var moduleFileUri, out var moduleGetPathFailureBuilder)) + if (!dispatcher.TryGetLocalArtifactEntryPointUri(artifactReference).IsSuccess(out var moduleFileUri, out var moduleGetPathFailureBuilder)) { - return (moduleReference, new(new UriResolutionError(moduleGetPathFailureBuilder, false))); + return (artifactReference, new(new UriResolutionError(moduleGetPathFailureBuilder, false))); } - if (forceModulesRestore) + if (forceRestore) { //override the status to force restore - return (moduleReference, new(new UriResolutionError(x => x.ModuleRequiresRestore(moduleReference.FullyQualifiedReference), true))); + return (artifactReference, new(new UriResolutionError(x => x.ModuleRequiresRestore(artifactReference.FullyQualifiedReference), true))); } - var restoreStatus = moduleDispatcher.GetArtifactRestoreStatus(moduleReference, out var restoreErrorBuilder); + var restoreStatus = dispatcher.GetArtifactRestoreStatus(artifactReference, out var restoreErrorBuilder); switch (restoreStatus) { case ArtifactRestoreStatus.Unknown: // we have not yet attempted to restore the module, so let's do it - return (moduleReference, new(new UriResolutionError(x => x.ModuleRequiresRestore(moduleReference.FullyQualifiedReference), true))); + return (artifactReference, new(new UriResolutionError(x => x.ModuleRequiresRestore(artifactReference.FullyQualifiedReference), true))); case ArtifactRestoreStatus.Failed: // the module has not yet been restored or restore failed // in either case, set the error - return (moduleReference, new(new UriResolutionError(restoreErrorBuilder ?? (x => x.ModuleRestoreFailed(moduleReference.FullyQualifiedReference)), false))); + return (artifactReference, new(new UriResolutionError(restoreErrorBuilder ?? (x => x.ModuleRestoreFailed(artifactReference.FullyQualifiedReference)), false))); default: break; } - return (moduleReference, new(moduleFileUri)); + return (artifactReference, new(moduleFileUri)); } private ILookup ReportFailuresForCycles() diff --git a/src/Bicep.LangServer.IntegrationTests/HoverTests.cs b/src/Bicep.LangServer.IntegrationTests/HoverTests.cs index 61afdf666bd..7ab581e5332 100644 --- a/src/Bicep.LangServer.IntegrationTests/HoverTests.cs +++ b/src/Bicep.LangServer.IntegrationTests/HoverTests.cs @@ -1055,7 +1055,7 @@ private IModuleDispatcher GetModuleDispatcher( tag); var moduleDispatcher = StrictMock.Of(); - moduleDispatcher.Setup(m => m.TryGetModuleReference(moduleDeclarationSyntax, parentModuleUri)).Returns(ResultHelper.Create(ociArtifactModuleReference, null)); + moduleDispatcher.Setup(m => m.TryGetArtifactReference(moduleDeclarationSyntax, parentModuleUri)).Returns(ResultHelper.Create(ociArtifactModuleReference, null)); return moduleDispatcher.Object; } diff --git a/src/Bicep.LangServer.IntegrationTests/Registry/ModuleRestoreSchedulerTests.cs b/src/Bicep.LangServer.IntegrationTests/Registry/ModuleRestoreSchedulerTests.cs index 08f0f2f090f..2f4b8408223 100644 --- a/src/Bicep.LangServer.IntegrationTests/Registry/ModuleRestoreSchedulerTests.cs +++ b/src/Bicep.LangServer.IntegrationTests/Registry/ModuleRestoreSchedulerTests.cs @@ -177,12 +177,12 @@ private class MockRegistry : IArtifactRegistry public bool IsArtifactRestoreRequired(ArtifactReference reference) => true; - public Task PublishArtifact(ArtifactReference moduleReference, Stream compiledArmTemplates, Stream? bicepSources, string? documentationUri, string? description) + public Task PublishArtifact(ArtifactReference reference, Stream compiledArmTemplates, Stream? bicepSources, string? documentationUri, string? description) { throw new NotImplementedException(); } - public Task CheckArtifactExists(ArtifactReference moduleReference) => throw new NotImplementedException(); + public Task CheckArtifactExists(ArtifactReference reference) => throw new NotImplementedException(); public Task> InvalidateArtifactsCache(IEnumerable references) { @@ -204,17 +204,17 @@ public ResultWithDiagnostic TryGetLocalArtifactEntryPointUri(ArtifactRefere public Task TryGetDescription(ArtifactReference reference) => Task.FromResult(null); - public ResultWithDiagnostic TryParseArtifactReference(string? aliasName, string reference) + public ResultWithDiagnostic TryParseArtifactReference(ArtifactType artifactType, string? _, string reference) { - return new(new MockModuleRef(reference, PathHelper.FilePathToFileUrl(Path.GetTempFileName()))); + return new(new MockArtifactRef(reference, PathHelper.FilePathToFileUrl(Path.GetTempFileName()))); } public SourceArchive? TryGetSource(ArtifactReference artifactReference) => null; } - private class MockModuleRef : ArtifactReference + private class MockArtifactRef : ArtifactReference { - public MockModuleRef(string value, Uri parentModuleUri) + public MockArtifactRef(string value, Uri parentModuleUri) : base("mock", parentModuleUri) { this.Value = value; diff --git a/src/Bicep.LangServer.UnitTests/BicepRegistryCacheRequestHandlerTests.cs b/src/Bicep.LangServer.UnitTests/BicepRegistryCacheRequestHandlerTests.cs index 6f11eb7cea1..37a2af7f7f8 100644 --- a/src/Bicep.LangServer.UnitTests/BicepRegistryCacheRequestHandlerTests.cs +++ b/src/Bicep.LangServer.UnitTests/BicepRegistryCacheRequestHandlerTests.cs @@ -11,6 +11,7 @@ using Bicep.Core.FileSystem; using Bicep.Core.Modules; using Bicep.Core.Registry; +using Bicep.Core.Registry.Oci; using Bicep.Core.SourceCode; using Bicep.Core.UnitTests; using Bicep.Core.UnitTests.Mock; @@ -39,7 +40,7 @@ public async Task InvalidModuleReferenceShouldThrow() const string ModuleRefStr = "hello"; var dispatcher = StrictMock.Of(); - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(null as ArtifactReference, x => x.ModuleRestoreFailed("blah"))); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(null as ArtifactReference, x => x.ModuleRestoreFailed("blah"))); var resolver = StrictMock.Of(); @@ -64,7 +65,7 @@ public async Task LocalModuleReferenceShouldThrow() localRef.Should().NotBeNull(); ArtifactReference? outRef = localRef; - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, failureBuilder)); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, failureBuilder)); var resolver = StrictMock.Of(); @@ -90,11 +91,11 @@ public async Task ExternalModuleNotInCacheShouldThrow() var configuration = IConfigurationManager.GetBuiltInConfiguration(); var parentModuleLocalPath = "/foo/main.bicep"; var parentModuleUri = new Uri($"file://{parentModuleLocalPath}"); - OciModuleReference.TryParse(null, UnqualifiedModuleRefStr, configuration, parentModuleUri).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParseModule(null, UnqualifiedModuleRefStr, configuration, parentModuleUri).IsSuccess(out var moduleReference).Should().BeTrue(); moduleReference.Should().NotBeNull(); ArtifactReference? outRef = moduleReference; - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, parentModuleUri)).Returns(ResultHelper.Create(outRef, failureBuilder)); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, parentModuleUri)).Returns(ResultHelper.Create(outRef, failureBuilder)); dispatcher.Setup(m => m.GetArtifactRestoreStatus(moduleReference!, out failureBuilder)).Returns(ArtifactRestoreStatus.Unknown); var resolver = StrictMock.Of(); @@ -120,13 +121,13 @@ public async Task ExternalModuleFailedEntryPointShouldThrow() var configuration = IConfigurationManager.GetBuiltInConfiguration(); var parentModuleLocalPath = "/main.bicep"; var parentModuleUri = new Uri($"file://{parentModuleLocalPath}"); - OciModuleReference.TryParse(null, UnqualifiedModuleRefStr, configuration, parentModuleUri).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParseModule(null, UnqualifiedModuleRefStr, configuration, parentModuleUri).IsSuccess(out var moduleReference).Should().BeTrue(); moduleReference.Should().NotBeNull(); ArtifactReference? outRef = moduleReference; - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, parentModuleUri)).Returns(ResultHelper.Create(outRef, failureBuilder)); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, parentModuleUri)).Returns(ResultHelper.Create(outRef, failureBuilder)); dispatcher.Setup(m => m.GetArtifactRestoreStatus(moduleReference!, out failureBuilder)).Returns(ArtifactRestoreStatus.Succeeded); - dispatcher.Setup(m => m.TryGetLocalModuleEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(null as Uri, x => x.ModuleRestoreFailed("blah"))); + dispatcher.Setup(m => m.TryGetLocalArtifactEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(null as Uri, x => x.ModuleRestoreFailed("blah"))); var resolver = StrictMock.Of(); @@ -155,13 +156,13 @@ public async Task FailureToReadEntryPointShouldThrow() var fileUri = new Uri("file:///main.bicep"); var configuration = IConfigurationManager.GetBuiltInConfiguration(); - OciModuleReference.TryParse(null, UnqualifiedModuleRefStr, configuration, fileUri).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParseModule(null, UnqualifiedModuleRefStr, configuration, fileUri).IsSuccess(out var moduleReference).Should().BeTrue(); moduleReference.Should().NotBeNull(); ArtifactReference? outRef = moduleReference; - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, null)); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, null)); dispatcher.Setup(m => m.GetArtifactRestoreStatus(moduleReference!, out nullBuilder)).Returns(ArtifactRestoreStatus.Succeeded); - dispatcher.Setup(m => m.TryGetLocalModuleEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(fileUri, null)); + dispatcher.Setup(m => m.TryGetLocalArtifactEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(fileUri, null)); SourceArchive? sourceArchive = null; dispatcher.Setup(m => m.TryGetModuleSources(moduleReference!)).Returns(sourceArchive); @@ -195,13 +196,13 @@ public async Task RestoredValidModule_WithNoSources_ShouldReturnJsonContents() var fileUri = new Uri("file:///foo/bar/main.bicep"); var configuration = ConfigurationManager.GetConfiguration(fileUri); - OciModuleReference.TryParse(null, UnqualifiedModuleRefStr, configuration, fileUri).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParseModule(null, UnqualifiedModuleRefStr, configuration, fileUri).IsSuccess(out var moduleReference).Should().BeTrue(); moduleReference.Should().NotBeNull(); ArtifactReference? outRef = moduleReference; - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, null)); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, null)); dispatcher.Setup(m => m.GetArtifactRestoreStatus(moduleReference!, out nullBuilder)).Returns(ArtifactRestoreStatus.Succeeded); - dispatcher.Setup(m => m.TryGetLocalModuleEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(fileUri, null)); + dispatcher.Setup(m => m.TryGetLocalArtifactEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(fileUri, null)); SourceArchive? sourceArchive = null; dispatcher.Setup(m => m.TryGetModuleSources(moduleReference!)).Returns(sourceArchive); @@ -234,13 +235,13 @@ public async Task RestoredValidModule_WithSource_ShouldReturnBicepContents() var fileUri = new Uri("file:///foo/bar/main.bicep"); var configuration = ConfigurationManager.GetConfiguration(fileUri); - OciModuleReference.TryParse(null, UnqualifiedModuleRefStr, configuration, fileUri).IsSuccess(out var moduleReference).Should().BeTrue(); + OciArtifactReference.TryParseModule(null, UnqualifiedModuleRefStr, configuration, fileUri).IsSuccess(out var moduleReference).Should().BeTrue(); moduleReference.Should().NotBeNull(); ArtifactReference? outRef = moduleReference; - dispatcher.Setup(m => m.TryGetModuleReference(ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, null)); + dispatcher.Setup(m => m.TryGetArtifactReference(ArtifactType.Module, ModuleRefStr, It.IsAny())).Returns(ResultHelper.Create(outRef, null)); dispatcher.Setup(m => m.GetArtifactRestoreStatus(moduleReference!, out nullBuilder)).Returns(ArtifactRestoreStatus.Succeeded); - dispatcher.Setup(m => m.TryGetLocalModuleEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(fileUri, null)); + dispatcher.Setup(m => m.TryGetLocalArtifactEntryPointUri(moduleReference!)).Returns(ResultHelper.Create(fileUri, null)); var bicepSource = "metadata hi 'mom'"; var sourceArchive = SourceArchive.FromStream(SourceArchive.PackSourcesIntoStream(fileUri, new Core.Workspaces.ISourceFile[] { diff --git a/src/Bicep.LangServer/Handlers/BicepDefinitionHandler.cs b/src/Bicep.LangServer/Handlers/BicepDefinitionHandler.cs index 1f744960a45..d87816048f4 100644 --- a/src/Bicep.LangServer/Handlers/BicepDefinitionHandler.cs +++ b/src/Bicep.LangServer/Handlers/BicepDefinitionHandler.cs @@ -115,7 +115,7 @@ private LocationOrLocationLinks HandleUnboundSymbolLocation(DefinitionParams req && matchingNodes[^3] is ModuleDeclarationSyntax moduleDeclarationSyntax && matchingNodes[^2] is StringSyntax stringToken && context.Compilation.SourceFileGrouping.TryGetSourceFile(moduleDeclarationSyntax).IsSuccess(out var sourceFile) - && this.moduleDispatcher.TryGetModuleReference(moduleDeclarationSyntax, request.TextDocument.Uri.ToUriEncoded()).IsSuccess(out var moduleReference)) + && this.moduleDispatcher.TryGetArtifactReference(moduleDeclarationSyntax, request.TextDocument.Uri.ToUriEncoded()).IsSuccess(out var moduleReference)) { return HandleModuleReference(context, stringToken, sourceFile, moduleReference); } @@ -128,7 +128,7 @@ private LocationOrLocationLinks HandleUnboundSymbolLocation(DefinitionParams req && matchingNodes[^4] is CompileTimeImportDeclarationSyntax importDeclarationSyntax && matchingNodes[^2] is StringSyntax stringToken && context.Compilation.SourceFileGrouping.TryGetSourceFile(importDeclarationSyntax).IsSuccess(out var sourceFile) - && this.moduleDispatcher.TryGetModuleReference(importDeclarationSyntax, request.TextDocument.Uri.ToUriEncoded()).IsSuccess(out var moduleReference)) + && this.moduleDispatcher.TryGetArtifactReference(importDeclarationSyntax, request.TextDocument.Uri.ToUriEncoded()).IsSuccess(out var moduleReference)) { // goto beginning of the module file. return GetFileDefinitionLocation( @@ -173,19 +173,19 @@ private LocationOrLocationLinks HandleUnboundSymbolLocation(DefinitionParams req return new(); } - private LocationOrLocationLinks HandleModuleReference(CompilationContext context, StringSyntax stringToken, ISourceFile sourceFile, ArtifactReference moduleReference) + private LocationOrLocationLinks HandleModuleReference(CompilationContext context, StringSyntax stringToken, ISourceFile sourceFile, ArtifactReference reference) { // Return the correct link format so our language client can display the sources return GetFileDefinitionLocation( - GetModuleSourceLinkUri(sourceFile, moduleReference), + GetModuleSourceLinkUri(sourceFile, reference), stringToken, context, new() { Start = new(0, 0), End = new(0, 0) }); } - private Uri GetModuleSourceLinkUri(ISourceFile sourceFile, ArtifactReference moduleReference) + private Uri GetModuleSourceLinkUri(ISourceFile sourceFile, ArtifactReference reference) { - if (!this.CanClientAcceptRegistryContent() || !moduleReference.IsExternal) + if (!this.CanClientAcceptRegistryContent() || !reference.IsExternal) { // the client doesn't support the bicep-cache scheme or we're dealing with a local module // just use the file URI @@ -198,7 +198,7 @@ private Uri GetModuleSourceLinkUri(ISourceFile sourceFile, ArtifactReference mod var sourceFilePath = sourceFile.FileUri.AbsolutePath; - if (moduleDispatcher.TryGetModuleSources(moduleReference) is SourceArchive sourceArchive) + if (moduleDispatcher.TryGetModuleSources(reference) is SourceArchive sourceArchive) { // We have Bicep source code available. // Replace the local cached JSON name (always main.json) with the actual source entrypoint filename (e.g. @@ -212,7 +212,7 @@ private Uri GetModuleSourceLinkUri(ISourceFile sourceFile, ArtifactReference mod // The file path and fully qualified reference may contain special characters (like :) that need to be url-encoded. sourceFilePath = WebUtility.UrlEncode(sourceFilePath); - var fullyQualifiedReference = WebUtility.UrlEncode(moduleReference.FullyQualifiedReference); + var fullyQualifiedReference = WebUtility.UrlEncode(reference.FullyQualifiedReference); // Encode the source file path as a path and the fully qualified reference as a fragment. // VsCode will pass it to our language client, which will respond by requesting the source to display via @@ -226,7 +226,7 @@ private Uri GetModuleSourceLinkUri(ISourceFile sourceFile, ArtifactReference mod private LocationOrLocationLinks HandleWildcardImportDeclaration(CompilationContext context, DefinitionParams request, SymbolResolutionResult result, WildcardImportSymbol wildcardImport) { if (context.Compilation.SourceFileGrouping.TryGetSourceFile(wildcardImport.EnclosingDeclaration).IsSuccess(out var sourceFile) && - wildcardImport.TryGetModuleReference().IsSuccess(out var moduleReference)) + wildcardImport.TryGetArtifactReference().IsSuccess(out var moduleReference)) { return GetFileDefinitionLocation( GetModuleSourceLinkUri(sourceFile, moduleReference), diff --git a/src/Bicep.LangServer/Handlers/BicepForceModulesRestoreCommandHandler.cs b/src/Bicep.LangServer/Handlers/BicepForceModulesRestoreCommandHandler.cs index 8ded4470705..90aa3278943 100644 --- a/src/Bicep.LangServer/Handlers/BicepForceModulesRestoreCommandHandler.cs +++ b/src/Bicep.LangServer/Handlers/BicepForceModulesRestoreCommandHandler.cs @@ -79,7 +79,7 @@ private async Task ForceModulesRestoreAndGenerateOutputMessage(DocumentU } // restore is supposed to only restore the module references that are syntactically valid - await this.moduleDispatcher.RestoreModules(modulesToRestoreReferences, forceModulesRestore: true); + await this.moduleDispatcher.RestoreModules(modulesToRestoreReferences, forceRestore: true); // if all are marked as success var sbRestoreSummary = new StringBuilder(); diff --git a/src/Bicep.LangServer/Handlers/BicepHoverHandler.cs b/src/Bicep.LangServer/Handlers/BicepHoverHandler.cs index 0215512fcd7..557be49de4c 100644 --- a/src/Bicep.LangServer/Handlers/BicepHoverHandler.cs +++ b/src/Bicep.LangServer/Handlers/BicepHoverHandler.cs @@ -193,7 +193,7 @@ private static async Task GetModuleMarkdown( if (registries != null && registries.Any() && - moduleDispatcher.TryGetModuleReference(module.DeclaringModule, uri).IsSuccess(out var moduleReference) && + moduleDispatcher.TryGetArtifactReference(module.DeclaringModule, uri).IsSuccess(out var moduleReference) && moduleReference is not null) { var registry = registries.FirstOrDefault(r => r.Scheme == moduleReference.Scheme); diff --git a/src/Bicep.LangServer/Handlers/BicepRegistryCacheRequestHandler.cs b/src/Bicep.LangServer/Handlers/BicepRegistryCacheRequestHandler.cs index 796fa2a982e..9f4afa7efd9 100644 --- a/src/Bicep.LangServer/Handlers/BicepRegistryCacheRequestHandler.cs +++ b/src/Bicep.LangServer/Handlers/BicepRegistryCacheRequestHandler.cs @@ -48,7 +48,7 @@ public Task Handle(BicepRegistryCacheParams request, // it indicates a code defect client or server-side. // In normal operation, the user should never see them regardless of how malformed their code is. - if (!moduleDispatcher.TryGetModuleReference(request.Target, request.TextDocument.Uri.ToUriEncoded()).IsSuccess(out var moduleReference)) + if (!moduleDispatcher.TryGetArtifactReference(ArtifactType.Module, request.Target, request.TextDocument.Uri.ToUriEncoded()).IsSuccess(out var moduleReference)) { throw new InvalidOperationException( $"The client specified an invalid module reference '{request.Target}'."); @@ -66,7 +66,7 @@ public Task Handle(BicepRegistryCacheParams request, $"The module '{moduleReference.FullyQualifiedReference}' has not yet been successfully restored."); } - if (!moduleDispatcher.TryGetLocalModuleEntryPointUri(moduleReference).IsSuccess(out var uri)) + if (!moduleDispatcher.TryGetLocalArtifactEntryPointUri(moduleReference).IsSuccess(out var uri)) { throw new InvalidOperationException( $"Unable to obtain the entry point URI for module '{moduleReference.FullyQualifiedReference}'."); diff --git a/src/vscode-bicep/schemas/bicepconfig.schema.json b/src/vscode-bicep/schemas/bicepconfig.schema.json index d5f9e19bcbc..87d66bbbb13 100644 --- a/src/vscode-bicep/schemas/bicepconfig.schema.json +++ b/src/vscode-bicep/schemas/bicepconfig.schema.json @@ -13,7 +13,12 @@ "title": "Diagnostic Level", "description": "Type of diagnostic to display, most rules default to warning", "type": "string", - "enum": [ "off", "info", "warning", "error" ] + "enum": [ + "off", + "info", + "warning", + "error" + ] }, "rule-def-level-warning": { "type": "object", @@ -25,7 +30,9 @@ "$ref": "#/definitions/level" } }, - "required": [ "level" ] + "required": [ + "level" + ] }, "rule-def-level-error": { "type": "object", @@ -37,7 +44,9 @@ "$ref": "#/definitions/level" } }, - "required": [ "level" ] + "required": [ + "level" + ] }, "rule-def-level-off": { "type": "object", @@ -49,12 +58,17 @@ "$ref": "#/definitions/level" } }, - "required": [ "level" ] + "required": [ + "level" + ] }, "cloudProfile": { "type": "object", "additionalProperties": false, - "required": [ "resourceManagerEndpoint", "activeDirectoryAuthority" ], + "required": [ + "resourceManagerEndpoint", + "activeDirectoryAuthority" + ], "properties": { "resourceManagerEndpoint": { "title": "Resource Manager Endpoint", @@ -84,7 +98,10 @@ "templateSpecModuleAlias": { "type": "object", "additionalProperties": false, - "required": [ "subscription", "resourceGroup" ], + "required": [ + "subscription", + "resourceGroup" + ], "properties": { "subscription": { "title": "Subscription ID", @@ -105,7 +122,9 @@ "bicepRegistryModuleAlias": { "type": "object", "additionalProperties": false, - "required": [ "registry" ], + "required": [ + "registry" + ], "properties": { "registry": { "title": "Registry", @@ -121,6 +140,28 @@ "minLength": 1 } } + }, + "bicepRegistryProviderAlias": { + "type": "object", + "additionalProperties": false, + "required": [ + "registry" + ], + "properties": { + "registry": { + "title": "Registry", + "description": "The OCI registry the alias is assigned to", + "type": "string", + "minLength": 1, + "maxLength": 255 + }, + "providerPath": { + "title": "Provider Path", + "description": "The OCI repository address to use for the alias", + "type": "string", + "minLength": 1 + } + } } }, "type": "object", @@ -130,14 +171,20 @@ "title": "Cloud", "type": "object", "additionalProperties": false, - "required": [ "currentProfile" ], + "required": [ + "currentProfile" + ], "properties": { "currentProfile": { "title": "Current Profile", "description": "The current cloud profile", "anyOf": [ { - "enum": [ "AzureCloud", "AzureChinaCloud", "AzureUSGovernment" ] + "enum": [ + "AzureCloud", + "AzureChinaCloud", + "AzureUSGovernment" + ] }, { "type": "string" @@ -364,7 +411,9 @@ "excludedhosts": { "description": "Customize the list of hosts to allow even if they contain an excluded host as a substring", "type": "array", - "default": [ "schema.management.azure.com" ], + "default": [ + "schema.management.azure.com" + ], "items": { "$id": "#/analyzers/core/rules/no-hardcoded-env-urls/excludedhosts/items", "title": "Items", @@ -668,12 +717,19 @@ "indentKind": { "type": "string", "description": "The indentation kind", - "enum": [ "Space", "Tab" ] + "enum": [ + "Space", + "Tab" + ] }, "newlineKind": { "type": "string", "description": "The newline kind", - "enum": [ "LF", "CRLF", "CR" ] + "enum": [ + "LF", + "CRLF", + "CR" + ] }, "indentSize": { "type": "integer", @@ -690,4 +746,4 @@ } } } -} +} \ No newline at end of file