Series context (Part 4 of 8): This article assumes familiarity with ACID transactions and database concurrency. Understanding why consistency guarantees are hard at the database layer is essential context before introducing distributed patterns here.

Why Microservices in Banking?

Microservices in banking is the architectural pattern where a core banking system is broken into independently deployable, domain-owned services (CIF, Payments, Lending, Notifications) connected by an event bus instead of direct database calls. This replaces monolithic systems like T24 or Flexcube — where a single change to the Payments module requires redeploying the entire application and risks taking down unrelated services.

  • High-risk deployments: Modifying a small module requires redeploying the entire system. A patch to the Payments module can take down CIF.
  • Inefficient scaling: You cannot scale just the Payments module during peak loads without scaling everything else — including parts that don’t need more capacity.
  • Technology lock-in: Bound to a single programming language and database. Adding a modern ML risk engine becomes an 18-month integration project.

The current trend is transitioning to Headless Core Banking — decoupling the domain logic from the delivery channels (Mobile App, Internet Banking, ATM) using a banking microservices architecture.


Overall Architecture

graph TD
    subgraph CHANNELS
        MA[Mobile App]
        IB[Internet Banking]
        ATM[ATM / POS]
    end
    
    API[API Gateway: Auth, Rate Limit, Routing]
    MA --> API
    IB --> API
    ATM --> API
    
    subgraph CoreServices[Core Services]
        CIF[CIF Service: Customer]
        ACC[Account Service: CASA, GL]
        PAY[Payment Service: Transfers, Fees]
    end
    
    API --> CIF
    API --> ACC
    API --> PAY
    
    BUS[Message Broker: Kafka / Redis / Dapr]
    
    CIF --> BUS
    ACC --> BUS
    PAY --> BUS
    
    subgraph AsyncServices[Asynchronous Services]
        LOAN[Loan Service: Lending]
        NOTIF[Notification Service: SMS, Push, Email]
        REP[Reporting Service: CQRS Read Side]
    end
    
    BUS --> LOAN
    BUS --> NOTIF
    BUS --> REP

Pattern 1: Event Sourcing for the Ledger

In traditional architectures, we store the current state. In Event Sourcing, we store a sequence of immutable events that produce that state.

Why does Event Sourcing fit Core Banking?

The ledger is already essentially Event Sourcing — every entry is an immutable event. The current balance is simply the result of replaying all entries from the beginning.

// Events in the Account domain
type AccountOpened struct {
    AccountID    string
    CIFNumber    string
    Currency     string
    OpenedAt     time.Time
}

type MoneyDeposited struct {
    AccountID     string
    Amount        int64
    TransactionID string
    OccurredAt    time.Time
}

type MoneyWithdrawn struct {
    AccountID     string
    Amount        int64
    TransactionID string
    OccurredAt    time.Time
}

// Calculate balance by replaying events
func calculateBalance(events []Event) int64 {
    var balance int64
    for _, event := range events {
        switch e := event.(type) {
        case MoneyDeposited:
            balance += e.Amount
        case MoneyWithdrawn:
            balance -= e.Amount
        }
    }
    return balance
}

Pattern 2: CQRS — Command Query Responsibility Segregation

Core Banking has a unique characteristic: writes must be exceptionally robust (ACID) but reads need to be lightning fast (dashboards, reports). CQRS completely separates these two flows:

WRITE SIDE (Command)                READ SIDE (Query)
────────────────────────            ──────────────────────────
POST /transfers            →        Materialized Views
POST /accounts             →        Elasticsearch Index
PUT /loans/repay           →        Redis Cache

↓ Event Published ↓                ↑ Subscribe & Update ↑
         └──────────────────────────┘
              (Event Bus / Kafka)

Real-world Example:

  • Write Side: Processes transfers using PostgreSQL with full ACID compliance, guaranteeing money isn’t lost.
  • Read Side: Dashboards display transaction history from Elasticsearch — ultra-fast queries, full-text search, and multi-condition filtering.

Pattern 3: Saga — Distributed Transactions Across Services

When a cross-bank transfer requires coordinating 3 services: Account Service (deduct money), Payment Service (send to clearing house), and Notification Service (send SMS), how do you ensure integrity?

Choreography Saga (Event-Driven)

Account Service                Payment Service           Notification Service
      │                               │                          │
      │── TransferInitiated ──────────▶│                          │
      │                               │── PaymentSubmitted ──────▶│
      │                               │                          │── SMS Sent
      │◀── PaymentCompleted ──────────│                          │
      │                               │                          │
   (release hold)                                            (done)

If Payment fails:
      │◀── PaymentFailed ─────────────│
      │                               │
   (cancel hold, refund)

Outbox Pattern — Guaranteeing Events are Never Lost

Problem: What if a service successfully commits to the database but fails to publish the event to Kafka?

Solution: Write the event to the database within the same transaction, then have a separate worker read and publish it to Kafka.

-- Outbox table: written in the same transaction as business data
CREATE TABLE outbox_events (
    id          UUID PRIMARY KEY DEFAULT gen_random_uuid(),
    topic       VARCHAR(100) NOT NULL,  -- 'account.transfer.completed'
    payload     JSONB        NOT NULL,
    status      VARCHAR(20)  NOT NULL DEFAULT 'PENDING',
    created_at  TIMESTAMPTZ  NOT NULL DEFAULT NOW(),
    published_at TIMESTAMPTZ
);

-- Inside the same Database Transaction:
-- 1. Update account balance
-- 2. Write ledger entries  
-- 3. INSERT into outbox_events

-- Separate worker running periodically:
-- SELECT * FROM outbox_events WHERE status = 'PENDING'
-- → Publish to Kafka
-- → UPDATE status = 'PUBLISHED'

API Design for Financial Transactions

Design Principles

  1. Stateless APIs: Every request must contain all necessary information.
  2. Mandatory Idempotency headers for all state-changing APIs.
  3. Strict separation between Request (commands) and Status (polling).
POST /v1/transfers                    → Initiate transfer command
  Header: Idempotency-Key: <uuid>
  Body: { from, to, amount, currency }
  Response: { transfer_id, status: "PROCESSING" }

GET  /v1/transfers/{transfer_id}      → Check result
  Response: { status: "COMPLETED" | "FAILED", ... }

Never design a transfer API as a synchronous block because processing through a central clearing network (like SWIFT) can take anywhere from seconds to minutes.


Technical Stack Selection

LayerPopular ChoicesReason
Service FrameworkGo (Kratos, Fiber), Java (Spring Boot)High performance, type-safe
Primary DatabasePostgreSQLStrong ACID, flexible JSONB
CacheRedisBalances, sessions, rate limiting
Event BusApache Kafka, Dapr PubSubDurable, ordered, replayable
Service MeshIstio, DaprmTLS, circuit breaking
OrchestrationKubernetesAuto-scaling, self-healing

Frequently Asked Questions (FAQ) about Core Banking Microservices

How do banking microservices differ from standard e-commerce microservices?

Data Integrity and ACID transactions are critical. In e-commerce, losing a click event is acceptable, but in banking, losing a money transfer event is catastrophic. Therefore, banks use the Outbox Pattern, Event Sourcing, and Choreography Sagas instead of standard orchestrations to ensure absolute consistency.

How do you handle data joins across services?

In a Microservices architecture, each service has its own database (Database per service). Direct SQL JOINs are not possible. Instead, Core Banking applies CQRS (Command Query Responsibility Segregation) to build a Read Database (like Elasticsearch) that aggregates data from Message Broker events for high-speed queries and reporting.

Does an Event-Driven Architecture make the system slower?

No, it actually massively increases throughput. Cross-bank transfers are not processed synchronously blocking the main thread. Instead, they are pushed to a Message Broker (Asynchronous). The initial response is “PROCESSING”, and the final “COMPLETED” status is updated once the process is done, ensuring the API Gateway never bottlenecks even with thousands of TPS.

References & Further Reading

🔗 Previous Step: Explore the foundational database layer in Part 3 — Database Design for Financial Transactions (ACID & Concurrency).

🔗 Next Step: Now that you understand banking microservices architecture and its event-driven patterns, see how these services communicate with the outside world through international financial standards. Continue reading Part 5 — International Integration Standards: ISO 8583 & ISO 20022.

🔗 Deep Dive: For a complete engineering guide to the full composable banking stack — ledger concurrency patterns, Strangler Fig migrations, RFC 8705 mTLS, and the next-gen vendor landscape — see Composable Banking Architecture: From Monolith to Modular Core.