Why this matters
For product and operations teams, modeling each balance as its own account means every segregation rule — a blocked outflow, a benefit-only spend, a promotional balance with an expiry — is enforced by the Ledger, not buried in application logic. Each account carries its own statement and reconciliation trail. For engineering teams, the customer’s external addresses (core banking numbers, payment-rail identifiers) resolve to a specific account before a transaction is posted. There’s no guessing which balance an incoming event belongs to, and no separate balance store to keep in sync with the Ledger.
| One account with labels | Many accounts per customer |
|---|---|
| Balance “types” live in metadata or app code | Each balance is its own Account with its own ledger and statement |
| Blocking or restricting funds requires custom logic | Account Type and route rules enforce restrictions at the Ledger |
| Statements must be filtered and reassembled per balance | Each account produces a clean, independent statement |
| External identifiers all point to the same balance | Each external identifier resolves to a specific account |
| Reconciliation mixes unrelated movements | Reconciliation is separated by account by design |
The reference architecture
The model is one owner, N accounts, N external identifiers:
- One owner — the customer, identified by a document (CPF, CNPJ, tax ID). The owner represents who holds the relationship. It does not carry a balance and does not decide transaction routing.
- N accounts — each account is a self-contained accounting position, with its own balance, ledger, statement, and rules.
- N external identifiers — the addresses other systems (a core banking platform, a payment rail) use to reach a specific account. Each identifier resolves to exactly one account.

Each external identifier is not just a nickname for one shared balance. When balance, ledger, statement, or operational rule differs by destination, each identifier must point to a distinct account.
How Midaz maps the model
Every part of this architecture maps to a native Midaz entity — you don’t need to build a separate balance store or invent an accounting layer.
| Concept | Midaz entity | What it does |
|---|---|---|
| The owner (customer) | Holder | Identity behind the accounts (NATURAL_PERSON or LEGAL_PERSON), keyed by document. Holds no balance. |
| Each balance position | Account | Source of truth for balance, entries, and statement. |
| The nature of each balance | Account Type | Classifies an account (main, benefit, blocked) and enables route validation. |
| All accounts of one customer | Portfolio | Groups a customer’s accounts to view the total relationship. |
| External address → account | Account alias + entityId | The alias is how transactions address an account; entityId links it to an external system’s identifier. |
| Banking and regulatory context | Alias Account (CRM) | Attaches branch, account number, and regulatory fields, linking a Holder to a specific account. |
Prerequisites
This example assumes a running Midaz environment with the following in place:
| Requirement | Details |
|---|---|
| Midaz (v3.x.x+) | Core Ledger with an Organization and Ledger already created |
| A registered asset | BRL registered as the operating asset in the Ledger |
| Account Type validation | Enabled per-Ledger so each account’s nature is enforced (see Step 1) |
| CRM (optional) | Running if you want to attach identity, banking, and regulatory context |
Values in Midaz are represented in the smallest unit of the currency. For BRL,
15000 means R$ 150.00 (centavos).Building three accounts for one customer
The customer with document
12345678900 needs three accounts: a main account for ordinary movement, a benefit account governed by product rules, and a blocked account that accepts inflows but restricts outflows.
Enable Account Type validation
Turn on validation so every account must declare a registered type. This is what lets the Ledger enforce each account’s nature.Settings take effect immediately — no redeployment needed.
Register the Account Types
Create one Account Type per balance nature. The Repeat for
keyValue is what each account’s type field must match.benefit_account (“Movement governed by product rules”) and restricted_account — the blocked account, where inflows are allowed but outflows are conditioned or blocked.Create the three accounts
Each account is linked to the Create the benefit account with
BRL asset, declares its type, and carries an alias (its in-Ledger address) and an entityId (its identifier in your external system).alias @cust_12345678900_benefit, entityId 0001/88888-2, type benefit_account; and the blocked account with alias @cust_12345678900_blocked, entityId 0002/77777-0, type restricted_account.Register the customer as a Holder
Create one Holder for the customer. The same Holder will own all three accounts, keeping identity centralized.Save the returned
CRM runs as a separate service, and every request requires the
X-Organization-Id header. See Getting started with CRM for service setup and the full schema.holderId — you’ll use it in the next step.Link each account to the Holder
Create an Alias Account per ledger account to attach banking and regulatory context. This is what powers CRM-driven features and keeps customer-facing details separate from the Ledger.Notice how the main account’s
entityId (0001/12345-1) decomposes into the branch (0001) and account (12345) you record here — that’s the external address resolving to one specific account. Repeat for the benefit and blocked accounts, pointing accountId at each one.| Owner | External identifier | Midaz account | Use | Treatment |
|---|---|---|---|---|
12345678900 | 0001/12345-1 | @cust_..._main | Main account | Free for ordinary movement |
12345678900 | 0001/88888-2 | @cust_..._benefit | Benefit account | Movement governed by product rules |
12345678900 | 0002/77777-0 | @cust_..._blocked | Court-ordered / blocked | Inflow allowed, outflow conditioned or blocked |
The transaction boundary
When an external event arrives, the resolution happens before Midaz is called. Keeping each layer in its lane is what preserves accounting clarity.
Middleware resolves the identifier
The middleware looks up the external identifier and resolves it to the correct Midaz account alias, applying status validation (active, blocked, closed).
| Layer | Responsibility | Should not do |
|---|---|---|
| CRM / registration | Hold the commercial and identity view of the customer, including the link between document and relationship. | Transaction routing, destination decisions, or settlement rules. |
| Middleware | Resolve external identifiers to a Midaz account before the transaction, and apply status validation. | Invent balances, duplicate accounting, or depend on the CRM in real time for routing. |
| Midaz | Record accounts, balances, entries, ledgers, and statements as the financial source of truth. | Know external-rail details beyond the identifiers needed for integration. |
What this unlocks
- Real segregation — each balance has its own ledger and statement, so a blocked balance can never be spent through the main account by accident.
- Unambiguous routing — every external event has a single, well-defined destination account.
- Centralized identity — one Holder owns many accounts; identity and contact data live in one place while balances stay separate.
- Native, not bolted-on — accounts, types, aliases, and
entityIdare platform primitives, so there’s no parallel balance store to reconcile against the Ledger.
What you need to get started
| Requirement | Details |
|---|---|
| Midaz (v3.x.x+) | Organization, Ledger, and a registered asset |
| Account Type validation | Enabled per-Ledger via the Ledger Settings API |
| Account Types | One per balance nature (main, benefit, blocked, …) |
| Accounts | One per balance, each with an alias and an entityId |
| CRM (optional) | A Holder per customer plus an Alias Account per ledger account |
Next steps
Accounts
The core financial unit — aliases,
entityId, and external accounts.Account Types
Classify accounts and enforce their nature with route validation.
Portfolios
Group a customer’s accounts to view the total relationship.
CRM: Holders & Alias Accounts
Centralize identity and attach banking and regulatory context.

