From 5b50f03b7c6cf4b7972bb9c4165e7b4b01c34193 Mon Sep 17 00:00:00 2001 From: Taylor Southwick Date: Thu, 17 Jul 2025 12:14:38 -0700 Subject: [PATCH 1/3] Allow requiring OpenTelemetry protocol if needed For some scenarios (mine is runnin it ASP.NET framework applications), gRPC is not supported. This change adds an overload for AddOtlpExporter APIs to have a required protocol. Since there's not a default http endpoint if one is not set, this will throw an exception pointing the user to set that variable. --- .../OtlpExporterAnnotation.cs | 6 +- .../OtlpConfigurationExtensions.cs | 102 ++++++++++++++---- src/Aspire.Hosting/OtlpProtocol.cs | 20 ++++ .../WithOtlpExporterTests.cs | 72 +++++++++++++ 4 files changed, 181 insertions(+), 19 deletions(-) create mode 100644 src/Aspire.Hosting/OtlpProtocol.cs create mode 100644 tests/Aspire.Hosting.Tests/WithOtlpExporterTests.cs diff --git a/src/Aspire.Hosting/ApplicationModel/OtlpExporterAnnotation.cs b/src/Aspire.Hosting/ApplicationModel/OtlpExporterAnnotation.cs index c2698c7f188..abe67a95231 100644 --- a/src/Aspire.Hosting/ApplicationModel/OtlpExporterAnnotation.cs +++ b/src/Aspire.Hosting/ApplicationModel/OtlpExporterAnnotation.cs @@ -11,4 +11,8 @@ namespace Aspire.Hosting.ApplicationModel; [DebuggerDisplay("Type = {GetType().Name,nq}")] public class OtlpExporterAnnotation : IResourceAnnotation { -} \ No newline at end of file + /// + /// Gets or sets the default protocol for the OTLP exporter. + /// + public OtlpProtocol? RequiredProtocol { get; init; } +} diff --git a/src/Aspire.Hosting/OtlpConfigurationExtensions.cs b/src/Aspire.Hosting/OtlpConfigurationExtensions.cs index 3387a19150b..217667d8a2d 100644 --- a/src/Aspire.Hosting/OtlpConfigurationExtensions.cs +++ b/src/Aspire.Hosting/OtlpConfigurationExtensions.cs @@ -32,6 +32,30 @@ public static void AddOtlpEnvironment(IResource resource, IConfiguration configu // Add annotation to mark this resource as having OTLP exporter configured resource.Annotations.Add(new OtlpExporterAnnotation()); + RegisterOtlpEnvironment(resource, configuration, environment); + } + + /// + /// Configures OpenTelemetry in projects using environment variables. + /// + /// The resource to add annotations to. + /// The configuration to use for the OTLP exporter endpoint URL. + /// The host environment to check if the application is running in development mode. + /// The protocol to use for the OTLP exporter. If not set, it will try gRPC then Http. + public static void AddOtlpEnvironment(IResource resource, IConfiguration configuration, IHostEnvironment environment, OtlpProtocol protocol) + { + ArgumentNullException.ThrowIfNull(resource); + ArgumentNullException.ThrowIfNull(configuration); + ArgumentNullException.ThrowIfNull(environment); + + // Add annotation to mark this resource as having OTLP exporter configured + resource.Annotations.Add(new OtlpExporterAnnotation { RequiredProtocol = protocol }); + + RegisterOtlpEnvironment(resource, configuration, environment); + } + + private static void RegisterOtlpEnvironment(IResource resource, IConfiguration configuration, IHostEnvironment environment) + { // Configure OpenTelemetry in projects using environment variables. // https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/configuration/sdk-environment-variables.md @@ -43,26 +67,13 @@ public static void AddOtlpEnvironment(IResource resource, IConfiguration configu return; } - var dashboardOtlpGrpcUrl = configuration.GetString(KnownConfigNames.DashboardOtlpGrpcEndpointUrl, KnownConfigNames.Legacy.DashboardOtlpGrpcEndpointUrl); - var dashboardOtlpHttpUrl = configuration.GetString(KnownConfigNames.DashboardOtlpHttpEndpointUrl, KnownConfigNames.Legacy.DashboardOtlpHttpEndpointUrl); - - // The dashboard can support OTLP/gRPC and OTLP/HTTP endpoints at the same time, but it can - // only tell resources about one of the endpoints via environment variables. - // If both OTLP/gRPC and OTLP/HTTP are available then prefer gRPC. - if (dashboardOtlpGrpcUrl != null) - { - SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpGrpcUrl, "grpc"); - } - else if (dashboardOtlpHttpUrl != null) - { - SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpHttpUrl, "http/protobuf"); - } - else + if (!resource.TryGetLastAnnotation(out var otlpExporterAnnotation)) { - // No endpoints provided to host. Use default value for URL. - SetOtelEndpointAndProtocol(context.EnvironmentVariables, DashboardOtlpUrlDefaultValue, "grpc"); + return; } + SetOtel(context, configuration, otlpExporterAnnotation); + // Set the service name and instance id to the resource name and UID. Values are injected by DCP. var dcpDependencyCheckService = context.ExecutionContext.ServiceProvider.GetRequiredService(); var dcpInfo = await dcpDependencyCheckService.GetDcpInfoAsync(cancellationToken: context.CancellationToken).ConfigureAwait(false); @@ -91,6 +102,42 @@ public static void AddOtlpEnvironment(IResource resource, IConfiguration configu } })); + static void SetOtel(EnvironmentCallbackContext context, IConfiguration configuration, OtlpExporterAnnotation otlpExporterAnnotation) + { + var dashboardOtlpGrpcUrl = configuration.GetString(KnownConfigNames.DashboardOtlpGrpcEndpointUrl, KnownConfigNames.Legacy.DashboardOtlpGrpcEndpointUrl); + var dashboardOtlpHttpUrl = configuration.GetString(KnownConfigNames.DashboardOtlpHttpEndpointUrl, KnownConfigNames.Legacy.DashboardOtlpHttpEndpointUrl); + + // Check if a specific protocol is required by the annotation + if (otlpExporterAnnotation.RequiredProtocol is OtlpProtocol.Grpc) + { + SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpGrpcUrl ?? DashboardOtlpUrlDefaultValue, "grpc"); + } + else if (otlpExporterAnnotation.RequiredProtocol is OtlpProtocol.HttpProtobuf) + { + SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpHttpUrl ?? throw new InvalidOperationException("OtlpExporter is configured to require http/protobuf, but no endpoint was configured for ASPIRE_DASHBOARD_OTLP_HTTP_ENDPOINT_URL"), "http/protobuf"); + } + else + { + // No specific protocol required, use the existing preference logic + // The dashboard can support OTLP/gRPC and OTLP/HTTP endpoints at the same time, but it can + // only tell resources about one of the endpoints via environment variables. + // If both OTLP/gRPC and OTLP/HTTP are available then prefer gRPC. + if (dashboardOtlpGrpcUrl is not null) + { + SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpGrpcUrl, "grpc"); + } + else if (dashboardOtlpHttpUrl is not null) + { + SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpHttpUrl, "http/protobuf"); + } + else + { + // No endpoints provided to host. Use default value for URL. + SetOtelEndpointAndProtocol(context.EnvironmentVariables, DashboardOtlpUrlDefaultValue, "grpc"); + } + } + } + static void SetOtelEndpointAndProtocol(Dictionary environmentVariables, string url, string protocol) { environmentVariables["OTEL_EXPORTER_OTLP_ENDPOINT"] = new HostUrl(url); @@ -112,7 +159,26 @@ public static IResourceBuilder WithOtlpExporter(this IResourceBuilder b ArgumentNullException.ThrowIfNull(builder); AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment); - + + return builder; + } + + /// + /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. + /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. + /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. + /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. + /// + /// The resource type. + /// The resource builder. + /// The protocol to use for the OTLP exporter. If not set, it will try gRPC then Http. + /// The . + public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder, OtlpProtocol protocol) where T : IResourceWithEnvironment + { + ArgumentNullException.ThrowIfNull(builder); + + AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment, protocol); + return builder; } } diff --git a/src/Aspire.Hosting/OtlpProtocol.cs b/src/Aspire.Hosting/OtlpProtocol.cs new file mode 100644 index 00000000000..1f02e7b7251 --- /dev/null +++ b/src/Aspire.Hosting/OtlpProtocol.cs @@ -0,0 +1,20 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +namespace Aspire.Hosting; + +/// +/// Protocols available for OTLP exporters. +/// +public enum OtlpProtocol +{ + /// + /// A gRPC-based OTLP exporter. + /// + Grpc, + + /// + /// Http/Protobuf-based OTLP exporter. + /// + HttpProtobuf +} diff --git a/tests/Aspire.Hosting.Tests/WithOtlpExporterTests.cs b/tests/Aspire.Hosting.Tests/WithOtlpExporterTests.cs new file mode 100644 index 00000000000..f6a966e0099 --- /dev/null +++ b/tests/Aspire.Hosting.Tests/WithOtlpExporterTests.cs @@ -0,0 +1,72 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using Aspire.Hosting.Tests.Utils; +using Aspire.Hosting.Utils; +using Microsoft.AspNetCore.InternalTesting; +using Microsoft.Extensions.DependencyInjection; + +namespace Aspire.Hosting.Tests; + +public class WithOtlpExporterTests +{ + [InlineData(default, "http://localhost:8889", null, "http://localhost:8889", "grpc")] + [InlineData(default, "http://localhost:8889", "http://localhost:8890", "http://localhost:8889", "grpc")] + [InlineData(default, null, "http://localhost:8890", "http://localhost:8890", "http/protobuf")] + [InlineData(OtlpProtocol.HttpProtobuf, "http://localhost:8889", "http://localhost:8890", "http://localhost:8890", "http/protobuf")] + [InlineData(OtlpProtocol.Grpc, "http://localhost:8889", "http://localhost:8890", "http://localhost:8889", "grpc")] + [InlineData(OtlpProtocol.Grpc, null, null, "http://localhost:18889", "grpc")] + [Theory] + public async Task OtlpEndpointSet(OtlpProtocol? protocol, string? grpcEndpoint, string? httpEndpoint, string expectedUrl, string expectedProtocol) + { + using var builder = TestDistributedApplicationBuilder.Create(); + + builder.Configuration["ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL"] = grpcEndpoint; + builder.Configuration["ASPIRE_DASHBOARD_OTLP_HTTP_ENDPOINT_URL"] = httpEndpoint; + + var container = builder.AddResource(new ContainerResource("testSource")); + + if (protocol is { } value) + { + container = container.WithOtlpExporter(value); + } + else + { + container = container.WithOtlpExporter(); + } + + using var app = builder.Build(); + + var serviceProvider = app.Services.GetRequiredService(); + + var config = await EnvironmentVariableEvaluator.GetEnvironmentVariablesAsync( + container.Resource, + serviceProvider: serviceProvider + ).DefaultTimeout(); + + Assert.Equal(expectedUrl, config["OTEL_EXPORTER_OTLP_ENDPOINT"]); + Assert.Equal(expectedProtocol, config["OTEL_EXPORTER_OTLP_PROTOCOL"]); + } + + [Fact] + public async Task RequiredHttpOtlpThrowsExceptionIfNotRegistered() + { + using var builder = TestDistributedApplicationBuilder.Create(); + + builder.Configuration["ASPIRE_DASHBOARD_OTLP_HTTP_ENDPOINT_URL"] = null; + + var container = builder.AddResource(new ContainerResource("testSource")) + .WithOtlpExporter(OtlpProtocol.HttpProtobuf); + + using var app = builder.Build(); + + var serviceProvider = app.Services.GetRequiredService(); + + await Assert.ThrowsAsync(() => + EnvironmentVariableEvaluator.GetEnvironmentVariablesAsync( + container.Resource, + serviceProvider: serviceProvider + ).DefaultTimeout() + ); + } +} From cd2f3b2be4252e593b82401f245af22944d406b2 Mon Sep 17 00:00:00 2001 From: Taylor Southwick Date: Fri, 18 Jul 2025 09:14:59 -0700 Subject: [PATCH 2/3] add comment and move private method down --- .../OtlpConfigurationExtensions.cs | 84 +++++++++---------- 1 file changed, 42 insertions(+), 42 deletions(-) diff --git a/src/Aspire.Hosting/OtlpConfigurationExtensions.cs b/src/Aspire.Hosting/OtlpConfigurationExtensions.cs index 217667d8a2d..8d91c989a3c 100644 --- a/src/Aspire.Hosting/OtlpConfigurationExtensions.cs +++ b/src/Aspire.Hosting/OtlpConfigurationExtensions.cs @@ -48,12 +48,49 @@ public static void AddOtlpEnvironment(IResource resource, IConfiguration configu ArgumentNullException.ThrowIfNull(configuration); ArgumentNullException.ThrowIfNull(environment); - // Add annotation to mark this resource as having OTLP exporter configured + // Add annotation to mark this resource as having OTLP exporter configured with a required protocol resource.Annotations.Add(new OtlpExporterAnnotation { RequiredProtocol = protocol }); RegisterOtlpEnvironment(resource, configuration, environment); } + /// + /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. + /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. + /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. + /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. + /// + /// The resource type. + /// The resource builder. + /// The . + public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder) where T : IResourceWithEnvironment + { + ArgumentNullException.ThrowIfNull(builder); + + AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment); + + return builder; + } + + /// + /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. + /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. + /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. + /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. + /// + /// The resource type. + /// The resource builder. + /// The protocol to use for the OTLP exporter. If not set, it will try gRPC then Http. + /// The . + public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder, OtlpProtocol protocol) where T : IResourceWithEnvironment + { + ArgumentNullException.ThrowIfNull(builder); + + AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment, protocol); + + return builder; + } + private static void RegisterOtlpEnvironment(IResource resource, IConfiguration configuration, IHostEnvironment environment) { // Configure OpenTelemetry in projects using environment variables. @@ -72,7 +109,7 @@ private static void RegisterOtlpEnvironment(IResource resource, IConfiguration c return; } - SetOtel(context, configuration, otlpExporterAnnotation); + SetOtel(context, configuration, otlpExporterAnnotation.RequiredProtocol); // Set the service name and instance id to the resource name and UID. Values are injected by DCP. var dcpDependencyCheckService = context.ExecutionContext.ServiceProvider.GetRequiredService(); @@ -102,17 +139,17 @@ private static void RegisterOtlpEnvironment(IResource resource, IConfiguration c } })); - static void SetOtel(EnvironmentCallbackContext context, IConfiguration configuration, OtlpExporterAnnotation otlpExporterAnnotation) + static void SetOtel(EnvironmentCallbackContext context, IConfiguration configuration, OtlpProtocol? requiredProtocol) { var dashboardOtlpGrpcUrl = configuration.GetString(KnownConfigNames.DashboardOtlpGrpcEndpointUrl, KnownConfigNames.Legacy.DashboardOtlpGrpcEndpointUrl); var dashboardOtlpHttpUrl = configuration.GetString(KnownConfigNames.DashboardOtlpHttpEndpointUrl, KnownConfigNames.Legacy.DashboardOtlpHttpEndpointUrl); // Check if a specific protocol is required by the annotation - if (otlpExporterAnnotation.RequiredProtocol is OtlpProtocol.Grpc) + if (requiredProtocol is OtlpProtocol.Grpc) { SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpGrpcUrl ?? DashboardOtlpUrlDefaultValue, "grpc"); } - else if (otlpExporterAnnotation.RequiredProtocol is OtlpProtocol.HttpProtobuf) + else if (requiredProtocol is OtlpProtocol.HttpProtobuf) { SetOtelEndpointAndProtocol(context.EnvironmentVariables, dashboardOtlpHttpUrl ?? throw new InvalidOperationException("OtlpExporter is configured to require http/protobuf, but no endpoint was configured for ASPIRE_DASHBOARD_OTLP_HTTP_ENDPOINT_URL"), "http/protobuf"); } @@ -144,41 +181,4 @@ static void SetOtelEndpointAndProtocol(Dictionary environmentVar environmentVariables["OTEL_EXPORTER_OTLP_PROTOCOL"] = protocol; } } - - /// - /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. - /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. - /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. - /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. - /// - /// The resource type. - /// The resource builder. - /// The . - public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder) where T : IResourceWithEnvironment - { - ArgumentNullException.ThrowIfNull(builder); - - AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment); - - return builder; - } - - /// - /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. - /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. - /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. - /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. - /// - /// The resource type. - /// The resource builder. - /// The protocol to use for the OTLP exporter. If not set, it will try gRPC then Http. - /// The . - public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder, OtlpProtocol protocol) where T : IResourceWithEnvironment - { - ArgumentNullException.ThrowIfNull(builder); - - AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment, protocol); - - return builder; - } } From 1cd3bb67deb51bc326a64e184cf9ee126deeaa37 Mon Sep 17 00:00:00 2001 From: Taylor Southwick Date: Wed, 23 Jul 2025 08:08:52 -0700 Subject: [PATCH 3/3] react to comment changes from merge --- .../OtlpConfigurationExtensions.cs | 58 +++++++------------ 1 file changed, 21 insertions(+), 37 deletions(-) diff --git a/src/Aspire.Hosting/OtlpConfigurationExtensions.cs b/src/Aspire.Hosting/OtlpConfigurationExtensions.cs index a3274ef7ab2..1c3be620536 100644 --- a/src/Aspire.Hosting/OtlpConfigurationExtensions.cs +++ b/src/Aspire.Hosting/OtlpConfigurationExtensions.cs @@ -54,43 +54,6 @@ public static void AddOtlpEnvironment(IResource resource, IConfiguration configu RegisterOtlpEnvironment(resource, configuration, environment); } - /// - /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. - /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. - /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. - /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. - /// - /// The resource type. - /// The resource builder. - /// The . - public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder) where T : IResourceWithEnvironment - { - ArgumentNullException.ThrowIfNull(builder); - - AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment); - - return builder; - } - - /// - /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. - /// 1. It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. - /// 2. It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. - /// 3. It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. - /// - /// The resource type. - /// The resource builder. - /// The protocol to use for the OTLP exporter. If not set, it will try gRPC then Http. - /// The . - public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder, OtlpProtocol protocol) where T : IResourceWithEnvironment - { - ArgumentNullException.ThrowIfNull(builder); - - AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment, protocol); - - return builder; - } - private static void RegisterOtlpEnvironment(IResource resource, IConfiguration configuration, IHostEnvironment environment) { // Configure OpenTelemetry in projects using environment variables. @@ -201,4 +164,25 @@ public static IResourceBuilder WithOtlpExporter(this IResourceBuilder b return builder; } + + /// + /// Injects the appropriate environment variables to allow the resource to enable sending telemetry to the dashboard. + /// + /// It sets the OTLP endpoint to the value of the ASPIRE_DASHBOARD_OTLP_ENDPOINT_URL environment variable. + /// It sets the service name and instance id to the resource name and UID. Values are injected by the orchestrator. + /// It sets a small batch schedule delay in development. This reduces the delay that OTLP exporter waits to sends telemetry and makes the dashboard telemetry pages responsive. + /// + /// + /// The resource type. + /// The resource builder. + /// The protocol to use for the OTLP exporter. If not set, it will try gRPC then Http. + /// The . + public static IResourceBuilder WithOtlpExporter(this IResourceBuilder builder, OtlpProtocol protocol) where T : IResourceWithEnvironment + { + ArgumentNullException.ThrowIfNull(builder); + + AddOtlpEnvironment(builder.Resource, builder.ApplicationBuilder.Configuration, builder.ApplicationBuilder.Environment, protocol); + + return builder; + } }