Private messaging - Part III
This guide has four distinct parts:
- Part I: initial implementation
- Part II : persistent user ID
- Part III (current): persistent messages
- Part IV: scaling up
Here's where we were at the end of the 2nd part:
data:image/s3,"s3://crabby-images/4e7af/4e7af267c34a4eca6f6b0b16484c3798228c1926" alt="Chat (v2)"
All is working pretty well, but there is a last issue which is quite annoying:
- when the sender gets disconnected, all the packets it sends are buffered until reconnection (which is great in this case)
data:image/s3,"s3://crabby-images/c98f4/c98f40e11a0f1474af410bac4f24ddc178e756d3" alt="Chat with sender that gets disconnected"
- but when the recipient gets disconnected, the packets are lost, since there is no listening Socket instance in the given room
data:image/s3,"s3://crabby-images/48a07/48a0739fc61698711d9263f42968fad4b9e175c3" alt="Chat with recipient that gets disconnected"
There are multiple solutions to this problem, and we will go for the easiest to implement: store all the messages on the server-side.
Note: this 3rd part will be brief, but it underlines an important property of Socket.IO: you cannot rely on the status of the connection. It should be up most of the time but there is a myriad of things that can kill a TCP connection (this is particularly true on mobile browsers).
Installationβ
Let's checkout the branch for part III:
git checkout examples/private-messaging-part-3
Here's what you should see in the current directory:
βββ babel.config.js
βββ package.json
βββ public
β βββ favicon.ico
β βββ fonts
β β βββ Lato-Regular.ttf
β βββ index.html
βββ README.md
βββ server
β βββ index.js (updated)
β βββ messageStore.js (created)
β βββ package.json
β βββ sessionStore.js
βββ src
βββ App.vue
βββ components
β βββ Chat.vue (updated)
β βββ MessagePanel.vue
β βββ SelectUsername.vue
β βββ StatusIcon.vue
β βββ User.vue
βββ main.js
βββ socket.js
The complete diff can be found here.
How it worksβ
Persistent messagesβ
On the server-side (server/index.js
), we now persist the message in our new store:
io.on("connection", (socket) => {
// ...
socket.on("private message", ({ content, to }) => {
const message = {
content,
from: socket.userID,
to,
};
socket.to(to).to(socket.userID).emit("private message", message);
messageStore.saveMessage(message);
});
// ...
});
And we fetch the list of messages upon connection:
io.on("connection", (socket) => {
// ...
const users = [];
const messagesPerUser = new Map();
messageStore.findMessagesForUser(socket.userID).forEach((message) => {
const { from, to } = message;
const otherUser = socket.userID === from ? to : from;
if (messagesPerUser.has(otherUser)) {
messagesPerUser.get(otherUser).push(message);
} else {
messagesPerUser.set(otherUser, [message]);
}
});
sessionStore.findAllSessions().forEach((session) => {
users.push({
userID: session.userID,
username: session.username,
connected: session.connected,
messages: messagesPerUser.get(session.userID) || [],
});
});
socket.emit("users", users);
// ...
});
The code is quite straightforward. We shouldn't lose messages upon disconnection anymore:
data:image/s3,"s3://crabby-images/964e6/964e6ed3b5174c7d0e14dfc965275c978d11a4e6" alt="Chat (v3)"
Reviewβ
Now that we have a functional chat, we will see in the 4th part of this guide how to scale to multiple Socket.IO servers.
Thanks for reading!