28 KiB
Design personal budget tracking app (Mint.com)
Note: This document links directly to relevant areas found in the system design topics to avoid duplication. Refer to the linked content for general talking points, trade-offs, and alternatives.
Step 1: Investigate the problem, use cases and constraints and establish design scope
Gather main functional requirements and scope the problem. Ask questions to clarify use cases and constraints. Discuss assumptions.
Adding clarifying questions is the first step in the process. Remember your goal is to understand the problem and establish the design scope.
What questions should you ask to clarify the problem?
Here is an example of the dialog you could have with the interviewer:
Example dialog with your interviewer:
Interviewer: Design personal budget tracking app (Mint.com).
Candidate: To clarify and understand the scope, may I start with a few quick questions?
Interviewer: Yes, please.
Candidate: What are the main features, users, and use cases of the system?
Interviewer: It aggregates users’ financial accounts, categorizes transactions, and helps them budget.
Candidate: Got it. How do we get the data from the financial accounts?
Interviewer: connects to a financial account.
Candidate: Should we focus on real-time updates or would a daily refresh suffice?
Interviewer: Daily is fine.
Candidate: How should categorization work?
Interviewer: Auto-categorize based on merchant, but allow manual overrides. No re-categorization once set.
Candidate: Understood. What budget features are needed?
Interviewer: Recommend budgets by category, allow manual budgets, and notify when users approach or exceed them.
Candidate: Any non-functional requirements?
Interviewer: High availability is a must; budget alerts don’t need millisecond latency.
Candidate: Great. And scale?
Interviewer: Target ~10 M users, ~30 M linked accounts, ~5 B transactions/mo, with ~500 M reads/mo.
Candidate: Thanks — that's clear and helps scope my design. Let's me summarize the scope as i understood it and the assumptions to make sure we are on the same page.
Interviewer: Ok.
Candidate: ok, here a reflection of what I understood from the requirements, I will write it down here
Use cases
- User connects to one or more financial accounts (bank, credit card, investment).
Background tasks
- Service extracts transactions from the account
- Daily batch updates for active users (last 30 days).
- Categorizes transactions
- Auto-categorization by merchant
- Allows manual category override by the user
- No automatic re-categorization after manual change.
- Analyzes monthly spending, by category
- Service Analyze spending & suggest budgets
- Compute monthly spend per category.
- Recommend a budget per category.
- Allow users to set or adjust budgets manually.
- Send notifications when spending approaches/exceeds budget.
Non functional requirements
- Service has high availability
Out of scope
- Service performs additional logging and analytics
- All authenticated users features
- Any other requirements
Other considerations we need to think about (Later on)
- System Quality Attributes (Reliability, Scalability, Security & Privacy, Operational Aspects, etc.)
- System life cycle (Deployment, Monitoring, Logging, Automation, CI/CD, etc.)
Constraints & assumptions
State assumptions
- Traffic is not evenly distributed
- Automatic daily update of accounts applies only to users active in the past 30 days
- Adding or removing financial accounts is relatively rare
- Budget notifications don't need to be instant
- 10 million users
- 10 budget categories per user = 100 million budget items
- Example categories:
- Housing = $1,000
- Food = $200
- Gas = $100
- Sellers are used to determine transaction category
- 50,000 sellers
- 30 million financial accounts
- 5 billion transactions per month
- 500 million read requests per month
- 10:1 write to read ratio
- Write-heavy, users make transactions daily, but few visit the site daily
Back-of-the-envelope usage calculations
Clarify with your interviewer if you should run back-of-the-envelope usage calculations. If you need to calculate usage, here is calculation example:
Size per transaction record:
user_id 8 bytes
timestamp 5 bytes
merchant_id 32 bytes
amount 5 bytes
category code 2 bytes
--------------------------
≈ 52 bytes/transaction
Monthly data volume:
52 bytes × 5 000 000 000 txns ≈ 260 GB/month
→ ~9 TB over 3 years
Request rates:
5 000 000 000 transactions / (2.5 million sec/mo) ≈ 2 000 writes/sec
500 000 000 reads / (2.5 million sec/mo) ≈ 200 reads/sec
- 260 GB of new transaction content per month
- 50 bytes per transaction * 5 billion transactions per month
- 9 TB of new transaction content in 3 years
- Assume most are new transactions instead of updates to existing ones
- 2,000 transactions per second on average
- 200 read requests per second on average
Handy conversion guide:
- 2.5 million seconds per month
- 1 request per second = 2.5 million requests per month
- 40 requests per second = 100 million requests per month
- 400 requests per second = 1 billion requests per month
Step 2: Create a high level design & Get buy-in
Outline a high level design with all important components.
%%{init: { "flowchart": { "htmlLabels": true } }}%%
flowchart TB
%% Client Layer
subgraph Client["**Client**"]
direction TB
WebClient[Web Client]
MobileClient[Mobile Client]
end
%% Web Server Layer
subgraph WebServer["**Web Server - (Reverse Proxy)**"]
direction LR
AccountsAPI[Accounts API]
Queue[Queue]
TransactionExtractionService[Transaction Extraction Service]
CategoryService[Category Service]
BudgetService[Budget Service]
NotificationService[Notification Service]
AccountsAPI --> Queue
Queue --> TransactionExtractionService
TransactionExtractionService --> CategoryService
TransactionExtractionService --> BudgetService
TransactionExtractionService --> NotificationService
end
%% Storage Layer
subgraph Storage["**Storage**"]
direction LR
ObjectStore[(Object Store)]
SQLDB[(SQL Database)]
end
%% Data Flow
Client --> WebServer
WebServer --> Storage
AccountsAPI --> SQLDB
TransactionExtractionService --> SQLDB
TransactionExtractionService --> ObjectStore
%% Styling Nodes
style WebClient fill:#FFCCCC,stroke:#CC0000,stroke-width:2px,rx:6,ry:6
style MobileClient fill:#FFD580,stroke:#AA6600,stroke-width:2px,rx:6,ry:6
style AccountsAPI fill:#CCE5FF,stroke:#004085,stroke-width:2px,rx:6,ry:6
style TransactionExtractionService fill:#CCE5FF,stroke:#004085,stroke-width:2px,rx:6,ry:6
style Queue fill:#D4EDDA,stroke:#155724,stroke-width:2px,rx:6,ry:6
style SQLDB fill:#E2E3E5,stroke:#6C757D,stroke-width:2px,rx:6,ry:6
style ObjectStore fill:#E2E3E5,stroke:#6C757D,stroke-width:2px,rx:6,ry:6
Get buy-in
✅ Why This Breakdown?
Rather than diving into implementation, this diagram tells a story:
The microservice breakdown is driven by Separation of Concerns:
- Single-purpose services: Extraction, Categorization, Budget and Notification each handle only one domain, which simplifies testing, deployment and independent versioning.
- Asynchronous decoupling: We buffer all raw ingestion through a message queue so that spikes or transient failures don’t block users. (The diagram shows the main queue; internal queues between downstream steps are omitted for clarity.)
- Optimized storage: Raw transaction dumps live in an object store, while structured metadata resides in SQL—letting us choose the right storage for each access pattern and consistency requirement.
- Scalable: Services are stateless and can be scaled or replaced independently.
You should ask for a feedback after you present the diagram, and get buy-in and some initial ideas about areas to dive into, based on the feedback.
Step 3: Design core components
Dive into details for each core component.
Use case: User connects to a financial account
We could store info on the 10 million users in a relational database. We should discuss the use cases and tradeoffs between choosing SQL or NoSQL.
- The Client sends a request to the Web Server, running as a reverse proxy
- The Web Server forwards the request to the Accounts API server
- The Accounts API server updates the SQL Database
accounts
table with the newly entered account info
Clarify with your interviewer the expected amount, style, and purpose of the code you should write.
The accounts
table could have the following structure:
id int NOT NULL AUTO_INCREMENT
created_at datetime NOT NULL
last_update datetime NOT NULL
account_url varchar(255) NOT NULL
account_login varchar(32) NOT NULL
account_password_hash char(64) NOT NULL
user_id int NOT NULL
PRIMARY KEY(id)
FOREIGN KEY(user_id) REFERENCES users(id)
We'll create an index on id
, user_id
, last_update
and created_at
to speed up lookups. Since indexes are typically implemented with B-trees, index lookup is O(log n) instead of O(n). Frequently accessed indexes (like by last_update
timestamps) are often cached automatically in RAM by the database’s internal cache and since the indexes are smaller, they are likely to stay in memory. Reading 1 MB sequentially from memory takes about 250 microseconds, while reading from SSD takes 4x and from disk takes 80x longer.1
REST API
We'll use a public REST API:
Connect to a financial account
$ curl -X POST --data '{ "user_id": "foo", "account_url": "bar", \
"account_login": "baz", "account_password": "qux" }' \
https://mint.com/api/v1/account
Response:
{
"user_id": "foo",
"account_id": "bar",
"action": "connect",
"status": "success"
}
Next, the service extracts transactions from the account.
Use case: Service extracts transactions from the account
We'll want to extract information from an account in these cases:
- The user first links the account
- The user manually refreshes the account
- Automatically each day for users who have been active in the past 30 days
those are the cases for the transaction extraction service.
So each of these cases will trigger
a transaction extraction service
request.
Data flow - user manually refreshes the account:
- The Client sends a request to the Web Server
- The Web Server forwards the request to the Accounts API server
- The Accounts API server places a job on a Queue such as Amazon SQS or RabbitMQ
- Extracting transactions could take awhile, we'd probably want to do this asynchronously with a queue, although this introduces additional complexity
- The Transaction Extraction Service does the following:
- Pulls from the Queue and extracts transactions according to job (e.g. for the given account from the financial institution)
- Stores the results as raw log or json files in the Object Store
- Uses the Category Service asynchronously to categorize the transactions
- Uses the Budget Service asynchronously to calculate aggregate monthly spending by category
- The Budget Service uses the Notification Service asynchronously to let users know if they are nearing or have exceeded their budget
- Updates the SQL Database
transactions
table with categorized transactions - Updates the SQL Database
monthly_spending
table with aggregate monthly spending by category - Notifies the user the transactions have completed through the Notification Service:
- Uses a Queue (not pictured) to asynchronously send out notifications
The transactions
table could have the following structure:
id int NOT NULL AUTO_INCREMENT
created_at datetime NOT NULL
seller varchar(32) NOT NULL
amount decimal NOT NULL
user_id int NOT NULL
PRIMARY KEY(id)
FOREIGN KEY(user_id) REFERENCES users(id)
We'll create an index on id
, user_id
, and created_at
.
The monthly_spending
table could have the following structure:
id int NOT NULL AUTO_INCREMENT
month_year date NOT NULL
category varchar(32)
amount decimal NOT NULL
user_id int NOT NULL
PRIMARY KEY(id)
FOREIGN KEY(user_id) REFERENCES users(id)
We'll create an index on id
and user_id
.
Category service
For the Category Service, we can seed a seller-to-category dictionary with the most popular sellers. If we estimate 50,000 sellers and estimate each entry to take less than 255 bytes, the dictionary would only take about 12 MB of memory.
Clarify with your interviewer the expected amount, style, and purpose of the code you should write.
class DefaultCategories(Enum):
HOUSING = 0
FOOD = 1
GAS = 2
SHOPPING = 3
...
seller_category_map = {}
seller_category_map['Exxon'] = DefaultCategories.GAS
seller_category_map['Target'] = DefaultCategories.SHOPPING
...
For sellers not initially seeded in the map, we could use a crowdsourcing effort by evaluating the manual category overrides our users provide.
We could use a heap to quickly lookup the top manual override per seller in O(1) time.
Here we actually want the most-popular category override for a given seller (i.e. the one with the highest user-vote count). However, Python’s heapq only provides a min-heap, so we store counts as negative numbers. Then the “minimum” of those negatives is the largest positive count. Heap queues are not designed to handle multiple threads writing the data at the same time, however since writes are rare and missing one count might be acceptable we can use it for a starter.
Generally, a heap (priority queue) shines when you need to:
- Incrementally insert items (e.g. new user overrides)
- Quickly retrieve the current top-priority element (peek or pop)
- Maintain the structure under continuous updates
import heapq
from collections import defaultdict
class Categorizer:
def __init__(self, seller_category_map):
self.seller_category_map = seller_category_map
# each value is a heap of (–override_count, category)
self.overrides: dict[str, list[tuple[int, DefaultCategories]]] = defaultdict(list)
def add_override(self, seller: str, category: DefaultCategories, count: int):
# push negative count so that the largest count comes out first
heapq.heappush(self.overrides[seller], ( -count, category ))
def categorize(self, transaction):
seller = transaction.seller
if seller in self.seller_category_map:
return self.seller_category_map[seller]
heap = self.overrides.get(seller)
if heap:
# peek the “min” of the heap, which is (–max_count, category)
_, top_category = heap[0]
self.seller_category_map[seller] = top_category
return top_category
return None
Transaction implementation:
class Transaction(object):
def __init__(self, created_at, seller, amount):
self.created_at = created_at
self.seller = seller
self.amount = amount
Solving the thread-safety issue / Scaling the categorizer
- Lock around the heap or switch to Python’s built-in PriorityQueue for immediate thread safety.
- Use a lightweight embedded database (e.g. SQLite in its default serialized mode) and do atomic SQL updates/queries.
- Adopt a simple external store like Redis for atomic counters with INCR.
Each approach trades off complexity versus performance and scalability; pick the simplest that meets your concurrency needs today, then evolve as you grow.
Use case: Service recommends a budget
To start, we could use a generic budget template that allocates category amounts based on income tiers. Using a common template, you avoid materializing all 100 million per-category records up front—each new user simply references the same in-memory or table-driven default rules. Only when a user actually changes one of those defaults do you write the data; everyone else continues to implicitly use the template values. When a user overrides a budget category, we could store the override in the TABLE budget_overrides
.
class Budget(object):
def __init__(self, income):
self.income = income
self.categories_to_budget_map = self.create_budget_template()
def create_budget_template(self):
return {
DefaultCategories.HOUSING: self.income * .4,
DefaultCategories.FOOD: self.income * .2,
DefaultCategories.GAS: self.income * .1,
DefaultCategories.SHOPPING: self.income * .2,
...
}
def override_category_budget(self, category, amount):
self.categories_to_budget_map[category] = amount
For the Budget Service, we can potentially run SQL queries on the transactions
table to generate the monthly_spending
aggregate table. The monthly_spending
table would likely have much fewer rows than the total 5 billion transactions, since users typically have many transactions per month.
As an alternative, we can run MapReduce jobs on the raw transaction files to:
- Categorize each transaction
- Generate aggregate monthly spending by category
Running analyses on the transaction files could significantly reduce the load on the database.
We could call the Budget Service to re-run the analysis if the user updates a category.
Clarify with your interviewer the expected amount, style, and purpose of the code you should write.
Sample log file format, tab delimited:
user_id timestamp seller amount
MapReduce implementation:
class SpendingByCategory(MRJob):
def __init__(self, categorizer):
self.categorizer = categorizer
self.current_year_month = calc_current_year_month()
...
def calc_current_year_month(self):
"""Return the current year and month."""
...
def extract_year_month(self, timestamp):
"""Return the year and month portions of the timestamp."""
...
def handle_budget_notifications(self, key, total):
"""Call notification API if nearing or exceeded budget."""
...
def mapper(self, _, line):
"""Parse each log line, extract and transform relevant lines.
Argument line will be of the form:
user_id timestamp seller amount
Using the categorizer to convert seller to category,
emit key value pairs of the form:
(user_id, 2025-01, shopping), 25
(user_id, 2025-01, shopping), 100
(user_id, 2025-01, gas), 50
"""
user_id, timestamp, seller, amount = line.split('\t')
category = self.categorizer.categorize(seller)
period = self.extract_year_month(timestamp)
if period == self.current_year_month:
yield (user_id, period, category), amount
def reducer(self, key, value):
"""Sum values for each key.
(user_id, 2025-01, shopping), 125
(user_id, 2025-01, gas), 50
"""
total = sum(values)
yield key, sum(values)
Scale the design
Identify and address bottlenecks, given the constraints.
Important: Do not simply jump right into the final design from the initial design!
State you would
- Benchmark/Load Test,
- Profile for bottlenecks
- address bottlenecks while evaluating alternatives and trade-offs, and
- repeat. See Design a system that scales to millions of users on AWS as a sample on how to iteratively scale the initial design.
It's important to discuss what bottlenecks you might encounter with the initial design and how you might address each of them. For example, what issues are addressed by adding a Load Balancer with multiple Web Servers? CDN? Master-Slave Replicas? What are the alternatives and Trade-Offs for each?
We'll introduce some components to complete the design and to address scalability issues. Internal load balancers are not shown to reduce clutter.
Step 4 Wrap up
To summarize, we've designed a financial management system that meets the core requirements. We've discussed the high-level design, identified potential bottlenecks, and proposed solutions to address scalability issues. Now it is time to align again with the interviewer expectations. See if she has any feedback or questions, suggest next steps, improvements, error handling, and monitoring if appropriate.
To avoid repeating discussions, refer to the following system design topics for main talking points, tradeoffs, and alternatives:
- DNS
- CDN
- Load balancer
- Horizontal scaling
- Web server (reverse proxy)
- API server (application layer)
- Cache
- Relational database management system (RDBMS)
- SQL write master-slave failover
- Master-slave replication
- Asynchronism
- Consistency patterns
- Availability patterns
We'll add an additional use case: User accesses summaries and transactions.
User sessions, aggregate stats by category, and recent transactions could be placed in a Memory Cache such as Redis or Memcached.
- The Client sends a read request to the Web Server
- The Web Server forwards the request to the Read API server
- Static content can be served from the Object Store such as S3, which is cached on the CDN
- The Read API server does the following:
- Checks the Memory Cache for the content
- If the url is in the Memory Cache, returns the cached contents
- Else
- If the url is in the SQL Database, fetches the contents
- Updates the Memory Cache with the contents
- If the url is in the SQL Database, fetches the contents
- Checks the Memory Cache for the content
Refer to When to update the cache for tradeoffs and alternatives. The approach above describes cache-aside.
Instead of keeping the monthly_spending
aggregate table in the SQL Database, we could create a separate Analytics Database using a data warehousing solution such as Amazon Redshift or Google BigQuery.
We might only want to store a month of transactions
data in the database, while storing the rest in a data warehouse or in an Object Store. An Object Store such as Amazon S3 can comfortably handle the constraint of 250 GB of new content per month.
To address the 200 average read requests per second (higher at peak), traffic for popular content should be handled by the Memory Cache instead of the database. The Memory Cache is also useful for handling the unevenly distributed traffic and traffic spikes. The SQL Read Replicas should be able to handle the cache misses, as long as the replicas are not bogged down with replicating writes.
2,000 average transaction writes per second (higher at peak) might be tough for a single SQL Write Master-Slave. We might need to employ additional SQL scaling patterns:
We should also consider moving some data to a NoSQL Database.
Additional talking points
Additional topics to dive into, depending on the problem scope and time remaining.
NoSQL
Caching
- Where to cache
- What to cache
- When to update the cache
Asynchronism and microservices
Communications
- Discuss tradeoffs:
- External communication with clients - HTTP APIs following REST
- Internal communications - RPC
- Service discovery
Security
Refer to the security section.
Latency numbers
See Latency numbers every programmer should know.
Ongoing
- Continue benchmarking and monitoring your system to address bottlenecks as they come up
- Scaling is an iterative process