@rapierphysicsplugin/server 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (78) hide show
  1. package/dist/__tests__/input-buffer.test.d.ts +2 -0
  2. package/dist/__tests__/input-buffer.test.d.ts.map +1 -0
  3. package/dist/__tests__/input-buffer.test.js +53 -0
  4. package/dist/__tests__/input-buffer.test.js.map +1 -0
  5. package/dist/__tests__/integration.test.d.ts +2 -0
  6. package/dist/__tests__/integration.test.d.ts.map +1 -0
  7. package/dist/__tests__/integration.test.js +182 -0
  8. package/dist/__tests__/integration.test.js.map +1 -0
  9. package/dist/__tests__/physics-world-collisions.test.d.ts +2 -0
  10. package/dist/__tests__/physics-world-collisions.test.d.ts.map +1 -0
  11. package/dist/__tests__/physics-world-collisions.test.js +129 -0
  12. package/dist/__tests__/physics-world-collisions.test.js.map +1 -0
  13. package/dist/__tests__/physics-world.test.d.ts +2 -0
  14. package/dist/__tests__/physics-world.test.d.ts.map +1 -0
  15. package/dist/__tests__/physics-world.test.js +164 -0
  16. package/dist/__tests__/physics-world.test.js.map +1 -0
  17. package/dist/__tests__/room.test.d.ts +2 -0
  18. package/dist/__tests__/room.test.d.ts.map +1 -0
  19. package/dist/__tests__/room.test.js +189 -0
  20. package/dist/__tests__/room.test.js.map +1 -0
  21. package/dist/__tests__/state-manager.test.d.ts +2 -0
  22. package/dist/__tests__/state-manager.test.d.ts.map +1 -0
  23. package/dist/__tests__/state-manager.test.js +122 -0
  24. package/dist/__tests__/state-manager.test.js.map +1 -0
  25. package/dist/client-connection.d.ts +18 -0
  26. package/dist/client-connection.d.ts.map +1 -0
  27. package/dist/client-connection.js +41 -0
  28. package/dist/client-connection.js.map +1 -0
  29. package/dist/clock-sync.d.ts +4 -0
  30. package/dist/clock-sync.d.ts.map +1 -0
  31. package/dist/clock-sync.js +10 -0
  32. package/dist/clock-sync.js.map +1 -0
  33. package/dist/index.d.ts +8 -0
  34. package/dist/index.d.ts.map +1 -0
  35. package/dist/index.js +35 -0
  36. package/dist/index.js.map +1 -0
  37. package/dist/input-buffer.d.ts +11 -0
  38. package/dist/input-buffer.d.ts.map +1 -0
  39. package/dist/input-buffer.js +40 -0
  40. package/dist/input-buffer.js.map +1 -0
  41. package/dist/physics-world.d.ts +33 -0
  42. package/dist/physics-world.d.ts.map +1 -0
  43. package/dist/physics-world.js +326 -0
  44. package/dist/physics-world.js.map +1 -0
  45. package/dist/room.d.ts +60 -0
  46. package/dist/room.d.ts.map +1 -0
  47. package/dist/room.js +393 -0
  48. package/dist/room.js.map +1 -0
  49. package/dist/server.d.ts +17 -0
  50. package/dist/server.d.ts.map +1 -0
  51. package/dist/server.js +268 -0
  52. package/dist/server.js.map +1 -0
  53. package/dist/simulation-loop.d.ts +14 -0
  54. package/dist/simulation-loop.d.ts.map +1 -0
  55. package/dist/simulation-loop.js +42 -0
  56. package/dist/simulation-loop.js.map +1 -0
  57. package/dist/state-manager.d.ts +20 -0
  58. package/dist/state-manager.d.ts.map +1 -0
  59. package/dist/state-manager.js +120 -0
  60. package/dist/state-manager.js.map +1 -0
  61. package/package.json +24 -0
  62. package/src/__tests__/input-buffer.test.ts +64 -0
  63. package/src/__tests__/integration.test.ts +227 -0
  64. package/src/__tests__/physics-world-collisions.test.ts +155 -0
  65. package/src/__tests__/physics-world.test.ts +197 -0
  66. package/src/__tests__/room.test.ts +232 -0
  67. package/src/__tests__/state-manager.test.ts +152 -0
  68. package/src/client-connection.ts +52 -0
  69. package/src/clock-sync.ts +16 -0
  70. package/src/index.ts +40 -0
  71. package/src/input-buffer.ts +46 -0
  72. package/src/physics-world.ts +400 -0
  73. package/src/room.ts +487 -0
  74. package/src/server.ts +312 -0
  75. package/src/simulation-loop.ts +48 -0
  76. package/src/state-manager.ts +136 -0
  77. package/tsconfig.json +11 -0
  78. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,189 @@
1
+ import { describe, it, expect, beforeAll } from 'vitest';
2
+ import RAPIER from '@dimforge/rapier3d-compat';
3
+ import { Room, RoomManager } from '../room.js';
4
+ import { decodeMessage } from '@rapierphysicsplugin/shared';
5
+ describe('Room', () => {
6
+ let rapier;
7
+ beforeAll(async () => {
8
+ await RAPIER.init();
9
+ rapier = RAPIER;
10
+ });
11
+ function makeBox(id, y = 5) {
12
+ return {
13
+ id,
14
+ shape: { type: 'box', params: { halfExtents: { x: 0.5, y: 0.5, z: 0.5 } } },
15
+ motionType: 'dynamic',
16
+ position: { x: 0, y, z: 0 },
17
+ rotation: { x: 0, y: 0, z: 0, w: 1 },
18
+ mass: 1.0,
19
+ };
20
+ }
21
+ it('should create a room with initial bodies', () => {
22
+ const room = new Room('test', rapier);
23
+ room.loadInitialState([makeBox('box1', 5), makeBox('box2', 10)]);
24
+ const snapshot = room.getSnapshot();
25
+ expect(snapshot.bodies).toHaveLength(2);
26
+ room.destroy();
27
+ });
28
+ it('should add and remove bodies', () => {
29
+ const room = new Room('test', rapier);
30
+ room.loadInitialState([makeBox('box1')]);
31
+ const snapshot1 = room.getSnapshot();
32
+ expect(snapshot1.bodies).toHaveLength(1);
33
+ room.destroy();
34
+ });
35
+ it('should step physics and advance tick', () => {
36
+ const room = new Room('test', rapier);
37
+ room.loadInitialState([makeBox('box1', 10)]);
38
+ expect(room.tickNumber).toBe(0);
39
+ room.tick();
40
+ expect(room.tickNumber).toBe(1);
41
+ room.tick();
42
+ expect(room.tickNumber).toBe(2);
43
+ room.destroy();
44
+ });
45
+ it('should process buffered inputs during tick', () => {
46
+ const room = new Room('test', rapier);
47
+ room.loadInitialState([makeBox('box1', 5)]);
48
+ // Simulate a client connection by directly buffering input
49
+ // We need to add a mock client first
50
+ const mockConn = {
51
+ id: 'client1',
52
+ roomId: null,
53
+ ws: { readyState: 1, OPEN: 1, send: () => { } },
54
+ send: () => { },
55
+ rtt: 0,
56
+ clockOffset: 0,
57
+ lastAcknowledgedTick: 0,
58
+ inputSequence: 0,
59
+ updateClockSync: () => { },
60
+ mapClientTickToServerTick: () => 0,
61
+ };
62
+ room.addClient(mockConn);
63
+ room.bufferInput('client1', {
64
+ tick: 0,
65
+ sequenceNum: 0,
66
+ actions: [
67
+ { type: 'applyImpulse', bodyId: 'box1', data: { impulse: { x: 10, y: 0, z: 0 } } },
68
+ ],
69
+ });
70
+ room.tick();
71
+ const state = room.getSnapshot();
72
+ const box1 = state.bodies.find(b => b.id === 'box1');
73
+ expect(box1.linVel.x).toBeGreaterThan(0);
74
+ room.destroy();
75
+ });
76
+ it('should not auto-start simulation when a client joins', () => {
77
+ const room = new Room('test', rapier);
78
+ room.loadInitialState([makeBox('box1', 5)]);
79
+ const mockConn = {
80
+ id: 'client1',
81
+ roomId: null,
82
+ ws: { readyState: 1, OPEN: 1, send: () => { } },
83
+ send: () => { },
84
+ rtt: 0,
85
+ clockOffset: 0,
86
+ lastAcknowledgedTick: 0,
87
+ inputSequence: 0,
88
+ updateClockSync: () => { },
89
+ mapClientTickToServerTick: () => 0,
90
+ };
91
+ room.addClient(mockConn);
92
+ expect(room.isSimulationRunning).toBe(false);
93
+ room.destroy();
94
+ });
95
+ it('should start simulation on startSimulation call', () => {
96
+ const room = new Room('test', rapier);
97
+ room.loadInitialState([makeBox('box1', 5)]);
98
+ const sentMessages = [];
99
+ const mockConn = {
100
+ id: 'client1',
101
+ roomId: null,
102
+ ws: { readyState: 1, OPEN: 1, send: () => { } },
103
+ send: (msg) => { sentMessages.push(decodeMessage(msg)); },
104
+ rtt: 0,
105
+ clockOffset: 0,
106
+ lastAcknowledgedTick: 0,
107
+ inputSequence: 0,
108
+ updateClockSync: () => { },
109
+ mapClientTickToServerTick: () => 0,
110
+ };
111
+ room.addClient(mockConn);
112
+ expect(room.isSimulationRunning).toBe(false);
113
+ room.startSimulation();
114
+ expect(room.isSimulationRunning).toBe(true);
115
+ // Should have broadcast SIMULATION_STARTED
116
+ const startedMsg = sentMessages.find(m => m.type === 'simulation_started');
117
+ expect(startedMsg).toBeDefined();
118
+ room.destroy();
119
+ });
120
+ it('should reset physics on startSimulation when already running', () => {
121
+ const room = new Room('test', rapier);
122
+ room.loadInitialState([makeBox('box1', 10)]);
123
+ const mockConn = {
124
+ id: 'client1',
125
+ roomId: null,
126
+ ws: { readyState: 1, OPEN: 1, send: () => { } },
127
+ send: () => { },
128
+ rtt: 0,
129
+ clockOffset: 0,
130
+ lastAcknowledgedTick: 0,
131
+ inputSequence: 0,
132
+ updateClockSync: () => { },
133
+ mapClientTickToServerTick: () => 0,
134
+ };
135
+ room.addClient(mockConn);
136
+ room.startSimulation();
137
+ // Run some ticks so physics changes
138
+ for (let i = 0; i < 60; i++) {
139
+ room.tick();
140
+ }
141
+ expect(room.tickNumber).toBeGreaterThan(0);
142
+ const stateBefore = room.getSnapshot();
143
+ const box = stateBefore.bodies.find(b => b.id === 'box1');
144
+ expect(box.position.y).not.toBeCloseTo(10);
145
+ // Reset
146
+ room.startSimulation();
147
+ expect(room.tickNumber).toBe(0);
148
+ const stateAfter = room.getSnapshot();
149
+ const boxAfter = stateAfter.bodies.find(b => b.id === 'box1');
150
+ expect(boxAfter.position.y).toBeCloseTo(10);
151
+ room.destroy();
152
+ });
153
+ });
154
+ describe('RoomManager', () => {
155
+ let rapier;
156
+ beforeAll(async () => {
157
+ await RAPIER.init();
158
+ rapier = RAPIER;
159
+ });
160
+ it('should create and retrieve rooms', () => {
161
+ const manager = new RoomManager(rapier);
162
+ manager.createRoom('room1');
163
+ expect(manager.getRoom('room1')).toBeDefined();
164
+ expect(manager.roomCount).toBe(1);
165
+ manager.destroyRoom('room1');
166
+ });
167
+ it('should throw on duplicate room creation', () => {
168
+ const manager = new RoomManager(rapier);
169
+ manager.createRoom('room1');
170
+ expect(() => manager.createRoom('room1')).toThrow('already exists');
171
+ manager.destroyRoom('room1');
172
+ });
173
+ it('should destroy rooms', () => {
174
+ const manager = new RoomManager(rapier);
175
+ manager.createRoom('room1');
176
+ manager.destroyRoom('room1');
177
+ expect(manager.getRoom('room1')).toBeUndefined();
178
+ expect(manager.roomCount).toBe(0);
179
+ });
180
+ it('should list all room ids', () => {
181
+ const manager = new RoomManager(rapier);
182
+ manager.createRoom('room1');
183
+ manager.createRoom('room2');
184
+ expect(manager.getAllRoomIds().sort()).toEqual(['room1', 'room2']);
185
+ manager.destroyRoom('room1');
186
+ manager.destroyRoom('room2');
187
+ });
188
+ });
189
+ //# sourceMappingURL=room.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"room.test.js","sourceRoot":"","sources":["../../src/__tests__/room.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=state-manager.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-manager.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/state-manager.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,122 @@
1
+ import { describe, it, expect, beforeAll, beforeEach, afterEach } from 'vitest';
2
+ import RAPIER from '@dimforge/rapier3d-compat';
3
+ import { PhysicsWorld } from '../physics-world.js';
4
+ import { StateManager } from '../state-manager.js';
5
+ import { FIELD_POSITION, FIELD_LIN_VEL, FIELD_ALL } from '@rapierphysicsplugin/shared';
6
+ describe('StateManager', () => {
7
+ let world;
8
+ let stateManager;
9
+ beforeAll(async () => {
10
+ await RAPIER.init();
11
+ });
12
+ beforeEach(() => {
13
+ world = new PhysicsWorld(RAPIER);
14
+ stateManager = new StateManager();
15
+ });
16
+ afterEach(() => {
17
+ world.destroy();
18
+ });
19
+ function makeBox(id, y = 5) {
20
+ return {
21
+ id,
22
+ shape: { type: 'box', params: { halfExtents: { x: 0.5, y: 0.5, z: 0.5 } } },
23
+ motionType: 'dynamic',
24
+ position: { x: 0, y, z: 0 },
25
+ rotation: { x: 0, y: 0, z: 0, w: 1 },
26
+ mass: 1.0,
27
+ };
28
+ }
29
+ function makeStaticBox(id, y = 0) {
30
+ return {
31
+ id,
32
+ shape: { type: 'box', params: { halfExtents: { x: 5, y: 0.5, z: 5 } } },
33
+ motionType: 'static',
34
+ position: { x: 0, y, z: 0 },
35
+ rotation: { x: 0, y: 0, z: 0, w: 1 },
36
+ };
37
+ }
38
+ it('should create a full snapshot', () => {
39
+ world.addBody(makeBox('box1', 5));
40
+ world.addBody(makeBox('box2', 10));
41
+ const snapshot = stateManager.createSnapshot(world, 42);
42
+ expect(snapshot.tick).toBe(42);
43
+ expect(snapshot.bodies).toHaveLength(2);
44
+ expect(snapshot.timestamp).toBeGreaterThan(0);
45
+ });
46
+ it('should create delta with all bodies on first call (fieldMask = FIELD_ALL)', () => {
47
+ world.addBody(makeBox('box1', 5));
48
+ const delta = stateManager.createDelta(world, 1);
49
+ expect(delta.bodies).toHaveLength(1);
50
+ expect(delta.bodies[0].fieldMask).toBe(FIELD_ALL);
51
+ expect(delta.isDelta).toBe(true);
52
+ });
53
+ it('should create empty delta when nothing changed', () => {
54
+ world.addBody(makeBox('box1', 5));
55
+ // First delta includes everything
56
+ stateManager.createDelta(world, 1);
57
+ // No stepping, nothing changed — delta should be empty
58
+ const delta2 = stateManager.createDelta(world, 2);
59
+ expect(delta2.bodies).toHaveLength(0);
60
+ });
61
+ it('should detect changes after physics step', () => {
62
+ world.addBody(makeBox('box1', 5));
63
+ // First delta
64
+ stateManager.createDelta(world, 1);
65
+ // Step physics — body should fall
66
+ for (let i = 0; i < 10; i++) {
67
+ world.step();
68
+ }
69
+ const delta2 = stateManager.createDelta(world, 2);
70
+ expect(delta2.bodies).toHaveLength(1);
71
+ expect(delta2.bodies[0].id).toBe('box1');
72
+ });
73
+ it('should set per-field mask for position+linVel change (falling body)', () => {
74
+ world.addBody(makeBox('box1', 5));
75
+ // First delta
76
+ stateManager.createDelta(world, 1);
77
+ // Step physics — body falls (position and linVel change, rotation stays identity)
78
+ for (let i = 0; i < 10; i++) {
79
+ world.step();
80
+ }
81
+ const delta2 = stateManager.createDelta(world, 2);
82
+ expect(delta2.bodies).toHaveLength(1);
83
+ const mask = delta2.bodies[0].fieldMask;
84
+ // A falling body should have position and linVel changed
85
+ expect(mask & FIELD_POSITION).toBeTruthy();
86
+ expect(mask & FIELD_LIN_VEL).toBeTruthy();
87
+ });
88
+ it('should assign body indices', () => {
89
+ world.addBody(makeBox('box1', 5));
90
+ world.addBody(makeBox('box2', 10));
91
+ stateManager.createSnapshot(world, 0);
92
+ expect(stateManager.getBodyIndex('box1')).toBe(0);
93
+ expect(stateManager.getBodyIndex('box2')).toBe(1);
94
+ });
95
+ it('should produce ID mapping record', () => {
96
+ world.addBody(makeBox('box1', 5));
97
+ world.addBody(makeBox('box2', 10));
98
+ stateManager.createSnapshot(world, 0);
99
+ const record = stateManager.getIdToIndexRecord();
100
+ expect(record).toEqual({ box1: 0, box2: 1 });
101
+ });
102
+ it('should keep index after body removal (indices are never reused)', () => {
103
+ world.addBody(makeBox('box1', 5));
104
+ world.addBody(makeBox('box2', 10));
105
+ stateManager.createSnapshot(world, 0);
106
+ stateManager.removeBody('box1');
107
+ // box1 index is still in the map (not reused)
108
+ expect(stateManager.getBodyIndex('box1')).toBe(0);
109
+ // Adding a new body gets the next index
110
+ world.addBody(makeBox('box3', 15));
111
+ stateManager.ensureBodyIndex('box3');
112
+ expect(stateManager.getBodyIndex('box3')).toBe(2);
113
+ });
114
+ it('should clear all state on clear()', () => {
115
+ world.addBody(makeBox('box1', 5));
116
+ stateManager.createSnapshot(world, 0);
117
+ stateManager.clear();
118
+ expect(stateManager.getBodyIndex('box1')).toBeUndefined();
119
+ expect(stateManager.getIdToIndexRecord()).toEqual({});
120
+ });
121
+ });
122
+ //# sourceMappingURL=state-manager.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-manager.test.js","sourceRoot":"","sources":["../../src/__tests__/state-manager.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import type { WebSocket } from 'ws';
2
+ export declare class ClientConnection {
3
+ readonly id: string;
4
+ readonly ws: WebSocket;
5
+ roomId: string | null;
6
+ rtt: number;
7
+ clockOffset: number;
8
+ lastAcknowledgedTick: number;
9
+ inputSequence: number;
10
+ private rttSamples;
11
+ private offsetSamples;
12
+ private static readonly MAX_SAMPLES;
13
+ constructor(id: string, ws: WebSocket);
14
+ send(data: Uint8Array): void;
15
+ updateClockSync(clientTimestamp: number, serverTimestamp: number): void;
16
+ mapClientTickToServerTick(clientTick: number, serverTickRate: number): number;
17
+ }
18
+ //# sourceMappingURL=client-connection.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client-connection.d.ts","sourceRoot":"","sources":["../src/client-connection.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AAEpC,qBAAa,gBAAgB;IAC3B,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,EAAE,EAAE,SAAS,CAAC;IACvB,MAAM,EAAE,MAAM,GAAG,IAAI,CAAQ;IAC7B,GAAG,SAAK;IACR,WAAW,SAAK;IAChB,oBAAoB,SAAK;IACzB,aAAa,SAAK;IAElB,OAAO,CAAC,UAAU,CAAgB;IAClC,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,WAAW,CAAM;gBAE7B,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,SAAS;IAKrC,IAAI,CAAC,IAAI,EAAE,UAAU,GAAG,IAAI;IAM5B,eAAe,CAAC,eAAe,EAAE,MAAM,EAAE,eAAe,EAAE,MAAM,GAAG,IAAI;IAoBvE,yBAAyB,CAAC,UAAU,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,GAAG,MAAM;CAK9E"}
@@ -0,0 +1,41 @@
1
+ export class ClientConnection {
2
+ constructor(id, ws) {
3
+ this.roomId = null;
4
+ this.rtt = 0;
5
+ this.clockOffset = 0;
6
+ this.lastAcknowledgedTick = 0;
7
+ this.inputSequence = 0;
8
+ this.rttSamples = [];
9
+ this.offsetSamples = [];
10
+ this.id = id;
11
+ this.ws = ws;
12
+ }
13
+ send(data) {
14
+ if (this.ws.readyState === this.ws.OPEN) {
15
+ this.ws.send(data);
16
+ }
17
+ }
18
+ updateClockSync(clientTimestamp, serverTimestamp) {
19
+ const now = Date.now();
20
+ const rtt = now - clientTimestamp;
21
+ this.rttSamples.push(rtt);
22
+ if (this.rttSamples.length > ClientConnection.MAX_SAMPLES) {
23
+ this.rttSamples.shift();
24
+ }
25
+ const offset = serverTimestamp - clientTimestamp - rtt / 2;
26
+ this.offsetSamples.push(offset);
27
+ if (this.offsetSamples.length > ClientConnection.MAX_SAMPLES) {
28
+ this.offsetSamples.shift();
29
+ }
30
+ // Rolling average
31
+ this.rtt = this.rttSamples.reduce((a, b) => a + b, 0) / this.rttSamples.length;
32
+ this.clockOffset = this.offsetSamples.reduce((a, b) => a + b, 0) / this.offsetSamples.length;
33
+ }
34
+ mapClientTickToServerTick(clientTick, serverTickRate) {
35
+ // Use clock offset to map client tick to server tick
36
+ const offsetInTicks = Math.round((this.clockOffset / 1000) * serverTickRate);
37
+ return clientTick + offsetInTicks;
38
+ }
39
+ }
40
+ ClientConnection.MAX_SAMPLES = 10;
41
+ //# sourceMappingURL=client-connection.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client-connection.js","sourceRoot":"","sources":["../src/client-connection.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,gBAAgB;IAa3B,YAAY,EAAU,EAAE,EAAa;QAVrC,WAAM,GAAkB,IAAI,CAAC;QAC7B,QAAG,GAAG,CAAC,CAAC;QACR,gBAAW,GAAG,CAAC,CAAC;QAChB,yBAAoB,GAAG,CAAC,CAAC;QACzB,kBAAa,GAAG,CAAC,CAAC;QAEV,eAAU,GAAa,EAAE,CAAC;QAC1B,kBAAa,GAAa,EAAE,CAAC;QAInC,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;QACb,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;IACf,CAAC;IAED,IAAI,CAAC,IAAgB;QACnB,IAAI,IAAI,CAAC,EAAE,CAAC,UAAU,KAAK,IAAI,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;YACxC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACrB,CAAC;IACH,CAAC;IAED,eAAe,CAAC,eAAuB,EAAE,eAAuB;QAC9D,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,GAAG,GAAG,GAAG,GAAG,eAAe,CAAC;QAElC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC1B,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,gBAAgB,CAAC,WAAW,EAAE,CAAC;YAC1D,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QAC1B,CAAC;QAED,MAAM,MAAM,GAAG,eAAe,GAAG,eAAe,GAAG,GAAG,GAAG,CAAC,CAAC;QAC3D,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAChC,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,gBAAgB,CAAC,WAAW,EAAE,CAAC;YAC7D,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;QAC7B,CAAC;QAED,kBAAkB;QAClB,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC;QAC/E,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC;IAC/F,CAAC;IAED,yBAAyB,CAAC,UAAkB,EAAE,cAAsB;QAClE,qDAAqD;QACrD,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,cAAc,CAAC,CAAC;QAC7E,OAAO,UAAU,GAAG,aAAa,CAAC;IACpC,CAAC;;AArCuB,4BAAW,GAAG,EAAE,AAAL,CAAM"}
@@ -0,0 +1,4 @@
1
+ import type { ClockSyncRequestMessage } from '@rapierphysicsplugin/shared';
2
+ import type { ClientConnection } from './client-connection.js';
3
+ export declare function handleClockSyncRequest(conn: ClientConnection, message: ClockSyncRequestMessage): void;
4
+ //# sourceMappingURL=clock-sync.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"clock-sync.d.ts","sourceRoot":"","sources":["../src/clock-sync.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,uBAAuB,EAA4B,MAAM,6BAA6B,CAAC;AAErG,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAE/D,wBAAgB,sBAAsB,CACpC,IAAI,EAAE,gBAAgB,EACtB,OAAO,EAAE,uBAAuB,GAC/B,IAAI,CAQN"}
@@ -0,0 +1,10 @@
1
+ import { MessageType, encodeMessage } from '@rapierphysicsplugin/shared';
2
+ export function handleClockSyncRequest(conn, message) {
3
+ const response = {
4
+ type: MessageType.CLOCK_SYNC_RESPONSE,
5
+ clientTimestamp: message.clientTimestamp,
6
+ serverTimestamp: Date.now(),
7
+ };
8
+ conn.send(encodeMessage(response));
9
+ }
10
+ //# sourceMappingURL=clock-sync.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"clock-sync.js","sourceRoot":"","sources":["../src/clock-sync.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,aAAa,EAAE,MAAM,6BAA6B,CAAC;AAGzE,MAAM,UAAU,sBAAsB,CACpC,IAAsB,EACtB,OAAgC;IAEhC,MAAM,QAAQ,GAA6B;QACzC,IAAI,EAAE,WAAW,CAAC,mBAAmB;QACrC,eAAe,EAAE,OAAO,CAAC,eAAe;QACxC,eAAe,EAAE,IAAI,CAAC,GAAG,EAAE;KAC5B,CAAC;IAEF,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC;AACrC,CAAC"}
@@ -0,0 +1,8 @@
1
+ export { PhysicsServer } from './server.js';
2
+ export { PhysicsWorld } from './physics-world.js';
3
+ export { Room, RoomManager } from './room.js';
4
+ export { ClientConnection } from './client-connection.js';
5
+ export { SimulationLoop } from './simulation-loop.js';
6
+ export { StateManager } from './state-manager.js';
7
+ export { InputBuffer } from './input-buffer.js';
8
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAC9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,35 @@
1
+ import { DEFAULT_PORT, ComputeBackend, loadRapier } from '@rapierphysicsplugin/shared';
2
+ import { PhysicsServer } from './server.js';
3
+ export { PhysicsServer } from './server.js';
4
+ export { PhysicsWorld } from './physics-world.js';
5
+ export { Room, RoomManager } from './room.js';
6
+ export { ClientConnection } from './client-connection.js';
7
+ export { SimulationLoop } from './simulation-loop.js';
8
+ export { StateManager } from './state-manager.js';
9
+ export { InputBuffer } from './input-buffer.js';
10
+ async function main() {
11
+ const backend = process.env.PHYSICS_BACKEND ?? ComputeBackend.WASM_SIMD;
12
+ console.log(`Initializing Rapier WASM (${backend})...`);
13
+ const RAPIER = await loadRapier({ backend });
14
+ console.log('Rapier initialized.');
15
+ const port = parseInt(process.env.PORT || String(DEFAULT_PORT), 10);
16
+ const server = new PhysicsServer(RAPIER);
17
+ await server.start(port);
18
+ // Graceful shutdown
19
+ process.on('SIGINT', () => {
20
+ console.log('\nShutting down...');
21
+ server.stop();
22
+ process.exit(0);
23
+ });
24
+ process.on('SIGTERM', () => {
25
+ console.log('\nShutting down...');
26
+ server.stop();
27
+ process.exit(0);
28
+ });
29
+ }
30
+ // Only run main when executed directly
31
+ const isMainModule = process.argv[1]?.endsWith('index.ts') || process.argv[1]?.endsWith('index.js');
32
+ if (isMainModule) {
33
+ main().catch(console.error);
34
+ }
35
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,UAAU,EAAE,MAAM,6BAA6B,CAAC;AACvF,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE5C,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAC9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,KAAK,UAAU,IAAI;IACjB,MAAM,OAAO,GAAI,OAAO,CAAC,GAAG,CAAC,eAAkC,IAAI,cAAc,CAAC,SAAS,CAAC;IAC5F,OAAO,CAAC,GAAG,CAAC,6BAA6B,OAAO,MAAM,CAAC,CAAC;IACxD,MAAM,MAAM,GAAG,MAAM,UAAU,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;IAC7C,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;IAEnC,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,MAAM,CAAC,YAAY,CAAC,EAAE,EAAE,CAAC,CAAC;IACpE,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,MAAM,CAAC,CAAC;IACzC,MAAM,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAEzB,oBAAoB;IACpB,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;QACxB,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;QAClC,MAAM,CAAC,IAAI,EAAE,CAAC;QACd,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;IAEH,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;QACzB,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;QAClC,MAAM,CAAC,IAAI,EAAE,CAAC;QACd,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;AACL,CAAC;AAED,uCAAuC;AACvC,MAAM,YAAY,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,UAAU,CAAC,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,UAAU,CAAC,CAAC;AACpG,IAAI,YAAY,EAAE,CAAC;IACjB,IAAI,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;AAC9B,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { ClientInput } from '@rapierphysicsplugin/shared';
2
+ export declare class InputBuffer {
3
+ private buffer;
4
+ private oldestTick;
5
+ addInput(input: ClientInput, serverTick: number): void;
6
+ getInputsForTick(tick: number): ClientInput[];
7
+ private pruneOldEntries;
8
+ clear(): void;
9
+ get size(): number;
10
+ }
11
+ //# sourceMappingURL=input-buffer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"input-buffer.d.ts","sourceRoot":"","sources":["../src/input-buffer.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,6BAA6B,CAAC;AAG/D,qBAAa,WAAW;IACtB,OAAO,CAAC,MAAM,CAAyC;IACvD,OAAO,CAAC,UAAU,CAAK;IAEvB,QAAQ,CAAC,KAAK,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,GAAG,IAAI;IAatD,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,WAAW,EAAE;IAS7C,OAAO,CAAC,eAAe;IASvB,KAAK,IAAI,IAAI;IAIb,IAAI,IAAI,IAAI,MAAM,CAEjB;CACF"}
@@ -0,0 +1,40 @@
1
+ import { MAX_INPUT_BUFFER } from '@rapierphysicsplugin/shared';
2
+ export class InputBuffer {
3
+ constructor() {
4
+ this.buffer = new Map();
5
+ this.oldestTick = 0;
6
+ }
7
+ addInput(input, serverTick) {
8
+ // Map client tick to server tick using provided mapping
9
+ const targetTick = serverTick;
10
+ if (!this.buffer.has(targetTick)) {
11
+ this.buffer.set(targetTick, []);
12
+ }
13
+ this.buffer.get(targetTick).push(input);
14
+ // Clean up old entries
15
+ this.pruneOldEntries(targetTick);
16
+ }
17
+ getInputsForTick(tick) {
18
+ const inputs = this.buffer.get(tick);
19
+ if (inputs) {
20
+ this.buffer.delete(tick);
21
+ return inputs;
22
+ }
23
+ return [];
24
+ }
25
+ pruneOldEntries(currentTick) {
26
+ const cutoff = currentTick - MAX_INPUT_BUFFER;
27
+ for (const tick of this.buffer.keys()) {
28
+ if (tick < cutoff) {
29
+ this.buffer.delete(tick);
30
+ }
31
+ }
32
+ }
33
+ clear() {
34
+ this.buffer.clear();
35
+ }
36
+ get size() {
37
+ return this.buffer.size;
38
+ }
39
+ }
40
+ //# sourceMappingURL=input-buffer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"input-buffer.js","sourceRoot":"","sources":["../src/input-buffer.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,MAAM,6BAA6B,CAAC;AAE/D,MAAM,OAAO,WAAW;IAAxB;QACU,WAAM,GAA+B,IAAI,GAAG,EAAE,CAAC;QAC/C,eAAU,GAAG,CAAC,CAAC;IAwCzB,CAAC;IAtCC,QAAQ,CAAC,KAAkB,EAAE,UAAkB;QAC7C,wDAAwD;QACxD,MAAM,UAAU,GAAG,UAAU,CAAC;QAE9B,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,CAAC;YACjC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;QAClC,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,UAAU,CAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAEzC,uBAAuB;QACvB,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;IACnC,CAAC;IAED,gBAAgB,CAAC,IAAY;QAC3B,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACrC,IAAI,MAAM,EAAE,CAAC;YACX,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YACzB,OAAO,MAAM,CAAC;QAChB,CAAC;QACD,OAAO,EAAE,CAAC;IACZ,CAAC;IAEO,eAAe,CAAC,WAAmB;QACzC,MAAM,MAAM,GAAG,WAAW,GAAG,gBAAgB,CAAC;QAC9C,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;YACtC,IAAI,IAAI,GAAG,MAAM,EAAE,CAAC;gBAClB,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC3B,CAAC;QACH,CAAC;IACH,CAAC;IAED,KAAK;QACH,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;IACtB,CAAC;IAED,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;IAC1B,CAAC;CACF"}
@@ -0,0 +1,33 @@
1
+ import type RAPIER from '@dimforge/rapier3d-compat';
2
+ import type { BodyDescriptor, BodyState, CollisionEventData, ConstraintDescriptor, Vec3, Quat, InputAction } from '@rapierphysicsplugin/shared';
3
+ export declare class PhysicsWorld {
4
+ private world;
5
+ private rapier;
6
+ private bodyMap;
7
+ private colliderMap;
8
+ private colliderHandleToBodyId;
9
+ private constraintMap;
10
+ private eventQueue;
11
+ constructor(rapier: typeof RAPIER, gravity?: Vec3);
12
+ addBody(descriptor: BodyDescriptor): string;
13
+ removeBody(id: string): void;
14
+ applyForce(id: string, force: Vec3, point?: Vec3): void;
15
+ applyImpulse(id: string, impulse: Vec3, point?: Vec3): void;
16
+ setBodyVelocity(id: string, linVel: Vec3, angVel?: Vec3): void;
17
+ setBodyPosition(id: string, position: Vec3): void;
18
+ setBodyRotation(id: string, rotation: Quat): void;
19
+ applyInput(action: InputAction): void;
20
+ addConstraint(descriptor: ConstraintDescriptor): string;
21
+ removeConstraint(id: string): void;
22
+ hasConstraint(id: string): boolean;
23
+ step(): CollisionEventData[];
24
+ getSnapshot(skipSleeping?: boolean): BodyState[];
25
+ isBodySleeping(id: string): boolean;
26
+ getBodyState(id: string): BodyState | null;
27
+ loadState(bodies: BodyDescriptor[]): void;
28
+ reset(bodies: BodyDescriptor[], constraints?: ConstraintDescriptor[]): void;
29
+ hasBody(id: string): boolean;
30
+ get bodyCount(): number;
31
+ destroy(): void;
32
+ }
33
+ //# sourceMappingURL=physics-world.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"physics-world.d.ts","sourceRoot":"","sources":["../src/physics-world.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,MAAM,2BAA2B,CAAC;AACpD,OAAO,KAAK,EACV,cAAc,EACd,SAAS,EACT,kBAAkB,EAClB,oBAAoB,EACpB,IAAI,EACJ,IAAI,EACJ,WAAW,EACZ,MAAM,6BAA6B,CAAC;AAGrC,qBAAa,YAAY;IACvB,OAAO,CAAC,KAAK,CAAe;IAC5B,OAAO,CAAC,MAAM,CAAgB;IAC9B,OAAO,CAAC,OAAO,CAA4C;IAC3D,OAAO,CAAC,WAAW,CAA2C;IAC9D,OAAO,CAAC,sBAAsB,CAAkC;IAChE,OAAO,CAAC,aAAa,CAA+C;IACpE,OAAO,CAAC,UAAU,CAAoB;gBAE1B,MAAM,EAAE,OAAO,MAAM,EAAE,OAAO,GAAE,IAA+B;IAO3E,OAAO,CAAC,UAAU,EAAE,cAAc,GAAG,MAAM;IAuF3C,UAAU,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI;IAc5B,UAAU,CAAC,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC,EAAE,IAAI,GAAG,IAAI;IAevD,YAAY,CAAC,EAAE,EAAE,MAAM,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,EAAE,IAAI,GAAG,IAAI;IAe3D,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,CAAC,EAAE,IAAI,GAAG,IAAI;IAU9D,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,IAAI,GAAG,IAAI;IAOjD,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,IAAI,GAAG,IAAI;IAUjD,UAAU,CAAC,MAAM,EAAE,WAAW,GAAG,IAAI;IAyCrC,aAAa,CAAC,UAAU,EAAE,oBAAoB,GAAG,MAAM;IAuBvD,gBAAgB,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI;IAOlC,aAAa,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO;IAIlC,IAAI,IAAI,kBAAkB,EAAE;IAiD5B,WAAW,CAAC,YAAY,UAAQ,GAAG,SAAS,EAAE;IAmB9C,cAAc,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO;IAKnC,YAAY,CAAC,EAAE,EAAE,MAAM,GAAG,SAAS,GAAG,IAAI;IAiB1C,SAAS,CAAC,MAAM,EAAE,cAAc,EAAE,GAAG,IAAI;IAMzC,KAAK,CAAC,MAAM,EAAE,cAAc,EAAE,EAAE,WAAW,CAAC,EAAE,oBAAoB,EAAE,GAAG,IAAI;IA0B3E,OAAO,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO;IAI5B,IAAI,SAAS,IAAI,MAAM,CAEtB;IAED,OAAO,IAAI,IAAI;CAQhB"}