From game-creator
Adds real-time or turn-based multiplayer to existing Phaser 3 or Three.js browser games using PartyKit on Cloudflare Durable Objects. Scaffolds room-based server, NetworkManager client, EventBus events, GameState fields, and extends render_game_to_text().
npx claudepluginhub opusgamelabs/game-creator --plugin game-creatorThis skill uses the workspace's default tool permissions.
Add real-time or turn-based multiplayer to an existing single-player browser game. This skill scaffolds:
Adds real-time multiplayer to browser games using @vibedgames/multiplayer for shared state, player state, events, co-op/PvP. Supports React hooks, vanilla JS, Phaser, Three.js.
Builds complete browser games from scratch: scaffolds with Phaser/Three.js, generates assets/audio, adds Playwright tests, deploys to here.now, monetizes via Play.fun. Use for new 2D/3D game concepts.
Provides expertise in real-time multiplayer game networking: lag compensation, state synchronization, authoritative servers, rollback netcode, matchmaking, and anti-cheat.
Share bugs, ideas, or general feedback.
Add real-time or turn-based multiplayer to an existing single-player browser game. This skill scaffolds:
NetworkManager wired through EventBus that mirrors the existing playfun.js external-service pattern.EventBus, GameState, Constants, and render_game_to_text() — single-player gameplay must remain identical when the server is unreachable.The default state is "single-player works." If the WebSocket connection fails, NetworkManager swallows the error and the game runs locally as before. When connected, remote players appear via network:player-joined and synchronize via network:state-received.
architecture.md — event taxonomy, GameState schema, NetworkManager contract, Phaser vs Three.js placement notes.partykit-server.md — server templates (realtime.ts and turn-based.ts), state shape, broadcast helpers, rate limiting.client-integration.md — MultiplayerClient, NetworkManager, RemotePlayerRegistry source, EventBus/GameState/Constants append patterns, render_game_to_text extension.deploy.md — npx partykit dev and npx partykit deploy walkthrough, capturing the deployed URL, .env handling, and client redeploy.These are rules, not guidelines:
network:disconnected instead of throwing.EventBus.js, GameState.js, Constants.js, main.js, and render_game_to_text() under a // === Multiplayer === banner. Never rename, remove, or change existing fields.realtime mode validation is light (last-write-wins). In turn-based mode validation is strict (rejects out-of-turn moves).partysocket calls go through MultiplayerClient. If a future user wants Colyseus or fly.io+ws, only MultiplayerClient.js changes — game code does not.'lobby'. No matchmaking UI in v1. Users override by emitting multiplayer:join-room with a custom room id.src/core/EventBus.js, src/core/GameState.js, src/core/Constants.js, src/main.js with window.render_game_to_text()).npx partykit deploy (the CLI walks the user through login on first deploy; free tier is sufficient for prototyping).The user wants to add multiplayer to the game at $ARGUMENTS (or the current directory if no path given). Optional --mode=realtime (default) or --mode=turn-based chooses the server template.
Parse $ARGUMENTS for the game path and --mode flag. If no path, use cwd. Verify it's a game by reading package.json and confirming Phaser or Three.js dependency.
Read these files in full before touching anything:
package.json — engine + scripts.src/main.js — orchestrator, window.render_game_to_text(), window.advanceTime().src/core/EventBus.js — exact event names already in use.src/core/GameState.js — current state shape and reset() semantics.src/core/Constants.js — config block conventions.progress.md if present — pipeline context.Then tell the creator one sentence confirming what you saw:
Game is
<engine>with<N>events and a<player|bird|ship>entity. I'll add a multiplayer layer that broadcasts the local<entity>'s state atTICK_RATE_HZand renders remote players from server broadcasts. Single-player will continue to work when the server is offline.
Pick the server template:
| Mode | When to use | Wire model |
|---|---|---|
realtime (default) | Action games, runners, dodgers, platformers, anything with continuous movement | Local setInterval at TICK_RATE_HZ broadcasts the local entity's {x, y, [z], score, state}; server fans out; clients render last-known remote state |
turn-based | Card games, board games, puzzles, anything with discrete moves | EventBus events (player:moved, player:played-card) forward as {type, payload} messages; server validates and broadcasts; clients apply on network:state-received |
If the user did not pass --mode, infer from the game's existing events. If you see continuous-position events (bird:flap, player:moved, position-updating physics), use realtime. If you see discrete actions (card:played, move:submitted), use turn-based. State the choice and proceed.
Create a sibling multiplayer-server/ directory inside the game project. See partykit-server.md for the full template content.
Create:
multiplayer-server/partykit.json — manifest with name (use the game's directory name), main: "src/server.ts", compatibilityDate.multiplayer-server/package.json — partykit dep, dev/deploy scripts.multiplayer-server/tsconfig.json — minimal TypeScript config that PartyKit accepts.multiplayer-server/src/server.ts — paste the appropriate template from partykit-server.md (realtime or turn-based).multiplayer-server/.gitignore — node_modules, .partykit.Run cd multiplayer-server && npm install to install partykit (which provides partysocket for the client too via npm workspaces, but we'll add partysocket explicitly to the client).
Create three new files. See client-integration.md for the full source.
src/multiplayer/MultiplayerClient.js — backend-agnostic interface around partysocket (connect, send, onMessage, disconnect, isConnected).src/multiplayer/RemotePlayerRegistry.js — Map<playerId, RemotePlayer> with upsert, remove, prune(staleMs), list().src/systems/NetworkManager.js — wires MultiplayerClient ↔ EventBus, owns the broadcast tick (in realtime mode), handles reconnect with exponential backoff, emits network:* events.Add partysocket to the game's package.json deps:
cd <game-path> && npm install partysocket
Make additive edits only. See architecture.md for full schemas and client-integration.md for the exact append blocks.
src/core/EventBus.js — append under // === Multiplayer === banner:
// === Multiplayer ===
NETWORK_CONNECTED: 'network:connected',
NETWORK_DISCONNECTED: 'network:disconnected',
NETWORK_PLAYER_JOINED: 'network:player-joined',
NETWORK_PLAYER_LEFT: 'network:player-left',
NETWORK_STATE_RECEIVED: 'network:state-received',
MULTIPLAYER_JOIN_ROOM: 'multiplayer:join-room',
MULTIPLAYER_LEAVE_ROOM: 'multiplayer:leave-room',
src/core/GameState.js — append a multiplayer field with persistent (roomId, playerId) and transient (connected, remotePlayers) parts. Update reset() to clear only the transient parts so rejoin works after a game restart.
src/core/Constants.js — append a MULTIPLAYER block with SERVER_URL (filled by Step 6), DEFAULT_ROOM, MAX_PLAYERS, TICK_RATE_HZ, reconnect backoff, stale-player threshold, PROTOCOL_VERSION. No magic numbers — every value is a named constant.
src/main.js — instantiate NetworkManager after EventBus + GameState, before the engine starts. Expose window.__NETWORK_MANAGER__ for tests. Extend window.render_game_to_text() to additively include multiplayer: {...} and remotePlayers: [...].
Inspect existing events. The wiring depends on mode:
realtime: NetworkManager owns a setInterval at TICK_RATE_HZ. Each tick it reads the local entity from GameState and calls client.send({type: 'state', payload: {...}}). No EventBus subscription needed — it just samples GameState. Add a single network:state-received listener in the relevant scene/system that calls RemotePlayerRegistry.upsert() and triggers a re-render.
turn-based: NetworkManager subscribes to the game's existing move events (e.g., card:played, move:submitted) and forwards them. The scene/system listens for network:state-received and applies the validated remote move. Local optimistic UI is allowed, but the server is the source of truth.
In Phaser games, remote-player rendering happens in the active GameScene — instantiate sprites on network:player-joined, update positions on network:state-received, destroy on network:player-left. In Three.js games, the active orchestrator (Game.js) creates and updates remote-player meshes.
See client-integration.md for example scene patches for both engines.
Run the dev server first to confirm everything works locally:
cd <game-path>/multiplayer-server && npx partykit dev
This starts a local CF Worker emulator on http://127.0.0.1:1999. In another terminal, set VITE_MULTIPLAYER_SERVER_URL=http://127.0.0.1:1999 in <game-path>/.env and run the client (cd <game-path> && npm run dev).
For first-time deployment, the user must authenticate with PartyKit. Always pass --provider github — the default clerk flow is broken in 2026 (the dashboard.partykit.io callback was retired after Cloudflare absorbed PartyKit, and login hangs forever):
cd <game-path>/multiplayer-server && npx partykit login --provider github
This uses GitHub's device-code OAuth flow. The CLI prints a code; the user visits https://github.com/login/device, pastes it, and authorizes. Credentials persist in ~/.partykit/config.json. See deploy.md for the full walkthrough and troubleshooting.
After login, deploy:
cd <game-path>/multiplayer-server && npx partykit deploy
Capture the deployed URL from the output (format: https://<project>.<cloudflare-username>.partykit.dev). The TLS cert may take 30-60 seconds to provision after the deploy reports success.
Update three places with the deployed URL:
src/core/Constants.js → MULTIPLAYER.SERVER_URL<game-path>/.env → VITE_MULTIPLAYER_SERVER_URL=https://...<game-path>/.env.example → VITE_MULTIPLAYER_SERVER_URL=https://your-project.your-username.partykit.devAdd .env to .gitignore if not already present.
See deploy.md for the full walkthrough including offline-first authentication and troubleshooting.
Reuse the existing host detection logic (same as monetize-game Step 5):
.herenow/state.json exists → redeploy via ~/.agents/skills/here-now/scripts/publish.sh dist/.gh is configured and the repo has a GitHub Pages workflow → npx gh-pages -d dist.vercel is configured → vercel --prod.Always run npm run build first.
Build cleanly:
cd <game-path> && npm run build
cd multiplayer-server && npm run build # if a build script exists
Single-player fallback (critical): with the partykit dev server stopped, reload http://localhost:3000. The game must boot, play, and reset normally. Confirm network:disconnected fired and no uncaught errors in the console. If the game depends on the server to start, you violated Principle 1 — revise.
Two-tab smoke test: start npx partykit dev in one terminal and npm run dev in another. Open two browser tabs at http://localhost:3000. Confirm:
network:connected (check console).window.render_game_to_text() includes the other tab in remotePlayers.1000 / TICK_RATE_HZ * 2 ms.Reconnect: kill the partykit dev server, wait, restart it. The client should reconnect within RECONNECT_MAX_BACKOFF_MS and re-emit network:connected.
Regression: existing tests/e2e/*.spec.js must still pass. Single-player invariants (boot, score, game-over, reset) must hold whether the server is up or down.
progress.mdAppend a ## Multiplayer section:
## Multiplayer
- **Backend**: PartyKit (Cloudflare Durable Objects)
- **Server URL**: https://<project>.<user>.partykit.dev
- **Mode**: realtime | turn-based
- **Max players per room**: 4
- **Tick rate**: 20 Hz (realtime mode)
- **Default room**: lobby
- **Known limitations (v1)**: no matchmaking UI, no spectator mode, no persistent accounts, server-side rate limiting only.
Tell the user:
multiplayer-server/, client in src/multiplayer/ + src/systems/NetworkManager.js, additive edits to four core files.https://<project>.<user>.partykit.dev. Already wired into Constants and .env.cd multiplayer-server && npx partykit dev then npm run dev, open two tabs.src/multiplayer/MultiplayerClient.js. Future Colyseus or fly.io migration only changes that one file./add-multiplayer
Result: detects engine, scaffolds multiplayer-server/ with the realtime template, creates client networking files, deploys server, redeploys client, prints play URL and server URL.
/add-multiplayer ./examples/card-game --mode=turn-based
Result: uses the turn-based server template; NetworkManager forwards the game's existing move events instead of running a position-broadcast tick.
/add-multiplayer --dry-run
Result: prints the full file list and patches without writing or deploying. Useful for review before committing.
npx partykit login redirects to dashboard.partykit.io/patience and never completesCause: The default clerk provider was retired after Cloudflare absorbed PartyKit; the dashboard the OAuth callback expects is gone.
Fix: Use npx partykit login --provider github instead — GitHub device-code flow, prints a code, you paste at https://github.com/login/device. Credentials persist in ~/.partykit/config.json.
Cause: Welcome-race — the WebSocket welcome arrived before the scene's create() registered its NETWORK_PLAYER_JOINED listener. The events fired into the void.
Fix: After registering the listener, seed from gameState.multiplayer.remotePlayers directly. See client-integration.md → "Welcome-race gotcha" for the idempotent pattern.
Cause: They joined different rooms (random room IDs from URL parsing) or the server's broadcast logic excludes the sender by default.
Fix: Check the room id in window.render_game_to_text().multiplayer.roomId on both tabs — it should be the same (default 'lobby'). If different, audit NetworkManager.connect() for stray query-string parsing. The server template's room.broadcast(message, [sender.id]) excludes the sender, which is correct — each client renders only remote players, not itself.
Cause: onClose did not fire (browser killed the tab without a clean close), or the client did not run RemotePlayerRegistry.prune().
Fix: The server's onClose handler is the canonical "player left" signal. Additionally, NetworkManager runs RemotePlayerRegistry.prune(STALE_PLAYER_MS) on every tick — verify this is wired. If a remote player has not sent state in STALE_PLAYER_MS, prune emits network:player-left even without an explicit close.
Cause: Either too-high TICK_RATE_HZ (you're broadcasting and rendering 60 times per second) or the scene re-creates remote-player sprites every frame instead of reusing them.
Fix: Lower TICK_RATE_HZ to 20 (default) or 10 for slow games. Confirm scenes maintain a Map<playerId, sprite> and only update positions on network:state-received, never recreate.
Cause: NetworkManager throws or blocks game boot when the server is unreachable. This violates Principle 1.
Fix: Audit MultiplayerClient.connect() and NetworkManager.init() — both must catch all errors, log a warning, emit network:disconnected, and return. The constructor and init() must never throw out of main.js.
render_game_to_text() snapshot tests failCause: Tests use exact toEqual on the output; you added new top-level fields.
Fix: Regenerate baselines — additions are intentional and backward-compatible. The fields added are multiplayer (object) and remotePlayers (array, may be empty).
Cause: Mixed content (HTTP page → WSS server) or the server URL was written without the https:// scheme.
Fix: Confirm Constants.MULTIPLAYER.SERVER_URL is the full https://...partykit.dev URL. partysocket derives the WSS URL by replacing the scheme. The deployed game must also be served over HTTPS for the WSS connection to succeed (here.now and GitHub Pages both serve HTTPS by default).
Cause: Many concurrent rooms or chatty clients.
Fix: Cloudflare's Workers free tier allows 100k requests/day. Each client tick at 20 Hz is one request — that's 1.7M / day for a single 24/7 player. For prototyping you'll never hit this; for production, lower TICK_RATE_HZ or upgrade to Workers Paid ($5/mo flat for 10M requests/day).
Run
/add-multiplayeronce per game. If you later change modes, editmultiplayer-server/src/server.tsdirectly — both templates are checked in and the switch is small.The default
'lobby'room is suitable for a single open room. To support private rooms, emitmultiplayer:join-roomwith a room id from a URL query string or invite code. NetworkManager listens for that event and reconnects to the new room.For graduation to a more featureful backend (matchmaking, schema sync, server-authoritative physics), the only file that needs to change is
src/multiplayer/MultiplayerClient.js. Replace thepartysocketcalls with Colyseus'scolyseus.jsclient; keep the same public API (connect,send,onMessage,disconnect,isConnected).The server runs in your Cloudflare account, not OpusGameLabs'. Costs and quotas accrue to you. PartyKit itself is open source and free; you only pay Cloudflare's pass-through pricing (free tier is generous).