The list of posts in this series:
- Part 1: A Simple Framework For Mobile System Design Interviews
- Part 2: Additional Topics For Mobile System Design Interviews
- Part 3: Mobile System Design Exercise: File Downloader Library
- Part 4: Mobile System Design Exercise: Caching Library
- Part 5: Mobile System Design Exercise: Image Library
-
Part 6: Mobile System Design Exercise: Chat Application
Disclaimer
The proposed solution is far from being perfect but it is not the point of a system design interview round: no one expects you to build a robust system in just 30–40 min — the interviewer is mostly looking for specific “signals” from your thought process and communication.
Defining The Task
The task might be simply defined as “Design Chat Application”. The definition is purposely vague and requires some clarification.
Gathering Requirements
You are expected to ask clarifying questions and narrow down the scope of the task. This should not take more than 5 minutes.
Candidate: “Are we designing a general usage app (like WhatsApp) or something more specific (like an internal chat app for a company)?”
Interviewer: “Something like WhatsApp.”Candidate: “Do we have any information on the number of monthly active users?”
Interviewer: “Why is this important for a mobile app?”
Candidate: “We need to make sure that our clients won’t accidentally create a DDoS attack on our backend services.”Interviewer: “In what cases a DDoS attack is possible?”
Candidate: “The most likely reason is a frequent HTTP-polling that produces large volumes of network traffic.”
Candidate: “Another reason — not using an exponential backoff while retrying failing requests. Million of clients retrying their requests at the same time can produce an unnecessary load to our backend services.”
Interviewer: “Got it. Let’s say we’re expecting to have 1,000,000 monthly active users.”Candidate: “Do we know anything about our target market and the audience?”
Interviewer: “Why do you think this is important?”
Candidate: “An app targetted primarily for North America and Europe might have a different set of performance requirements compared to a similar app targetted to developing countries. The users in developed countries usually have more powerful phones compared to their peers in developing countries. The cost of cellular traffic in Europe and North America is much lower than in India.”
Interviewer: “Let’s design an app for North America.”Candidate: “Do we want to be able to edit/delete messages?”
Interviewer: “Let’s leave it out of scope.”Candidate: “Do we want to support group chats?”
Interviewer: “No, let’s leave it out of scope.”Candidate: “Do we need to support sign-up and log-in?”
Interviewer: “We can leave it out of scope.”
Make sure not to overload the system requirements with unnecessary features. Think in terms of MVP (Minimum Viable Product) and pick features that have the biggest value. You can learn more about requirements gathering here.
Functional requirements
- Users should be able to see the list of recent chats sorted by date.
- Users should be able to open a 1-on-1 chat, send and receive messages.
- Users should be able to add photo attachments to chat messages.
- Users should be able to see chat message status (sending, sent/failed, received) and read receipts.
Non-functional requirements
- Offline support (reading chat messages).
- Secure message storage.
- Real-time notifications.
Out of scope
- Video and voice attachments.
- Sign-up and log-in.
- Editing/Deleting messages.
- Group chats.
NOTE: It is tempting to include additional requirements such as “last seen”, message replies/forwarding, reactions, etc. Proper time management is crucial during the interview — try keeping everything closer to an MVP and move on to the next section.
High-Level Diagram
A high-level diagram shows all major system components and their interactions (without implementation details). You can learn more about high-level diagrams here.
High-Level Diagram
NOTE: This diagram looks familiar to other cases from the guide. The main reason is the adopted architectural pattern combination MVx + repository + coordinator.
Server-side components:
- Backend
Represents the whole server-sider infrastructure. Most likely, your interviewer won’t be interested in discussing it. - Push Provider
Represents the Mobile Push Provider infrastructure. Receives push payloads from the Backend and delivers them to clients. - CDN (Content Delivery Network)
Responsible for delivering static content to clients.
Client-side components:
- API Service
Abstracts client-server communications from the rest of the system. - Persistence
A single source of truth. The data your system receives gets persisted on the disk and then propagated to other components. - Repository
A mediator component between API Service and Persistence. - Chat Lobby
Represents a set of components responsible for displaying a list of recent chats. - Chat Room
Represents a set of components responsible for displaying a single chat room. - DI Graph
Dependency injection graph. See Dependency injection in Android and Dependency injection in iOS. - Image Loader
Responsible for loading images from web/disk into UI elements. - Attachment Provider
Responsible for loading image attachments from the media library and camera into memory. - Coordinator
Organizes flow logic between Chat Lobby and Chat Room components. Helps decouple components of the system from each other. - App Module
An executable part of the system which “glues” components together.
Deep Dive
After a high-level discussion, your interviewer might want to discuss some specific components of the system. Make sure to keep your explanation brief and don’t overload it with details — let your interviewer guide the conversation and ask questions instead. You can learn more about deep-dive discussions here.
Deep Dive: API Service
Interviewer: “Can you tell me more about the API Service component?”
Candidate: “It’s an abstraction for the network communication layer.”
Candidate: “The idea is to isolate the low-level transport primitives from the rest of the app. This way, we can better support modularity and testability of the project.”Interviewer: “What protocols would you use for the network communication?”
Candidate: “I would split everything into three major components:”
1. Bi-directional Communication Layer
Candidate: “Real-time bi-directional communication layer for sending/receiving chat messages and status. We need to decide between connection-based (TCP) and connectionless (UDP) protocols.”
Candidate: “TCP-based clients establish a virtual connection and provide an ordered delivery guarantee by re-transmitting lost packets. It might be more expensive in terms of battery life (especially on flaky networks where the participants need to frequently repeat handshakes to restore lost connection). Another disadvantage is a 64k limit for the connection count for any given host port and a bigger packet header size compared to UDP. Some examples of TCP-based protocols: WebSocket (Slack), XMPP (WhatsApp, Zoom, Google Talk), MQTT (IoT, Smart Home, etc).”
Candidate: “UDP-based clients are more lightweight and don’t require any handshakes. As a result, UDP can’t provide an ordered delivery guarantee and has no error checking beyond simple checksums. Some examples of UDP-based protocols: WebRTC (Discord, Google Hangouts, Facebook Messenger) — also works over TCP.”
NOTE: You can learn more about the differences between TCP and UDP here.
Candidate: “I don’t have much experience working with real-time protocols, so I select WebSockets to simplify the design. Some disadvantages of this choice — the protocol is schemeless and does not provide automatic reconnection. There are also several security flaws. Alternatively, we can use HTTP-polling — might be a poor choice since it would generate an excessive amount of backend traffic.”
Candidate: “We can also use gRPC (bi-directional streaming) or GraphQL (subscriptions), but I don’t have enough experience with them.”
NOTE: It is better not to bring unfamiliar technologies to the discussion. You can mention some of their advantages — but don’t go too deep unless you’re prepared to discuss implementation-specific details.
Interviewer: “How would you use WebSockets?”
Candidate: “I only need them to send and receive real-time chat events. For everything else, we can use HTTP-based protocols.”Interviewer: “Can you describe the event data format?”
Candidate: “Sure, a typical data frame might look like this:”
{ connection_id: String? event_type: "HELLO|MSG_IN|MSG_OUT|MSG_READ|BUY" payload: { ... } }
Job Offers
Events:
HELLO
– initiates a new connection session (bi-directional).MSG_IN
– incoming message.MSG_OUT
– outgoing message.MSG_READ
– message “read” acknowledgement (bi-directional).BUY
– closes a connection session
Interviewer: “What is a ‘bi-directional’ event?”
Candidate: “That means the same event name could be sent both from the client and the server. For example, the client sendsHELLO
and the server responds with anotherHELLO
. Alternatively, we can use prefixes likeCLT_HELLO
andSRV_HELLO
“.Candidate: “Another approach could be push-only events: the client uses
POST
orPUT
HTTP requests to send the data to the server. The server can use WebSocket to push events to clients. The disadvantage of such approach – a client needs to create a new connection every time it needs to send an event.”Interviewer: “Why do you need a
connection_id
? Where is it coming from?”
Candidate: “To differentiate between clients on the backend. A client receives it with theHELLO
event from the server.”Interviewer: “How would you keep a socket connection when the app goes to the background?”
Candidate: “I don’t need to keep it alive while the app is not active — a push notification can be used to notify the user about new messages and bring the app back to the foreground.”
Interviewer: “Sounds good.”
NOTE: It’s tempting for iOS engineers to mention URLSessionWebSocketTask as a solution for bi-directional communication. But it’s also important to remember that the API is not available until iOS 13. Make sure to keep OS version compatibility in mind.
2. HTTP-based layer
Candidate: “Request-response layer for fetching a paginated list of chats or message history. REST protocol could be a good choice since we don’t need much of the request customization to bring up GraphQL (and have an extra complexity on the backend side).”
Interviewer: “Could you briefly describe the API design?”
Candidate: “We would need to have a couple of end-points:”
GET /login
– initiates a new client session and returns a JSON Web Token token for authorizing further requests.GET /chats?after_id=<X>&limit=<Y>
– receives a paginated list of chats.GET /chats/<chat_id>/messages?after_id=<X>&limit=<Y>
– receives a paginated list of messages from a specific chat.
NOTE: Don’t over-complicate your design — aim to cover more ground (unless the interviewer wants to dig deeper into the protocols).
Interviewer: “Why do you need a JWT?”
Candidate: “For client authentication: each request (besides/login
) should include an authorization header:Authorization: Bearer <token>
.”Interviewer: “Can you explain these query params
messages?after_id=<X>&limit=<Y>
?”
Candidate: “It’s a cursor-based pagination:before_id
andafter_id
contain the first/last message id in the current page;limit
represents the maximum amount of items in a single page. Alternatively, we can use a keyset-pagination (using the timestamp of the last message) but it might not be reliable since we can’t trust the client device clock.”
NOTE: Learn more about pagination here.
Candidate: “We should also discuss some specific HTTP response codes.”
401 Unauthorized
– the client authorization token is missing or expired: thelogin
request must be sent before continuing.422 Unprocessable Entity
– the client request data is malformed and should not be retried.429 Too Many Requests
– the client reached the rate-limiting threshold.500 Internal Server Error
– the client should use exponential back-off to retry a failed request.
3. Cloud Messaging Layer
Candidate: “A typical push payload could look like this:”
{ user_id: String messages: [ { user_name: String text: String created_at: String }, ... ] }
Interviewer: “Why do we need a
user_id
along withuser_name
?”
Candidate: “user_id
contains the id of the recipient to make sure that nobody else would see the message. Imagine the situation when another user logs in on the same device and receives a push for the previously logged-in user.”
Candidate: “user_name
contains a display name of the sender;text
contains the first 100-120 characters of the message. This way we don’t need to make any additional requests and can display a status bar notification right away.”
API Service Diagram
API Service Diagram
Candidate: “I would introduce separate interfaces (protocols) to abstract communication between the network layer and the business logic: ChatLobbyService
(receive the list of chats, create/delete chats) and ChatRoomService
(receive the message history, send/receive messages, download/upload attachments).”
Interviewer: “Why do you need to have separate interfaces?”
Candidate: “To decouple different modules of the app. For example, the Chat Lobby
flow can receive the ChatLobbyService
instance from DI and only use the public API contract without knowing anything about the underlying network module implementation. It also makes development/testing easier since developers can swap real implementation with fake instances and serve predefined data from disk or memory.”
Candidate: “I would also use a Model Converter
layer to turn network response classes into model classes to hide protocol/implementation details and perform simple data transformations.”
Interviewer: “Not sure if I’m following – can you elaborate?”
Candidate: “Imagine that you received a data frame containing a chat message and deserialized it into an object.”
ChatMessageData: + id: String + user_id: String + text: String + status: String + created_at: Long + attachments: String // comma-separated list
Candidate: “You can convert it into a model object.”
ChatMessage + id: String + userId: String + text: String + status: ChatMessageStatus + createdAt: Date + attachments: [Attachments]
Candidate: “This way the business logic does not need to know about the network layer and the data format of the transport protocols. You can change the networking implementation without affecting the rest of the app.”
Deep Dive: Data Model
Interviewer: “How would you organize your data storage?”
Candidate: “I would use a relational database (ORM) to store chats, messages, users, and attachments.”
Candidate: “Other alternatives might be less robust since we’re looking for querying support and data integrity.”
NOTE: Don’t waste time describing approaches that obviously won’t fit the task at hand (like app preferences, text/binary files, etc).
Candidate: “I would use a relational database (ORM) and create tables for chats, messages, users, and attachments.”
Entity-relationship “like” Diagram
NOTE: This diagram loosely follows the entity-relationship diagram notation. The cardinality/ordinality relationships are purposely omitted to save the interview time. A good rule of thumb is to avoid unnecessary drawings altogether.
Candidate: “The
User
table is pretty straightforward: we would store user id, name, and profile picture URL.”Candidate: “The
Chat
table would contain the list of active chats: we would store the chat id, the id of the user who sent the last message, and the last message id.”Candidate: “The
Message
table would contain all messages from all chats: we would store the message id, the user id, the chat id, the message status (PENDING
,SENT
,READ
,FAILED
), and a comma-separated list of attachment ids.”Candidate: “The
Attachment
table would contain all attachments for all messages from all chats: we would store the attachment id, the image URLs (full-size and the low-res), the cached local versions paths, the attachment status (UPLOADING
,DOWNLOADING
,FAILED
,READY
), the total file size and the progress size (uploaded/downloaded bytes).Candidate: “We can join the
Chat
table with theMessage
table (on thelast_message_id
column) and with theUser
table (on thelast_user_id
column) to get a list of recent chats. Each result could be represented by the following model class:”
ChatInfo: + chatId: String + lastUsername: String + lastUserProfileUrl: Url + lastMessageText: String + lastMessageTimestamp: Date
Candidate: “We can get the list of messages for a specific chat by selecting on the
chat_id
column.”Interviewer: “What would you use for message and attachment ids?”
Candidate: “What do you mean?”Interviewer: “Are those local ids or server ids?”
Candidate: “We can use client-generated 128-bit UUIDs. The outgoing messages can be identified byuser_id
and the outgoing attachment by emptyurl
columns: once an attachment is uploaded – it’s undistinguished from a remote attachment.”
Candidate: “The advantage of such approach is its simplicity and built-in idempotency; the disadvantage – clients would generate resource ids which is less reliable and less secure compared to the backend generation.”Candidate: “Alternatively, we can maintain separate local and server ids: all local operations would be using local ids while the backend communication would use server ids. We would also need to build a bijection between them.”
NOTE: Learn more about Local and Servier ids here.
Deep Dive: Attachments
Interviewer: “How would you handle attachments?”
Candidate: “I’ll keep the incoming/outgoing attachments metadata in the same table to simplify the message queries — this way we would only need to join on a single table. Alternatively, we can create a separate table for outgoing attachments but this might complicate the overall design.”
Candidate: “The state of the attachments would be controlled by thestatus
column. TheREADY
status corresponds to an outgoing attachment that has been successfully uploaded or to an incoming attachment that has been successfully downloaded. Other status values – for handling uploads/downloads/failures.”
Candidate: “We also need to keep track of uploading/downloading progress to support resumable uploads/downloads.”Interviewer: “How would you distinguish between incoming and outgoing attachments?”
Candidate: “We don’t need to keep them distinct — only keep track if the attachment needs to be uploaded or uploaded. This can be determined by theurl
column: it will beNULL
for outgoing attachments.”Interviewer: “What component would download/upload attachments?”
Candidate: “The API Service via HTTP client. We would also need a task dispatcher to limit the number of the concurrent network operations.”
NOTE: For more information on task dispatchers see this.
Candidate: “We can automatically download incoming attachments on WiFi and ask for user input on cellular networks. The application settings might be provided for better user experience.”
Follow-up Questions
Some interviewers might ask follow-up questions that might change the original design and introduce new requirements.
Timestamps
Interviewer: “How would you handle timestamps?”
Candidate: “This is a tricky question. I would probably use UTC timestamps (not adjusted to daylight saving and timezones) for messages and just format them to the localized time before displaying.”Interviewer: “Would you use the client or the server time?”
Candidate: “I would use the client time for outgoing messages and the server time for incoming. For example, when a user sends a message — the client local UTC time would be applied. When the server receives a message — it would assign the server local UTC time so the other chat participant would see it as a timestamp.”
Interviewer: “Why won’t you use the client time for both cases?”
Candidate: “We can’t trust the client time: if the device clock reports incorrect time — we can end up with messages from the future.”
Candidate: “There’s still a chance for the wrong message ordering. For example, when the user sends messages while being offline.”
NOTE: Ensuring a proper message order is a tricky problem: it’s pretty unlikely for a candidate to know a solution unless they’ve been working on chat apps in the past.
Security & Privacy
Interviewer: “Where would you store chat messages?”
Candidate: “I would need to have a cached copy on the device for offline viewing. We can also store it on the backend but it imposes privacy issues even if we encrypt them.”Interviewer: “What kind of issues are you talking about?”
Candidate: “Even if the messages are encrypted on the backend — the developer still has an access to encryption keys so an attacker who has the access to the servers can still compromise user privacy.”
Candidate: “Additionally, the user perception of privacy is as important as the privacy implementation itself. If the developer has the access to the messages — the customers might assume that those messages would be read without any consent.”
Candidate: “An end-to-end encryption could be used but I don’t have any experience with the algorithms for key exchange.”
NOTE: For more information about messaging privacy check WhatsApp Encryption Overview.
Conclusion
Keep this in mind while preparing for a system design interview:
- Don’t try to make it perfect — provide a “signal” instead.
- Listen to your interviewer and keep track of the time.
- Try to cover as much ground as possible without digging too much into the implementation details.
Looking for more content?
Join the “Mobile System Design” Discord server for general discussions and feedback.
Originally published at: https://github.com/weeeBox/mobile-system-design/blob/master/exercises/chat-app.md