A billing system should be able to
- Create an invoice with items that must have a description, a quantity, and an amount
- Pay an invoice by charging with a payment method previously registered
- Contest/Cancel/Refund an invoice
- Change Payment Gateway dynamically
- Register a payment method like credit card or pix and make one of them default
- List all transactions made by a user with filter as period, amount, items, and so on
- Docker
- Docker Compose
make dev # create database and run migrations and attach a bash on container
cargo run # download all dependencies and start http server (only works on make dev)
cargo run --bin kafka # download all dependencies and start kafka consumer server (only works on make dev)
cargo run --bin cronjob # download all dependencies and start cronjob (only works on make dev)
cargo test # run integration and unit tests (only works on make dev)
cargo clippy # run lint (don't work on make dev)
cargo fmt # run formatter (don't work on make dev)
cargo audit # run audit on packages (don't work on make dev), see https://crates.io/crates/cargo-audit
nix build
nix develop # enter dev shell
- Creating an invoice:
POST /invoice
-> draft - Finish building invoice and try to charge:
POST /charge
-> pending -> Go to Kafka - Receive a TOPIC on Kafka to try to charge:
TOPIC charge_invoice
-> Gateway success -> paidTOPIC charge_invoice
-> Gateway fail -> charged_with_error
- A cronjob eventually gets all invoices in charged_with_error status and tries to charge again
- A user can:
- Cancel:
PATCH /invoice
-> Gateway success -> canceled - Refund before 7 days:
PATCH /invoice
-> Gateway success -> refunded - Contest by fraud:
PATCH /invoice
-> Gateway success -> in_protest -> chargeback
- Cancel:
All flows that involve Gateway that change of status is made by a webhook.
An invoice is a list of items containing a description, amount, and quantity. A charge is the acting of register a invoice in a payment gateway. We currently track a lifetime of an invoice by using the column status. A payment method is like credit-card or pix key.
There's no precise answer to this, but could be: a cost for each transaction, temporary unavailability, payment method issue (pix only in brazil), batch reconciliation distribution, etc.
- Add GraphQL
- Add Mutating Test
- Add load testing with k6
- Add preferable system for payment gateway as Vindi, Iugu, PagSeguro, or Paypal
- Add a webhook for each payment-gateway