A simple bookmark manager written in C#
aspnet core 5
sql server
ef core 5
rabbit mq
Adding a bookmark to Bookmark Manager is executed in two steps:
- Saves the provided URL on the DB and sends a message to RabbitMQ
- A consumer process the message fetching the url details from the site response
To make sure that bookmarks saved on te DB are always processed by the consumer it is used the Outbox Pattern.
In this implementation of the outbox pattern messages to be sent to RabbitMQ are saved on the DB and processed by a background service periodically. OutboxSendingService
gets the messages not sent and publish them to the queue updating their status. This way is guaranteed that each message is sent at least one time.
For this reason, to the outbox pattern work is important that the consumer is idempotent. In this example, a separated table is used the store the processed messages. When commiting the transaction afer making changes to other entities the violation of the unique constraint error will occur and none of the change will be stored.
# start dependencies
docker-compose up -d db queue
# see help
dotnet run -- -h
# run api
dotnet run -- api
# run consumer
dotnet run -- bookmark-inserted-consumer
- https://docs.microsoft.com/en-us/aspnet/core/tutorials/first-web-api?view=aspnetcore-5.0&tabs=visual-studio
- https://medium.com/@sergiobarriel/how-to-automatically-validate-a-model-with-mvc-filter-and-fluent-validation-package-ae51098bcf5b
- https://www.rabbitmq.com/dotnet-api-guide.html#connection-and-channel-lifspan
- https://github.com/Tyrrrz/CliFx
- https://devblogs.microsoft.com/aspnet/improvements-in-net-core-3-0-for-troubleshooting-and-monitoring-distributed-apps/
- https://docs.microsoft.com/en-us/aspnet/core/fundamentals/host/hosted-services?view=aspnetcore-5.0&tabs=visual-studio#timed-background-tasks
- https://github.com/davidfowl/AspNetCoreDiagnosticScenarios/blob/master/AsyncGuidance.md#timer-callbacks
- https://altkomsoftware.pl/en/blog/microservices-outbox-rabbitmq/
- https://chrisrichardson.net/post/microservices/patterns/2020/10/16/idempotent-consumer.html
- https://cap.dotnetcore.xyz/user-guide/en/cap/idempotence/
- https://www.thereformedprogrammer.net/entity-framework-core-validating-data-and-catching-sql-errors/
- https://github.com/willianantunes/tutorials/tree/master/2021/XX/ef-core-how-to-handle-migrations-in-production