Ask Learn Preview
Please sign in to use this experience.
Sign inThis browser is no longer supported.
Upgrade to Microsoft Edge to take advantage of the latest features, security updates, and technical support.
This module requires a sandbox to complete. A sandbox gives you access to free resources. Your personal subscription will not be charged. The sandbox may only be used to complete training on Microsoft Learn. Use for any other reason is prohibited, and may result in permanent loss of access to the sandbox.
Microsoft provides this lab experience and related content for educational purposes. All presented information is owned by Microsoft and intended solely for learning about the covered products and services in this Microsoft Learn module.
You've decided to use an Azure Service Bus topic to distribute sales performance messages in your salesforce application. Sales personnel will use the app on their mobile devices to send messages that summarize sales figures for each area and time period. Those messages are distributed to web services that are located in the company's operational regions, including the Americas and Europe.
You've already implemented the necessary infrastructure in your Azure subscriptions for the topic. Now, you want to write the code that sends messages to the topic and write code that retrieves messages from a subscription. Then, you'll send a message to a topic and retrieve the message for a specific subscription.
Make sure you're working in the correct directory by running the following commands in Azure Cloud Shell:
cd ~/mslearn-connect-services-together/implement-message-workflows-with-service-bus/src/start
code .
To complete the component that sends messages about sales performance, complete these steps:
In the Azure Cloud Shell editor, open performancemessagesender/Program.cs and find the following line of code:
const string ServiceBusConnectionString = "";
Between the quotation marks, paste the connection string you saved in the previous exercise.
If you used a name different from salesperformancemessages for the queue name, update the value for TopicName
property in the code:
const string TopicName = "salesperformancemessages";
Find the SendPerformanceMessageAsync()
method. (Hint: It's at or near line 26.) Within that method, find the following line of code:
// Create a Service Bus client here
Replace that line of code with this code:
// By leveraging "await using", the DisposeAsync method will be called automatically when the client variable goes out of scope.
// In more realistic scenarios, you would store off a class reference to the client (rather than to a local variable) so that it can be used throughout your program.
await using var client = new ServiceBusClient(ServiceBusConnectionString);
Within the SendPerformanceMessageAsync()
method, find the following line of code:
// Create a sender here
Replace that line of code with this code:
await using ServiceBusSender sender = client.CreateSender(TopicName);
In the try...catch
block, find the following line of code:
// Create and send a message here
Replace that line of code with this code:
string messageBody = "Total sales for Brazil in August: $13m.";
var message = new ServiceBusMessage(messageBody);
To display the message in the console, insert the following code on the next line:
Console.WriteLine($"Sending message: {messageBody}");
To send the message to the topic, insert the following code on the next line:
await sender.SendMessageAsync(message);
Check that your final code resembles the following example:
using System;
using System.Threading.Tasks;
using Azure.Messaging.ServiceBus;
namespace performancemessagesender
{
class Program
{
const string ServiceBusConnectionString = "Endpoint=sb://example.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=xxxxxx";
const string TopicName = "salesperformancemessages";
static void Main(string[] args)
{
Console.WriteLine("Sending a message to the Sales Performance topic...");
SendPerformanceMessageAsync().GetAwaiter().GetResult();
Console.WriteLine("Message was sent successfully.");
}
static async Task SendPerformanceMessageAsync()
{
// By leveraging "await using", the DisposeAsync method will be called automatically once the client variable goes out of scope.
// In more realistic scenarios, you would store off a class reference to the client (rather than to a local variable) so that it can be used throughout your program.
await using var client = new ServiceBusClient(ServiceBusConnectionString);
await using ServiceBusSender sender = client.CreateSender(TopicName);
try
{
string messageBody = "Total sales for Brazil in August: $13m.";
var message = new ServiceBusMessage(messageBody);
Console.WriteLine($"Sending message: {messageBody}");
await sender.SendMessageAsync(message);
}
catch (Exception exception)
{
Console.WriteLine($"{DateTime.Now} :: Exception: {exception.Message}");
}
}
}
}
To save your changes, select Ctrl+S, and then select Ctrl+Q to close the editor.
To run the component that sends a message about a sale, run the following command in Cloud Shell:
dotnet run --project performancemessagesender
As the program executes, watch for notifications in Cloud Shell that indicate that a message is being sent. Each time you run the app, another message is added to the topic, and a copy becomes available for each subscription.
Sending a message to the Sales Performance topic...
Sending message: Total sales for Brazil in August: $13m.
Message was sent successfully.
When you see Message was sent successfully
, run the following command to see how many messages are in the Americas
subscription. Remember to replace <namespace-name> with your Service Bus namespace.
az servicebus topic subscription show \
--resource-group "[sandbox resource group name]" \
--topic-name salesperformancemessages \
--name Americas \
--query messageCount \
--namespace-name <namespace-name>
If you replace Americas
with EuropeAndAsia
and run the command again, you'll see that both subscriptions have the same number of messages.
To create the component that retrieves messages about sales performance, complete these steps:
Run code .
to launch the editor.
In the editor, open performancemessagereceiver/Program.cs and find the following line of code:
const string ServiceBusConnectionString = "";
Between the quotation marks, paste the connection string that you saved in the previous exercise.
To create a Service Bus client, find the MainAsync()
method. Within that method, locate the following line of code:
// Create a Service Bus client that will authenticate using a connection string
Replace that line with this code:
var client = new ServiceBusClient(ServiceBusConnectionString);
To configure message handling options, find the following line of code:
// Create the options to use for configuring the processor
Replace that line with this code:
var processorOptions = new ServiceBusProcessorOptions
{
MaxConcurrentCalls = 1,
AutoCompleteMessages = false
};
To create a processor, find the following line of code:
// Create a processor that we can use to process the messages
Replace that line with this code:
ServiceBusProcessor processor = client.CreateProcessor(TopicName, SubscriptionName, processorOptions);
To configure the handler, find the following line of code:
// Configure the message and error handler to use
Replace that line with this code:
processor.ProcessMessageAsync += MessageHandler;
processor.ProcessErrorAsync += ErrorHandler;
To start processing, find the following line of code:
// Start processing
Replace that line with this code:
await processor.StartProcessingAsync();
Look for the following line of code:
// Since we didn't use the "await using" syntax here, we need to explicitly dispose the processor and client
Replace the line with this code:
await processor.DisposeAsync();
await client.DisposeAsync();
To display incoming messages in the console, find the MessageHandler()
method. You've registered this method to handle incoming messages.
Replace all the code within that method with the following code:
Console.WriteLine($"Received message: SequenceNumber:{args.Message.SequenceNumber} Body:{args.Message.Body}");
To remove the received message from the subscription, on the next line, add the following code:
await args.CompleteMessageAsync(args.Message);
Check that your final code resembles the following example:
using System;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using Azure.Messaging.ServiceBus;
namespace performancemessagereceiver
{
class Program
{
const string ServiceBusConnectionString = "Endpoint=sb://alexgeddyneil.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=xxxxxx";
const string TopicName = "salesperformancemessages";
const string SubscriptionName = "Americas";
static void Main(string[] args)
{
MainAsync().GetAwaiter().GetResult();
}
static async Task MainAsync()
{
var client = new ServiceBusClient(ServiceBusConnectionString);
Console.WriteLine("======================================================");
Console.WriteLine("Press ENTER key to exit after receiving all the messages.");
Console.WriteLine("======================================================");
var processorOptions = new ServiceBusProcessorOptions
{
MaxConcurrentCalls = 1,
AutoCompleteMessages = false
};
ServiceBusProcessor processor = client.CreateProcessor(TopicName, SubscriptionName, processorOptions);
processor.ProcessMessageAsync += MessageHandler;
processor.ProcessErrorAsync += ErrorHandler;
await processor.StartProcessingAsync();
Console.Read();
await processor.DisposeAsync();
await client.DisposeAsync();
}
static async Task MessageHandler(ProcessMessageEventArgs args)
{
Console.WriteLine($"Received message: SequenceNumber:{args.Message.SequenceNumber} Body:{args.Message.Body}");
await args.CompleteMessageAsync(args.Message);
}
static Task ErrorHandler(ProcessErrorEventArgs args)
{
Console.WriteLine($"Message handler encountered an exception {args.Exception}.");
Console.WriteLine("Exception context for troubleshooting:");
Console.WriteLine($"- Endpoint: {args.FullyQualifiedNamespace}");
Console.WriteLine($"- Entity Path: {args.EntityPath}");
Console.WriteLine($"- Executing Action: {args.ErrorSource}");
return Task.CompletedTask;
}
}
}
To save your changes, select Ctrl+S, and then select Ctrl+Q to close the editor.
To run the component that retrieves a message about sales performance for a subscription, run the following command:
dotnet run --project performancemessagereceiver
You'll see output similar to the following example:
Received message: SequenceNumber:1 Body:Total sales for Brazil in August: $13m.
When the program has returned notifications that it's receiving messages, press Enter to stop the app.
Run the following command to confirm that there are no remaining messages in the Americas
subscription. Be sure to replace <namespace-name> with your Service Bus namespace.
az servicebus topic subscription show \
--resource-group "[sandbox resource group name]" \
--topic-name salesperformancemessages \
--name Americas \
--query messageCount \
--namespace-name <namespace-name>
If you replace Americas
with EuropeAndAsia
in this code to see the current message count for the EuropeAndAsia
subscription, you'll see that the message count is 1
. In the preceding code, only Americas
was set to retrieve topic messages, so that message is still waiting for EuropeAndAsia
to retrieve it.
Having an issue? We can help!
Please sign in to use this experience.
Sign in