As the sole purpose of this project is education, the system is intentionally overengineered and heavily relies on async communications.
According to the requirements described above, I have split the solution into four domains - auth, task tracker, accounting and analytics.
Additionally, I created web-ui (Next.js) service for UI and view-service as a backend service for web.
Events are delivered using Kafka, they use strict schemas, which are registered in the schema-registry service.
To ensure event delivery, I use transaction outbox pattern. When a service produces an event, it writes the event data into a separate table and later this table is read by a sidecar service publish-service
and events are sent out in the same order as they were written. See publisher.service.ts
All data objects are streamed to each service via Kafka, i.e. services do not request data from each other directly. Instead, they store streamed data in their respective databases. It reduces coupling and allows services to work independently.