Examples
Chat
Lets create a simple chat application using Nuxt WebSocket.
We will create a simple chat application where users can send messages to each other. The server will validate the incoming messages, store them in memory and broadcast them to all connected clients. New users will receive the chat history and the current number of users.
Setup
nuxt.config.ts
export default defineNuxtConfig({
modules: ['nuxt-ws'],
ws: {
route: '/_ws', // WebSocket endpoint to auto-connect
topics: {
defaults: ['session'], // Auto-subscribed topics
}
}
})
Client implementation
pages/chat.client.vue
<template>
<div>
<div v-if="status === 'OPEN'">
Connected, Users: {{ states['session'].users }}
</div>
<div v-else>
Disconnected
</div>
<div v-for="({ user, text }, key) in states['chat']" :key>
{{ user }}: {{ text }}
</div>
<input v-model="textRef" @keyup.enter="sendMessage()">
</div>
</template>
<script setup lang="ts">
const textRef = ref('')
const { states, status, send } = useWS<{
chat: {
user?: string
text: string
}[]
session: {
users: number
}
}>(['chat'])
function sendMessage() {
send('publish', 'chat', {
text: textRef.value,
})
textRef.value = ''
}
</script>
Server implementation
server/routes/_ws.ts
import * as v from 'valibot'
export default defineWSHandler({
async open(peer) {
// Update peer with 'chat' data from storage
const chat = await useStorage('ws').getItem('chat')
if (chat)
peer.send(JSON.stringify({
topic: 'chat',
payload: chat,
}), { compress: true })
// Update everyone's session metadata
const payload = JSON.stringify({ topic: 'session', payload: { users: peer.peers.size } })
peer.send(payload, { compress: true })
peer.publish('session', payload, { compress: true })
},
async message(peer, message) {
// Validate the incoming chat message
const parsedMessage = await wsValidateMessage( // built-in validation util
v.object({
type: v.literal('publish'),
topic: v.literal('chat'),
payload: v.object({ text: v.string() }),
}),
message,
)
// Update chat data in storage
const mem = useStorage('ws')
const { topic, payload } = parsedMessage
const _chat = await mem.getItem<Array<{ user: string, text: string }>>('chat') || []
const newChat = [..._chat, { ...payload, user: peer.id }]
await mem.setItem(topic, newChat)
// Broadcast the updated chat to everyone
peer.send(JSON.stringify({ topic, payload: newChat }), { compress: true })
peer.publish(topic, JSON.stringify({ topic, payload: newChat }), { compress: true })
},
close(peer) {
// Update everyone's session metadata
peer.publish(
'session',
JSON.stringify({
topic: 'session',
payload: {
users: peer.peers.size,
},
}),
{ compress: true },
)
},
})