.claude/skills/ably-realtime/SKILL.md
Ably real-time messaging patterns, WebSocket channel management, message validation and processing, staleness filtering, error recovery strategies, collaborative editing with drag-and-drop, optimistic updates for voting, real-time board collaboration, and Ably integration best practices for ree-board project
npx skillsauth add DW225/ree-board ably-realtimeInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Activate this skill when working on:
Channel Naming Convention:
// Pattern: `board:{boardId}`
const channelName = `board:${boardId}`;
Setting Up Channels:
"use client";
import { useChannel } from "ably/react";
import { useEffect } from "react";
export function PostChannelComponent({ boardId }: { boardId: string }) {
const { channel } = useChannel(`board:${boardId}`, (message) => {
processMessage(message);
});
return <PostList boardId={boardId} />;
}
Critical Pattern: Always validate messages before processing
// lib/realtime/messageProcessors.ts
import { z } from "zod";
// Define message schema
const PostUpdateMessageSchema = z.object({
type: z.literal("post:update"),
postId: z.string(),
content: z.string().min(1).max(1000),
userId: z.string(),
timestamp: z.number(),
});
// Message processor
export const processPostUpdate = (rawData: unknown) => {
try {
// ✅ Validate message structure
const data = PostUpdateMessageSchema.parse(rawData);
// ✅ Check staleness (30s threshold)
const now = Date.now();
const age = now - data.timestamp;
if (age > 30000) {
console.warn("Stale message discarded", {
type: data.type,
age,
postId: data.postId,
});
return;
}
// ✅ Process validated, fresh message
updatePostContent(data.postId, data.content);
} catch (error) {
if (error instanceof z.ZodError) {
console.error("Invalid message structure", {
details: error.errors,
rawData,
});
} else {
console.error("Message processing error", error);
}
}
};
30-Second Threshold: Prevents processing old messages after reconnection
const STALENESS_THRESHOLD_MS = 30000;
export function isMessageStale(timestamp: number): boolean {
const age = Date.now() - timestamp;
return age > STALENESS_THRESHOLD_MS;
}
// Usage in message handler
useChannel(`board:${boardId}`, (message) => {
const { timestamp } = message.data;
if (isMessageStale(timestamp)) {
console.warn("Dropping stale message", { age: Date.now() - timestamp });
return;
}
processMessage(message.data);
});
Connection Error Handling:
import { useConnectionStateListener } from "ably/react";
export function RealtimeProvider({ children }: { children: React.ReactNode }) {
const [connectionState, setConnectionState] = useState<string>("initialized");
useConnectionStateListener((stateChange) => {
setConnectionState(stateChange.current);
switch (stateChange.current) {
case "connected":
console.log("✅ Connected to Ably");
break;
case "disconnected":
console.warn("⚠️ Disconnected from Ably");
break;
case "suspended":
console.error("❌ Connection suspended");
// Optionally show user notification
break;
case "failed":
console.error("❌ Connection failed");
// Show error message to user
break;
}
});
return (
<>
{connectionState !== "connected" && (
<ConnectionBanner state={connectionState} />
)}
{children}
</>
);
}
Always Include Timestamp:
import { useChannel } from "ably/react";
export function usePublishPostUpdate() {
const { channel } = useChannel(`board:${boardId}`);
const publishUpdate = async (postId: string, content: string) => {
await channel.publish("post:update", {
type: "post:update",
postId,
content,
userId: currentUserId,
timestamp: Date.now(), // ✅ Always include timestamp
});
};
return publishUpdate;
}
Pattern: Update UI immediately, sync in background
"use client";
import { voteSignal } from "@/lib/signal/postSignals";
import { useChannel } from "ably/react";
export function VoteButton({
postId,
boardId,
}: {
postId: string;
boardId: string;
}) {
const { channel } = useChannel(`board:${boardId}`);
const handleVote = async () => {
// ✅ Optimistic update (immediate UI feedback)
voteSignal.value = {
...voteSignal.value,
[postId]: (voteSignal.value[postId] || 0) + 1,
};
try {
// Persist to database
await submitVote(postId);
// Broadcast to other users
await channel.publish("post:vote", {
type: "post:vote",
postId,
increment: 1,
timestamp: Date.now(),
});
} catch (error) {
// ❌ Rollback on error
voteSignal.value = {
...voteSignal.value,
[postId]: voteSignal.value[postId] - 1,
};
console.error("Vote failed", error);
}
};
return <button onClick={handleVote}>Vote</button>;
}
Lazy-Loaded with Real-Time Sync:
// components/board/PostProvider.tsx
"use client";
import { useChannel } from "ably/react";
import dynamic from "next/dynamic";
// ✅ Lazy load drag-and-drop (reduces initial bundle)
const DragDropArea = dynamic(() => import("./DragDropArea"), { ssr: false });
export function PostProvider({ boardId }: { boardId: string }) {
useChannel(`board:${boardId}`, (message) => {
if (message.name === "post:move") {
handlePostMove(message.data);
}
});
const handleDrop = async (postId: string, newType: PostType) => {
// Update locally
movePostSignal(postId, newType);
// Persist to database
await updatePostType(postId, newType);
// Broadcast to other users
channel.publish("post:move", {
type: "post:move",
postId,
newType,
timestamp: Date.now(),
});
};
return <DragDropArea onDrop={handleDrop} />;
}
Bad:
useChannel(`board:${boardId}`, (message) => {
// ❌ Trusts message data completely
updatePost(message.data.postId, message.data.content);
});
Good:
useChannel(`board:${boardId}`, (message) => {
// ✅ Validates before processing
const validated = PostUpdateSchema.safeParse(message.data);
if (!validated.success) return;
updatePost(validated.data.postId, validated.data.content);
});
Bad:
useChannel(channelName, (message) => {
// ❌ Processes all messages, even old ones after reconnect
processMessage(message.data);
});
Good:
useChannel(channelName, (message) => {
// ✅ Filters stale messages
if (isMessageStale(message.data.timestamp)) return;
processMessage(message.data);
});
Bad:
// ❌ No error handling
const { channel } = useChannel(channelName);
Good:
// ✅ Monitor connection state
useConnectionStateListener((stateChange) => {
if (stateChange.current === "failed") {
showErrorNotification("Real-time connection lost");
}
});
Bad:
channel.publish("update", {
postId,
content,
// ❌ No timestamp for staleness check
});
Good:
channel.publish("update", {
postId,
content,
timestamp: Date.now(), // ✅ Include timestamp
});
Bad:
// ❌ Multiple updates could conflict
const handleVote = async () => {
const newCount = currentCount + 1;
await updateVoteCount(postId, newCount);
};
Good:
// ✅ Use atomic increment
const handleVote = async () => {
await db
.update(postTable)
.set({ voteCount: sql`vote_count + 1` })
.where(eq(postTable.id, postId));
};
lib/realtime/messageProcessors.ts - Message validation and processingcomponents/board/PostProvider.tsx - Real-time channel setupcomponents/board/PostChannelComponent.tsx - Channel subscriptionlib/realtime/__tests__/ - Message processor testsCurrent Message Types:
type MessageType =
| "post:create"
| "post:update"
| "post:delete"
| "post:move"
| "post:vote"
| "member:join"
| "member:leave";
One Channel Per Board:
board:{boardId}Automatic Reconnection:
Manual Recovery:
// Refresh data after long disconnection
if (
stateChange.previous === "suspended" &&
stateChange.current === "connected"
) {
await refreshBoardData();
}
Mock Ably in Tests:
jest.mock("ably/react", () => ({
useChannel: jest.fn(() => ({
channel: {
publish: jest.fn(),
subscribe: jest.fn(),
},
})),
}));
Last Updated: 2026-01-10
development
Jest testing strategies, test organization, factory patterns for test data, mocking strategies for authentication and external services, real-time message processor testing, test-driven development workflow, unit vs integration testing, fake timer usage for time-dependent tests, and testing best practices for ree-board project
development
Preact Signals for reactive state management, signal vs computed signal usage, batch updates for performance, action creator patterns, signal integration with React components, state management by domain (boards posts members), reactive patterns, and signal best practices for ree-board project
testing
Role-based access control (RBAC) patterns, authentication wrappers, authorization checks, input validation with Zod schemas, security boundaries, server action security, real-time message validation, preventing common vulnerabilities like XSS and SQL injection, and security best practices for ree-board project
tools
Next.js 16 App Router patterns including server components, client components, server actions, route handlers, layouts, metadata API, dynamic routes, file conventions, data fetching, caching strategies, and Next.js best practices for building modern React applications