This sample shows using Dapr with ASP.NET Core routing. This application is a simple and not-so-secure banking application. The application uses the Dapr state-store for its data storage.
It exposes the following endpoints over HTTP:
- GET
/{id}
: Get the balance for the account specified byid
- POST
/deposit
: Accepts a JSON payload to deposit money to an account - POST
/multideposit
: Accepts a JSON payload to deposit money multiple times to a bulk subscribed topic - POST
/withdraw
: Accepts a JSON payload to withdraw money from an account
The application also registers for pub/sub with the deposit
, multideposit
, and withdraw
topics.
To run the sample locally run this command in this project root directory:
dapr run --app-id routing --app-port 5000 -- dotnet run
The application will listen on port 5000 for HTTP.
Deposit Money
On Linux, MacOS:
curl -X POST http://127.0.0.1:5000/deposit \
-H 'Content-Type: application/json' \
-d '{ "id": "17", "amount": 12 }'
On Windows:
curl -X POST http://127.0.0.1:5000/deposit -H "Content-Type: application/json" -d "{ \"id\": \"17\", \"amount\": 12 }"
Or, we can also do this using the Visual Studio Code Rest Client Plugin
POST http://127.0.0.1:5000/deposit
Content-Type: application/json
{ "id": "17", "amount": 12 }
Output:
{"id":"17","balance":12}
Deposit Money multiple times to a bulk subscribed topic
On Linux, MacOS:
curl -X POST http://127.0.0.1:5000/multideposit \
-H 'Content-Type: application/json' \
-d '{
"entries":[
{
"entryId":"653dd9f5-f375-499b-8b2a-c4599bbd36b0",
"event":{
"data":{
"amount":10,
"id":"17"
},
"datacontenttype":"application/json",
"id":"DaprClient",
"pubsubname":"pubsub",
"source":"Dapr",
"specversion":"1.0",
"topic":"multideposit",
"type":"com.dapr.event.sent"
},
"metadata":null,
"contentType":"application/cloudevents+json"
},
{
"entryId":"7ea8191e-1e62-46d0-9ba8-ff6e571351cc",
"event":{
"data":{
"amount":20,
"id":"17"
},
"datacontenttype":"application/json",
"id":"DaprClient",
"pubsubname":"pubsub",
"source":"Dapr",
"specversion":"1.0",
"topic":"multideposit",
"type":"com.dapr.event.sent"
},
"metadata":null,
"contentType":"application/cloudevents+json"
}
],
"id":"fa68c580-1b96-40d3-aa2c-04bab05e954e",
"metadata":{
"pubsubName":"pubsub"
},
"pubsubname":"pubsub",
"topic":"multideposit",
"type":"com.dapr.event.sent.bulk"
}'
Output:
{
"statuses":[
{
"entryId":"653dd9f5-f375-499b-8b2a-c4599bbd36b0",
"status":"SUCCESS"
},
{
"entryId":"7ea8191e-1e62-46d0-9ba8-ff6e571351cc",
"status":"SUCCESS"
}
]
}
Withdraw Money On Linux, MacOS:
curl -X POST http://127.0.0.1:5000/withdraw \
-H 'Content-Type: application/json' \
-d '{ "id": "17", "amount": 10 }'
On Windows:
curl -X POST http://127.0.0.1:5000/withdraw -H "Content-Type: application/json" -d "{ \"id\": \"17\", \"amount\": 10 }"
or using the Visual Studio Code Rest Client Plugin
POST http://127.0.0.1:5000/withdraw
Content-Type: application/json
{ "id": "17", "amount": 5 }
Output:
{"id":"17","balance":2}
Get Balance
curl http://127.0.0.1:5000/17
or using the Visual Studio Code Rest Client Plugin
GET http://127.0.0.1:5000/17
Output:
{"id":"17","balance":2}
Withdraw Money (pubsub) Publish events using Dapr cli: On Linux, MacOS:
dapr publish --pubsub pubsub --publish-app-id routing -t withdraw -d '{"id": "17", "amount": 15 }'
On Windows:
dapr publish --pubsub pubsub --publish-app-id routing -t withdraw -d "{\"id\": \"17\", \"amount\": 15 }"
Deposit Money (pubsub) Publish events using Dapr cli: On Linux, MacOS:
dapr publish --pubsub pubsub --publish-app-id routing -t deposit -d '{"id": "17", "amount": 15 }'
On Windows:
dapr publish --pubsub pubsub --publish-app-id routing -t deposit -d "{\"id\": \"17\", \"amount\": 15 }"
Dead Letter Topic example (pubsub) Publish an event using the Dapr cli with an incorrect input, i.e. negative amount:
Deposit: On Linux, MacOS:
dapr publish --pubsub pubsub --publish-app-id routing -t deposit -d '{"id": "17", "amount": -15 }'
On Windows:
dapr publish --pubsub pubsub --publish-app-id routing -t deposit -d "{\"id\": \"17\", \"amount\": -15 }"
Withdraw: On Linux, MacOS:
dapr publish --pubsub pubsub --publish-app-id routing -t withdraw -d '{"id": "17", "amount": -15 }'
On Windows:
dapr publish --pubsub pubsub --publish-app-id routing -t withdraw -d "{\"id\": \"17\", \"amount\": -15 }"
First a message is sent from a publisher on a deposit
or withdraw
topic. Dapr receives the message on behalf of a subscriber application, however the deposit
or withdraw
topic message fails to be delivered to the /deposit
or /withdraw
endpoint on the application, even after retries. As a result of the failure to deliver, the message is forwarded to the amountDeadLetterTopic
topic which delivers this to the /deadLetterTopicRoute
endpoint.
All of the interesting code in this sample is in Startup.cs
public void ConfigureServices(IServiceCollection services)
{
services.AddDaprClient(builder =>
builder.UseJsonSerializationOptions(
new JsonSerializerOptions()
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
PropertyNameCaseInsensitive = true,
}));
...
}
AddDaprClient()
registers the DaprClient
service with the dependency injection container (using the sepcified DaprClientBuilder
for settings options). This service can be used to interact with the dapr runtime (e.g. invoke services, publish messages, interact with a state-store, ...).
app.UseCloudEvents();
UseCloudEvents()
registers the Cloud Events middleware in the request processing pipeline. This middleware will unwrap requests with Content-Type application/cloudevents+json
so that application code can access the event payload in the request body directly. This is recommended when using pub/sub unless you have a need to process the event metadata yourself.
app.UseEndpoints(endpoints =>
{
endpoints.MapSubscribeHandler();
endpoints.MapGet("{id}", Balance);
endpoints.MapPost("deposit", Deposit).WithTopic(PubsubName, "deposit");
endpoints.MapPost("multideposit", MultiDeposit).WithTopic(multiDepositTopicOptions).WithBulkSubscribe(bulkSubscribeTopicOptions);
endpoints.MapPost("withdraw", Withdraw).WithTopic(PubsubName, "withdraw");
});
MapSubscribeHandler()
registers an endpoint that will be called by the Dapr runtime to register for pub/sub topics. This is is not needed unless using pub/sub.
MapGet(...)
and MapPost(...)
are provided by ASP.NET Core routing - these are used to setup endpoints to handle HTTP requests.
WithTopic(...)
associates an endpoint with a pub/sub topic.
var depositTopicOptions = new TopicOptions();
depositTopicOptions.PubsubName = PubsubName;
depositTopicOptions.Name = "deposit";
depositTopicOptions.DeadLetterTopic = "amountDeadLetterTopic";
var withdrawTopicOptions = new TopicOptions();
withdrawTopicOptions.PubsubName = PubsubName;
withdrawTopicOptions.Name = "withdraw";
withdrawTopicOptions.DeadLetterTopic = "amountDeadLetterTopic";
var multiDepositTopicOptions = new TopicOptions
{ PubsubName = PubsubName, Name = "multideposit" };
var bulkSubscribeTopicOptions = new BulkSubscribeTopicOptions
{
TopicName = "multideposit", MaxMessagesCount = 250, MaxAwaitDurationMs = 1000
};
WithTopic(...)
now takes the TopicOptions(..)
instance that defines configurations for the subscribe endpoint.
WithBulkSubscribe(...)
now takes the BulkSubscribeTopicOptions(..)
instance that defines configurations for the bulk subscribe endpoint.
async Task Balance(HttpContext context)
{
var client = context.RequestServices.GetRequiredService<StateClient>();
var id = (string)context.Request.RouteValues["id"];
var account = await client.GetStateAsync<Account>(id);
if (account == null)
{
context.Response.StatusCode = 404;
return;
}
context.Response.ContentType = "application/json";
await JsonSerializer.SerializeAsync(context.Response.Body, account, serializerOptions);
}
Here GetRequiredService<StateClient>()
is used to retrieve the StateClient
from the service provider.
client.GetStateAsync<Account>(id)
is used to retrieve an Account
object from that state-store using the key in the variable id
. The Account
object stored in the state-store as JSON. If no entry is found for the specified key, then null
will be returned.
await client.SaveStateAsync(transaction.Id, account);
SaveStateAsync(...)
is used to save data to the state-store.