Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Extension configuration updates for stacks extensibility support. #16444

Draft
wants to merge 21 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
75c7d5b
Add integration test for defining extension configs on the module.
kalbert312 Feb 18, 2025
39298de
Add extensions to the semantic model to support providing extension c…
kalbert312 Feb 18, 2025
de4cae6
Extend test to include bicepparam.
kalbert312 Mar 3, 2025
277d7d9
Add ability to specific extension configs in the bicepparams file.
kalbert312 Mar 3, 2025
2736e45
Merge branch 'main' into kylea/stacks-ext-configs
kalbert312 Mar 3, 2025
f524962
Add some baselines for extension config decls.
kalbert312 Mar 3, 2025
27247a0
Add bicepparam baseline for extension configs.
kalbert312 Mar 3, 2025
ad3a459
Update the params grammar.
kalbert312 Mar 3, 2025
73bddb0
Test updates.
kalbert312 Mar 4, 2025
07ed198
Start semantics for extension config assignments.
kalbert312 Mar 4, 2025
d9de6d4
Revert baseline changes.
kalbert312 Mar 4, 2025
394bc50
Update invalid modules diagnostics baseline.
kalbert312 Mar 4, 2025
265f6aa
Fix bicepparam baseline.
kalbert312 Mar 4, 2025
e35e261
Add an Extensions baseline.
kalbert312 Mar 4, 2025
859554b
Fix semantics of extension config assignment.
kalbert312 Mar 4, 2025
7793a21
Update test for graph.
kalbert312 Mar 4, 2025
de1fb58
Progress on emit for extensionConfigs on modules.
kalbert312 Mar 4, 2025
383395d
Progress on emit of extension configs in parameters files.
kalbert312 Mar 4, 2025
c6bf6c1
Update AKS baseline for required extensionConfigs on the module.
kalbert312 Mar 4, 2025
4b31e37
Merge branch 'main' into kylea/stacks-ext-configs
kalbert312 Mar 4, 2025
6c65a29
Fixes after merge.
kalbert312 Mar 4, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions docs/params-file-grammar.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ program -> statement* EOF
statement ->
usingDecl |
parameterDecl |
extensionConfigDecl |
NL

usingDecl ->
Expand All @@ -13,6 +14,9 @@ usingDecl ->
parameterDecl ->
"parameter" IDENTIFIER(name) "=" literalValue NL

extensionConfigDecl ->
"extension" IDENTIFIER(name) "with" object NL

stringLiteral -> "'" STRINGCHAR* "'"

literalValue -> NUMBER | "true" | "false" | "null" | stringLiteral | object | array
Expand Down
146 changes: 144 additions & 2 deletions src/Bicep.Core.IntegrationTests/ExtensibilityTests.cs
Original file line number Diff line number Diff line change
@@ -1,14 +1,12 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using Bicep.Core.CodeAction;
using Bicep.Core.Configuration;
using Bicep.Core.Diagnostics;
using Bicep.Core.IntegrationTests.Extensibility;
using Bicep.Core.UnitTests;
using Bicep.Core.UnitTests.Assertions;
using Bicep.Core.UnitTests.Utils;
using FluentAssertions;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using Newtonsoft.Json.Linq;

Expand Down Expand Up @@ -656,5 +654,149 @@ extension foo as foo
result.Template.Should().HaveValueAtPath("$.extensions.foo.name", "Foo");
result.Template.Should().HaveValueAtPath("$.resources.myApp.extension", "foo");
}

[TestMethod]
public void Module_with_required_extension_config_can_be_compiled_successfully()
{
var paramsUri = new Uri("file:///main.bicepparam");
var mainUri = new Uri("file:///main.bicep");
var moduleAUri = new Uri("file:///modulea.bicep");

// TODO(kylealbert): Remove 'with' clause in template when that's removed
var files = new Dictionary<Uri, string>
{
[paramsUri] =
"""
using 'main.bicep'

param inputa = 'abc'

extension k8s with {
kubeConfig: 'abc'
namespace: 'other'
}
""",
[mainUri] =
"""
param inputa string

extension kubernetes with {
kubeConfig: 'DELETE'
namespace: 'DELETE'
} as k8s

extension 'br:mcr.microsoft.com/bicep/extensions/microsoftgraph/v1.0:0.1.8-preview'

module modulea 'modulea.bicep' = {
name: 'modulea'
params: {
inputa: inputa
}
extensionConfigs: {
kubernetes: {
kubeConfig: 'fromModule'
namespace: 'other'
}
}
}

output outputa string = modulea.outputs.outputa
""",
[moduleAUri] =
"""
param inputa string

extension kubernetes with {
kubeConfig: 'DELETE'
namespace: 'DELETE'
}

extension 'br:mcr.microsoft.com/bicep/extensions/microsoftgraph/v1.0:0.1.8-preview' as graph

output outputa string = inputa
"""
};

var compilation = Services.BuildCompilation(files, paramsUri);

compilation.Should().NotHaveAnyDiagnostics_WithAssertionScoping(d => d.IsError());
}

[DataTestMethod]
[DataRow("MissingExtensionConfigsDeclaration")]
[DataRow("MissingRequiredExtensionConfig")]
[DataRow("MissingRequiredConfigProperty")]
[DataRow("PropertyIsNotDefinedInSchema")]
[DataRow("ConfigProvidedForExtensionThatDoesNotAcceptConfig")]
public void Module_with_invalid_extension_config_produces_diagnostic(string scenario)
{
var mainUri = new Uri("file:///main.bicep");
var moduleAUri = new Uri("file:///modulea.bicep");

var extensionConfigsStr = scenario switch
{
"MissingExtensionConfigsDeclaration" => "",
"MissingRequiredExtensionConfig" => "extensionConfigs: {}",
"MissingRequiredConfigProperty" => "extensionConfigs: { kubernetes: { namespace: 'other' } }",
"PropertyIsNotDefinedInSchema" => "extensionConfigs: { kubernetes: { kubeConfig: 'test', namespace: 'other', extra: 'extra' } }",
"ConfigProvidedForExtensionThatDoesNotAcceptConfig" => "extensionConfigs: { kubernetes: { kubeConfig: 'test', namespace: 'other' }, graph: { } }",
_ => throw new NotImplementedException()
};

// TODO(kylealbert): Remove 'with' clause in template when that's removed
var files = new Dictionary<Uri, string>
{
[mainUri] =
$$"""
param inputa string

module modulea 'modulea.bicep' = {
name: 'modulea'
params: {
inputa: inputa
}
{{extensionConfigsStr}}
}

output outputa string = modulea.outputs.outputa
""",
[moduleAUri] =
"""
param inputa string

extension kubernetes with {
kubeConfig: ''
namespace: 'default'
}

extension microsoftGraph as graph

output outputa string = inputa
"""
};

var compilation = Services.BuildCompilation(files, mainUri);

if (scenario is "MissingExtensionConfigsDeclaration")
{
compilation.Should().ContainSingleDiagnostic("BCP035", DiagnosticLevel.Error, """The specified "module" declaration is missing the following required properties: "extensionConfigs".""");
}
else if (scenario is "MissingRequiredExtensionConfig")
{
compilation.Should().ContainSingleDiagnostic("BCP035", DiagnosticLevel.Error, """The specified "object" declaration is missing the following required properties: "kubernetes".""");
}
else if (scenario is "MissingRequiredConfigProperty")
{
compilation.Should().ContainSingleDiagnostic("BCP035", DiagnosticLevel.Error, """The specified "object" declaration is missing the following required properties: "kubeConfig".""");
}
else if (scenario is "PropertyIsNotDefinedInSchema")
{
compilation.Should().ContainSingleDiagnostic("BCP037", DiagnosticLevel.Error, """The property "extra" is not allowed on objects of type "configuration". Permissible properties include "context".""");
}
else if (scenario is "ConfigProvidedForExtensionThatDoesNotAcceptConfig")
{
compilation.Should().ContainSingleDiagnostic("BCP037", DiagnosticLevel.Error, """The property "graph" is not allowed on objects of type "extensionConfigs". No other properties are allowed.""");
}
}
}
}
45 changes: 12 additions & 33 deletions src/Bicep.Core.IntegrationTests/ModuleTests.cs
Original file line number Diff line number Diff line change
@@ -1,16 +1,14 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
using System.Collections.Immutable;

using System.Diagnostics.CodeAnalysis;
using System.Text;
using Bicep.Core.Configuration;
using Bicep.Core.Diagnostics;
using Bicep.Core.Emit;
using Bicep.Core.FileSystem;
using Bicep.Core.Registry;
using Bicep.Core.Semantics;
using Bicep.Core.UnitTests;
using Bicep.Core.UnitTests.Assertions;
using Bicep.Core.UnitTests.Extensions;
using Bicep.Core.UnitTests.Features;
using Bicep.Core.UnitTests.FileSystem;
using Bicep.Core.UnitTests.Utils;
Expand Down Expand Up @@ -83,10 +81,10 @@ param inputb string

var compilation = Services.BuildCompilation(files, mainUri);

var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile.Values.SelectMany(x => x).Should().BeEmpty();
success.Should().BeTrue();
GetTemplate(compilation).Should().NotBeEmpty();
compilation.GetTestTemplate().Should().NotBeEmpty();
}

[TestMethod]
Expand All @@ -112,7 +110,7 @@ param inputb string

var compilation = Services.BuildCompilation(files, mainUri);

var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile[mainUri].Should().HaveDiagnostics(new[] {
("BCP094", DiagnosticLevel.Error, "This module references itself, which is not allowed."),
});
Expand Down Expand Up @@ -166,7 +164,7 @@ param inputb string

var compilation = Services.BuildCompilation(files, mainUri);

var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile[mainUri].Should().HaveDiagnostics(new[] {
("BCP095", DiagnosticLevel.Error, "The file is involved in a cycle (\"/modulea.bicep\" -> \"/moduleb.bicep\" -> \"/main.bicep\")."),
});
Expand Down Expand Up @@ -224,7 +222,7 @@ param inputb string
var compiler = ServiceBuilder.Create(s => s.WithFileResolver(mockFileResolver.Object).WithDisabledAnalyzersConfiguration()).GetCompiler();
var compilation = await compiler.CreateCompilation(mainFileUri);

var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile[mainFileUri].Should().HaveDiagnostics(new[] {
("BCP093", DiagnosticLevel.Error, "File path \"modulea.bicep\" could not be resolved relative to \"/path/to/main.bicep\"."),
});
Expand Down Expand Up @@ -296,11 +294,11 @@ param inputb int

var compilation = Services.BuildCompilation(files, mainUri);

var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile.Values.SelectMany(x => x).Should().BeEmpty();
success.Should().BeTrue();

var templateString = GetTemplate(compilation);
var templateString = compilation.GetTestTemplate();
var template = JToken.Parse(templateString);
template.Should().NotBeNull();

Expand Down Expand Up @@ -366,7 +364,7 @@ param inputb string
var compiler = ServiceBuilder.Create(s => s.WithFileResolver(mockFileResolver.Object).WithDisabledAnalyzersConfiguration()).GetCompiler();
var compilation = await compiler.CreateCompilation(mainUri);

var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile[mainUri].Should().HaveDiagnostics(new[] {
("BCP091", DiagnosticLevel.Error, "An error occurred reading file. Mock read failure!"),
});
Expand Down Expand Up @@ -802,31 +800,12 @@ public void Module_collection_with_generated_name_can_be_referenced_correctly(st
result.Template.Should().HaveValueAtPath("$.outputs.allModNames.copy.input", $"[format('{symbolicNamePrefix}-{{0}}-{{1}}', range(0, 10)[copyIndex()], uniqueString('{symbolicName}', deployment().name))]");
}

private static string GetTemplate(Compilation compilation)
{
var stringBuilder = new StringBuilder();
var stringWriter = new StringWriter(stringBuilder);

var emitter = new TemplateEmitter(compilation.GetEntrypointSemanticModel());
emitter.Emit(stringWriter);

return stringBuilder.ToString();
}

private static (bool success, IDictionary<Uri, ImmutableArray<IDiagnostic>> diagnosticsByFile) GetSuccessAndDiagnosticsByFile(Compilation compilation)
{
var diagnosticsByFile = compilation.GetAllDiagnosticsByBicepFile().ToDictionary(kvp => kvp.Key.Uri, kvp => kvp.Value);
var success = diagnosticsByFile.Values.SelectMany(x => x).All(d => !d.IsError());

return (success, diagnosticsByFile);
}

private static void ModuleTemplateHashValidator(Compilation compilation, string expectedTemplateHash)
{
var (success, diagnosticsByFile) = GetSuccessAndDiagnosticsByFile(compilation);
var (success, diagnosticsByFile) = compilation.GetSuccessAndDiagnosticsByBicepFile();
diagnosticsByFile.Values.SelectMany(x => x).Should().BeEmpty();
success.Should().BeTrue();
var templateString = GetTemplate(compilation);
var templateString = compilation.GetTestTemplate();
var template = JToken.Parse(templateString);
template.Should().NotBeNull();
template.SelectToken(BicepTestConstants.GeneratorTemplateHashPath)?.ToString().Should().Be(expectedTemplateHash);
Expand Down
2 changes: 2 additions & 0 deletions src/Bicep.Core.Samples/BaselineData_Bicepparam.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using System.Reflection;
using Bicep.Core.UnitTests.Baselines;
using FluentAssertions;
Expand Down Expand Up @@ -98,6 +99,7 @@ private static IEnumerable<BaselineData_Bicepparam> GetAllExampleData()
// ensure this list is kept up-to-date to validate that we're picking all of the baseline tests
embeddedFiles.Select(x => x.StreamPath).Should().BeEquivalentTo(
"Files/baselines_bicepparam/Expressions/parameters.bicepparam",
"Files/baselines_bicepparam/Extensions/parameters.bicepparam",
"Files/baselines_bicepparam/Imports/parameters.bicepparam",
"Files/baselines_bicepparam/Extends/parameters.bicepparam",
"Files/baselines_bicepparam/Invalid_Expressions/parameters.bicepparam",
Expand Down
2 changes: 2 additions & 0 deletions src/Bicep.Core.Samples/DataSets.cs
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@ public static class DataSets

public static DataSet Empty => CreateDataSet();

public static DataSet Extensions_CRLF => CreateDataSet();

public static DataSet Functions_LF => CreateDataSet();

public static DataSet Imports_LF => CreateDataSet();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"experimentalFeaturesEnabled": {
"extensibility": true
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
extension kubernetes with {
kubeConfig: 'DELETE'
namespace: 'DELETE'
} as k8s

extension 'br:mcr.microsoft.com/bicep/extensions/microsoftgraph/v1.0:0.1.8-preview' as graph
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
extension kubernetes with {
kubeConfig: 'DELETE'
namespace: 'DELETE'
}

extension 'br:mcr.microsoft.com/bicep/extensions/microsoftgraph/v1.0:0.1.8-preview'
34 changes: 34 additions & 0 deletions src/Bicep.Core.Samples/Files/baselines/Extensions_CRLF/main.bicep
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
// BEGIN: Extension declarations

extension kubernetes with {
kubeConfig: 'DELETE'
namespace: 'DELETE'
} as k8s

extension 'br:mcr.microsoft.com/bicep/extensions/microsoftgraph/v1.0:0.1.8-preview' as graph

// END: Extension declarations

// BEGIN: Extension configs for modules

module moduleWithExtsWithAliases 'child/hasConfigurableExtensionsWithAlias.bicep' = {
name: 'moduleWithExtsWithAliases'
extensionConfigs: {
k8s: {
kubeConfig: 'kubeConfig2FromModule'
namespace: 'ns2FromModule'
}
}
}

module moduleWithExtsWithoutAliases 'child/hasConfigurableExtensionsWithoutAlias.bicep' = {
name: 'moduleWithExtsWithoutAlaises'
extensionConfigs: {
kubernetes: {
kubeConfig: 'kubeConfig2FromModule'
namespace: 'ns2FromModule'
}
}
}

// END: Extension configs for modules
Loading
Loading