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

feat: Add Azure EventHubs module #1183

Open
wants to merge 4 commits into
base: develop
Choose a base branch
from

Conversation

WakaToa
Copy link
Contributor

@WakaToa WakaToa commented May 22, 2024

What does this PR do?

Microsoft released an emulator for EventHubs yesterday, see Azure/azure-service-bus#223 (comment)

This PR included a module for the emulator.

Info

You need to provide an Azurite (emulator) instance/endpoint to start the EventHub emulator. This can be done using the EventHubsConfiguration.

You need to specify the following configuration file when starting the emulator.

{
  "UserConfig": {
    "NamespaceConfig": [
      {
        "Type": "EventHub",
        "Name": "emulatorNs1",
        "Entities": [
          {
            "Name": "eh1",
            "PartitionCount": "2",
            "ConsumerGroups": [
              {
                "Name": "cg1"
              }
            ]
          }
        ]
      }
    ], 
    "LoggingConfig": {
      "Type": "File"
    }
  }
}

It will be dynamically mapped and i have built a floating builder so that the user can set the configuration as easily as possible.
The namespace "emulatorNs1" is mandatory and it is the only one currently supported by the emulator.

var configurationBuilder = ConfigurationBuilder
    .Create()
    .WithEventHub(EventHubName, "2", new[] { EventHubConsumerGroupName });

var builder = new EventHubsBuilder()
    .WithNetwork(_network)
    .WithConfigurationBuilder(configurationBuilder)
    .WithAzuriteBlobEndpoint(AzuriteNetworkAlias)
    .WithAzuriteTableEndpoint(AzuriteNetworkAlias);

_eventHubsContainer = builder.Build();

await _eventHubsContainer.StartAsync();

Copy link

netlify bot commented May 22, 2024

Deploy Preview for testcontainers-dotnet ready!

Name Link
🔨 Latest commit b264e42
🔍 Latest deploy log https://app.netlify.com/sites/testcontainers-dotnet/deploys/664de1efb02bc800082444d3
😎 Deploy Preview https://deploy-preview-1183--testcontainers-dotnet.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@HofmeisterAn HofmeisterAn added enhancement New feature or request module An official Testcontainers module labels May 27, 2024
Comment on lines +5 to +23
private readonly AzuriteContainer _azuriteContainer;

private EventHubsContainer _eventHubsContainer;

private readonly INetwork _network = new NetworkBuilder().WithName(NetworkName).Build();

private const string NetworkName = "eh-network";
private const string AzuriteNetworkAlias = "azurite";

private const string EventHubName = "testeventhub";
private const string EventHubConsumerGroupName = "testconsumergroup";

private EventHubsContainerTest()
{
_azuriteContainer = new AzuriteBuilder()
.WithNetwork(_network)
.WithNetworkAliases(AzuriteNetworkAlias)
.Build();
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an idea still but due to EventHubs emulator will not work without Azurite then we can provide two ways to run EventHub Container implementation.

  1. Embed Azurite, so, users don't worry about it.
  2. Allow external Azurite in case users need a specific configuration

/cc @kiview WDYT?

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does a pattern exist already ?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it makes sense to align with the pattern Service Bus is using and then follow up with something that allows us to use an existing Azurite instance. Please also see this issue: #1323. This is something we need to figure out and support in general.

@eddumelendez
Copy link
Member

Also, please add documentation for the module.

@Saglodha
Copy link

Saglodha commented Jan 6, 2025

Hi @WakaToa / @eddumelendez / @kiview ,

I am from Azure Event Hubs Product team and was looking at this PR. I do realize that there were some plausible solutions offered above to get this integration setup.

Could you please let us know if we could extend any help from our side to integrate Event Hubs emulator with testcontainers framework? We'll be happy to provide any product related assistance here.

Copy link
Collaborator

@HofmeisterAn HofmeisterAn left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the PR. There are a few things we need to address before we can merge the PR.

{
Validate();

var waitStrategy = Wait.ForUnixContainer().UntilMessageIsLogged("Emulator Service is Successfully Up!");
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The wait strategy should be moved to Init().

@@ -0,0 +1,53 @@
namespace Testcontainers.EventHubs.Configuration
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please change all to file scoped namespaces.

Comment on lines +40 to +46
public EventHubsBuilder WithConfigurationBuilder(ConfigurationBuilder configurationBuilder)
{
var configBytes = Encoding.UTF8.GetBytes(configurationBuilder.Build());

return Merge(DockerResourceConfiguration, new EventHubsConfiguration(configurationBuilder: configurationBuilder))
.WithResourceMapping(configBytes, "Eventhubs_Emulator/ConfigFiles/Config.json");
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this builder cover all Event Hubs configuration? Are we responsible for it? Is there a default configuration? The EventHubsBuilder will fail if the user does not call WithConfigurationBuilder(ConfigurationBuilder) explicitly. Until now, no other module requires an additional explicit call to a builder method (expect license agreements). _ = new FooBuilder.Build() always returns a working configuration. This is something I favor.

{
return base.Init()
.WithImage(EventHubsImage)
.WithEnvironment("ACCEPT_EULA", "Y")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please do not hide license agreements. Right now, we use the following pattern.

Comment on lines +66 to +69
/// <summary>
/// Gets the configuration builder
/// </summary>
public ConfigurationBuilder ConfigurationBuilder { get; }
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like we do not use the ConfigurationBuilder sometime later. In that case, we do not need to store it in the EventHubsConfiguration.

@@ -0,0 +1,127 @@
using DotNet.Testcontainers;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please use the global usings file.


private EventHubsContainer _eventHubsContainer;

private readonly INetwork _network = new NetworkBuilder().WithName(NetworkName).Build();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please do not use static network names. This may clash with already existing resources and may prevent tests from running in parallel. See our best practices.

Comment on lines +5 to +23
private readonly AzuriteContainer _azuriteContainer;

private EventHubsContainer _eventHubsContainer;

private readonly INetwork _network = new NetworkBuilder().WithName(NetworkName).Build();

private const string NetworkName = "eh-network";
private const string AzuriteNetworkAlias = "azurite";

private const string EventHubName = "testeventhub";
private const string EventHubConsumerGroupName = "testconsumergroup";

private EventHubsContainerTest()
{
_azuriteContainer = new AzuriteBuilder()
.WithNetwork(_network)
.WithNetworkAliases(AzuriteNetworkAlias)
.Build();
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it makes sense to align with the pattern Service Bus is using and then follow up with something that allows us to use an existing Azurite instance. Please also see this issue: #1323. This is something we need to figure out and support in general.

@@ -0,0 +1,22 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFrameworks>net8.0</TargetFrameworks>
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
<TargetFrameworks>net8.0</TargetFrameworks>
<TargetFrameworks>net9.0</TargetFrameworks>

Comment on lines +19 to +21
<ItemGroup>
<PackageVersion Update="Azure.Messaging.EventHubs" Version="5.11.3" />
</ItemGroup>
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
<ItemGroup>
<PackageVersion Update="Azure.Messaging.EventHubs" Version="5.11.3" />
</ItemGroup>

Already included above.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request module An official Testcontainers module
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants