fix: websocket

This commit is contained in:
cha0s 2024-11-04 16:27:31 -06:00
parent 2e6e339d6d
commit 626df5b912
8 changed files with 132 additions and 145 deletions

View File

@ -63,6 +63,7 @@ module.exports = {
files: [
'.eslintrc.cjs',
'server.js',
'app/websocket/**',
'**/.server/**',
'*.server.{js,jsx}',
'**/{build,node}.js',

View File

@ -233,20 +233,6 @@ function Ui({disconnected}) {
mainEntityRef,
]);
usePacket('EcsChange', onEcsChangePacket);
const onTickPacket = useCallback((payload, client) => {
if (!ecsRef.current || 0 === Object.keys(payload.ecs).length) {
return;
}
try {
ecsRef.current.apply(payload.ecs);
client.emitter.invoke(':Ecs', payload.ecs);
}
catch (error) {
ecsRef.current = undefined;
console.error('tick crash', error);
}
}, [ecsRef]);
usePacket('Tick', onTickPacket);
const onEcsTick = useCallback((payload, ecs) => {
for (const id in payload) {
const entity = ecs.get(id);

View File

@ -18,12 +18,13 @@ export default function PlaySpecific() {
const debugTuple = useState(false);
const [Components, setComponents] = useState();
const [Systems, setSystems] = useState();
const reconnectionBackoff = useRef(0);
const ecsRef = useRef();
const [disconnected, setDisconnected] = useState(false);
const params = useParams();
const [type, url] = params['*'].split('/');
useEffect(() => {
if (!Client) {
if (!Client || !Components || !Systems) {
return;
}
const client = new Client();
@ -35,7 +36,7 @@ export default function PlaySpecific() {
return () => {
client.disconnect();
};
}, [Client, url]);
}, [Client, Components, Systems, url]);
// Sneakily use beforeunload to snag some time to save.
useEffect(() => {
if ('local' !== type) {
@ -84,19 +85,52 @@ export default function PlaySpecific() {
client.removePacketListener('EcsChange', onEcsChangePacket);
};
}, [client, refreshEcs]);
const onTickPacket = useCallback((payload) => {
if (!ecsRef.current || 0 === Object.keys(payload.ecs).length) {
return;
}
reconnectionBackoff.current = 0;
setDisconnected(false);
try {
ecsRef.current.apply(payload.ecs);
client.emitter.invoke(':Ecs', payload.ecs);
}
catch (error) {
ecsRef.current = undefined;
console.error('tick crash', error);
}
}, [client, ecsRef]);
useEffect(() => {
if (!client) {
return;
}
client.addPacketListener('Tick', onTickPacket);
return () => {
client.removePacketListener('Tick', onTickPacket);
};
}, [client, onTickPacket]);
useEffect(() => {
if (!client) {
return;
}
let handle;
function onConnectionStatus(status) {
switch (status) {
case 'aborted': {
if (!handle) {
client.disconnect();
setDisconnected(true);
break;
reconnectionBackoff.current = (
Math.max(100, Math.min(1000, reconnectionBackoff.current * 2))
);
handle = setTimeout(
() => {
client.connect(url);
handle = null;
},
reconnectionBackoff.current,
);
}
case 'connected': {
setDisconnected(false);
break;
}
}
@ -104,8 +138,9 @@ export default function PlaySpecific() {
client.addPacketListener('ConnectionStatus', onConnectionStatus);
return () => {
client.removePacketListener('ConnectionStatus', onConnectionStatus);
clearTimeout(handle);
};
}, [client]);
}, [client, url]);
useEffect(() => {
if (!client) {
return;
@ -125,15 +160,6 @@ export default function PlaySpecific() {
client.removePacketListener('Download', onDownload);
};
}, [client]);
useEffect(() => {
if (!client || !disconnected) {
return;
}
async function reconnect() {
await client.connect(url);
}
reconnect();
}, [client, disconnected, mainEntityRef, url]);
// useEffect(() => {
// let source = true;
// async function play() {

View File

@ -31,6 +31,10 @@ export default class LocalClient extends Client {
new URL('./predictor.js', import.meta.url),
{type: 'module'},
);
// loaded
await new Promise((resolve) => {
this.predictor.addEventListener('message', resolve, {once: true});
});
this.predictor.addEventListener('message', (event) => {
const [flow, packet] = event.data;
switch (flow) {
@ -71,6 +75,9 @@ export default class LocalClient extends Client {
if (CLIENT_INTERPOLATION) {
this.interpolator.terminate();
}
if (CLIENT_PREDICTION) {
this.predictor.terminate();
}
}
transmit(packet) {
if (CLIENT_PREDICTION) {

View File

@ -161,3 +161,6 @@ onmessage = (event) => {
}
}
};
// sync with parent
postMessage(null);

View File

@ -7,17 +7,19 @@ export default class RemoteClient extends Client {
interpolator = null;
predictor = null;
async connect(host) {
this.interpolator = new Worker(
new URL('./interpolator.js', import.meta.url),
{type: 'module'},
);
if (CLIENT_INTERPOLATION) {
this.interpolator = new Worker(
new URL('./interpolator.js', import.meta.url),
{type: 'module'},
);
this.interpolator.addEventListener('message', (event) => {
this.accept(event.data);
const packet = event.data;
if (CLIENT_PREDICTION) {
this.predictor.postMessage([1, packet]);
}
else {
this.accept(packet);
}
});
}
if (CLIENT_PREDICTION) {
@ -25,6 +27,10 @@ export default class RemoteClient extends Client {
new URL('./predictor.js', import.meta.url),
{type: 'module'},
);
// loaded
await new Promise((resolve) => {
this.predictor.addEventListener('message', resolve, {once: true});
});
this.predictor.addEventListener('message', (event) => {
const [flow, packet] = event.data;
switch (flow) {
@ -35,29 +41,23 @@ export default class RemoteClient extends Client {
break;
}
case 1: {
if (CLIENT_INTERPOLATION) {
this.interpolator.postMessage(packet);
}
else {
this.accept(packet);
}
break;
}
}
});
}
const url = new URL(`wss://${host}/ws`)
this.socket = new WebSocket(url.href);
this.socket = new WebSocket(`//${host}/silphius`);
this.socket.binaryType = 'arraybuffer';
this.socket.addEventListener('message', (event) => {
this.throughput.$$down += event.data.byteLength;
const packet = decode(event.data);
if (CLIENT_PREDICTION) {
this.predictor.postMessage([1, packet]);
}
else if (CLIENT_INTERPOLATION) {
if (CLIENT_INTERPOLATION) {
this.interpolator.postMessage(packet);
}
else if (CLIENT_PREDICTION) {
this.predictor.postMessage([1, packet]);
}
else {
this.accept(packet);
}
@ -68,12 +68,17 @@ export default class RemoteClient extends Client {
this.socket.addEventListener('error', () => {
this.accept({type: 'ConnectionStatus', payload: 'aborted'});
});
this.socket.addEventListener('open', () => {
this.accept({type: 'ConnectionStatus', payload: 'connected'});
});
}
disconnect() {
if (CLIENT_INTERPOLATION) {
this.interpolator.terminate();
}
if (CLIENT_PREDICTION) {
this.predictor.terminate();
}
}
transmit(packet) {
if (CLIENT_PREDICTION) {

View File

@ -1,95 +0,0 @@
import {mkdir, readFile, unlink, writeFile} from 'node:fs/promises';
import {dirname, join} from 'node:path';
import {WebSocketServer} from 'ws';
import Server from '@/silphius/net/server.js';
import {getSession} from '@/silphius/server/session.server.js';
import {loadResources, readAsset} from '@/lib/resources.js';
import {loadResources as loadServerResources} from '@/lib/resources.server.js';
import Engine from './engine.js';
global.__silphiusWebsocket = null;
class SocketServer extends Server {
async ensurePath(path) {
await mkdir(path, {recursive: true});
}
async load() {
await loadResources(await loadServerResources());
}
static qualify(path) {
return join(process.cwd(), 'data', 'remote', 'UNIVERSE', path);
}
readAsset(path) {
return readAsset(path) ?? new ArrayBuffer(0);
}
async readData(path) {
const qualified = this.constructor.qualify(path);
await this.ensurePath(dirname(qualified));
return readFile(qualified);
}
async removeData(path) {
await unlink(path);
}
async writeData(path, view) {
const qualified = this.constructor.qualify(path);
await this.ensurePath(dirname(qualified));
await writeFile(qualified, view);
}
transmit(ws, packed) { ws.send(packed); }
}
export async function handleUpgrade(request, socket, head) {
if (!global.__silphiusWebsocket) {
const engine = new Engine(SocketServer);
await engine.load();
engine.start();
const handleConnection = async (ws, request) => {
ws.on('close', async () => {
await engine.disconnectPlayer(ws);
})
ws.on('message', (packed) => {
engine.server.accept(ws, new DataView(packed.buffer, packed.byteOffset, packed.length));
});
const session = await getSession(request.headers['cookie']);
await engine.connectPlayer(ws, session.get('id'));
};
const wss = new WebSocketServer({
noServer: true,
});
wss.on('connection', handleConnection);
global.__silphiusWebsocket = {engine, handleConnection, wss};
}
const {pathname} = new URL(request.url, 'wss://base.url');
if (pathname === '/ws') {
const {wss} = global.__silphiusWebsocket;
wss.handleUpgrade(request, socket, head, function done(ws) {
wss.emit('connection', ws, request);
});
}
else {
socket.destroy();
}
}
if (import.meta.hot) {
import.meta.hot.on('vite:beforeUpdate', async () => {
if (global.__silphiusWebsocket) {
const {engine, handleConnection, wss} = global.__silphiusWebsocket;
wss.off('connection', handleConnection);
const connections = [];
for (const [connection] of engine.connectedPlayers) {
engine.server.send(connection, {type: 'EcsChange'});
connections.push(connection);
}
await engine.stop();
for (const connection of connections) {
connection.close();
}
global.__silphiusWebsocket = null;
}
});
import.meta.hot.accept();
}

54
app/websocket/silphius.js Normal file
View File

@ -0,0 +1,54 @@
import {mkdir, readFile, unlink, writeFile} from 'node:fs/promises';
import {dirname, join} from 'node:path';
import Server from '@/silphius/net/server.js';
import Engine from '@/silphius/server/engine.js';
import {getSession} from '@/silphius/server/session.server.js';
import {loadResources, readAsset} from '@/lib/resources.js';
import {loadResources as loadServerResources} from '@/lib/resources.server.js';
class SocketServer extends Server {
async ensurePath(path) {
await mkdir(path, {recursive: true});
}
async load() {
await loadResources(await loadServerResources());
}
static qualify(path) {
return join(process.cwd(), 'data', 'remote', 'UNIVERSE', path);
}
readAsset(path) {
return readAsset(path) ?? new ArrayBuffer(0);
}
async readData(path) {
const qualified = this.constructor.qualify(path);
await this.ensurePath(dirname(qualified));
return readFile(qualified);
}
async removeData(path) {
await unlink(path);
}
async writeData(path, view) {
const qualified = this.constructor.qualify(path);
await this.ensurePath(dirname(qualified));
await writeFile(qualified, view);
}
transmit(ws, packed) {
ws.send(packed);
}
}
const engine = new Engine(SocketServer);
await engine.load();
engine.start();
export async function handleConnection(websocket, request) {
websocket.on('close', async () => {
await engine.disconnectPlayer(websocket);
})
websocket.on('message', (packed) => {
engine.server.accept(websocket, new DataView(packed.buffer, packed.byteOffset, packed.length));
});
const session = await getSession(request.headers['cookie']);
await engine.connectPlayer(websocket, session.get('id'));
}