@dcl/sdk 7.20.2-22169778016.commit-030cbfe → 7.20.2-22231111352.commit-d2f6f0a
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.
- package/network/binary-message-bus.d.ts +3 -6
- package/network/binary-message-bus.js +5 -9
- package/network/index.d.ts +2 -8
- package/network/index.js +3 -16
- package/network/message-bus-sync.d.ts +1 -14
- package/network/message-bus-sync.js +103 -166
- package/network/state.js +5 -3
- package/package.json +6 -6
- package/src/network/binary-message-bus.ts +4 -9
- package/src/network/index.ts +3 -40
- package/src/network/message-bus-sync.ts +110 -180
- package/src/network/state.ts +4 -3
- package/atom.d.ts +0 -19
- package/atom.js +0 -83
- package/future.d.ts +0 -8
- package/future.js +0 -26
- package/network/chunking.d.ts +0 -5
- package/network/chunking.js +0 -38
- package/network/events/implementation.d.ts +0 -93
- package/network/events/implementation.js +0 -230
- package/network/events/index.d.ts +0 -42
- package/network/events/index.js +0 -43
- package/network/events/protocol.d.ts +0 -27
- package/network/events/protocol.js +0 -66
- package/network/events/registry.d.ts +0 -8
- package/network/events/registry.js +0 -3
- package/network/server/index.d.ts +0 -14
- package/network/server/index.js +0 -219
- package/network/server/utils.d.ts +0 -18
- package/network/server/utils.js +0 -135
- package/server/env-var.d.ts +0 -15
- package/server/env-var.js +0 -31
- package/server/index.d.ts +0 -2
- package/server/index.js +0 -3
- package/server/storage/constants.d.ts +0 -23
- package/server/storage/constants.js +0 -2
- package/server/storage/index.d.ts +0 -22
- package/server/storage/index.js +0 -29
- package/server/storage/player.d.ts +0 -43
- package/server/storage/player.js +0 -92
- package/server/storage/scene.d.ts +0 -38
- package/server/storage/scene.js +0 -90
- package/server/storage-url.d.ts +0 -10
- package/server/storage-url.js +0 -29
- package/server/utils.d.ts +0 -35
- package/server/utils.js +0 -56
- package/src/atom.ts +0 -98
- package/src/future.ts +0 -38
- package/src/network/chunking.ts +0 -45
- package/src/network/events/implementation.ts +0 -286
- package/src/network/events/index.ts +0 -48
- package/src/network/events/protocol.ts +0 -94
- package/src/network/events/registry.ts +0 -18
- package/src/network/server/index.ts +0 -301
- package/src/network/server/utils.ts +0 -189
- package/src/server/env-var.ts +0 -36
- package/src/server/index.ts +0 -2
- package/src/server/storage/constants.ts +0 -22
- package/src/server/storage/index.ts +0 -44
- package/src/server/storage/player.ts +0 -156
- package/src/server/storage/scene.ts +0 -149
- package/src/server/storage-url.ts +0 -34
- package/src/server/utils.ts +0 -73
package/network/state.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer';
|
|
2
2
|
import { CrdtMessageProtocol, CrdtMessageType, PutComponentOperation, PutNetworkComponentOperation, NetworkEntity as _NetworkEntity, VideoEvent, AudioEvent, EngineInfo, GltfContainerLoadingState, PointerEventsResult, RaycastResult, RealmInfo, TweenState, UiDropdown, UiDropdownResult, UiInput, UiInputResult, UiText, UiTransform } from '@dcl/ecs';
|
|
3
|
-
import { LIVEKIT_MAX_SIZE } from '
|
|
3
|
+
import { LIVEKIT_MAX_SIZE } from '@dcl/ecs/dist/systems/crdt';
|
|
4
4
|
export const NOT_SYNC_COMPONENTS = [
|
|
5
5
|
VideoEvent,
|
|
6
6
|
TweenState,
|
|
@@ -38,7 +38,8 @@ export function engineToCrdt(engine) {
|
|
|
38
38
|
continue;
|
|
39
39
|
}
|
|
40
40
|
itComponentDefinition.dumpCrdtStateToBuffer(crdtBuffer, (entity) => {
|
|
41
|
-
|
|
41
|
+
const isNetworkEntity = NetworkEntity.has(entity);
|
|
42
|
+
return isNetworkEntity;
|
|
42
43
|
});
|
|
43
44
|
}
|
|
44
45
|
let header;
|
|
@@ -56,6 +57,7 @@ export function engineToCrdt(engine) {
|
|
|
56
57
|
networkBuffer.resetBuffer();
|
|
57
58
|
}
|
|
58
59
|
// If the message itself is larger than the limit, we need to handle it specially
|
|
60
|
+
// For now, we'll skip it to prevent infinite loops
|
|
59
61
|
if (messageSize / 1024 > LIVEKIT_MAX_SIZE) {
|
|
60
62
|
console.error(`Message too large (${messageSize} bytes), skipping component ${message.componentId} for entity ${message.entityId}`);
|
|
61
63
|
continue;
|
|
@@ -75,4 +77,4 @@ export function engineToCrdt(engine) {
|
|
|
75
77
|
}
|
|
76
78
|
return chunks;
|
|
77
79
|
}
|
|
78
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
80
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"state.js","sourceRoot":"","sources":["../src/network/state.ts"],"names":[],"mappings":"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","sourcesContent":["import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer'\nimport {\n  CrdtMessageHeader,\n  CrdtMessageProtocol,\n  CrdtMessageType,\n  IEngine,\n  PutComponentOperation,\n  PutNetworkComponentOperation,\n  SyncComponents as _SyncComponents,\n  NetworkEntity as _NetworkEntity,\n  INetowrkEntity,\n  VideoEvent,\n  AudioEvent,\n  EngineInfo,\n  GltfContainerLoadingState,\n  PointerEventsResult,\n  RaycastResult,\n  RealmInfo,\n  TweenState,\n  UiDropdown,\n  UiDropdownResult,\n  UiInput,\n  UiInputResult,\n  UiText,\n  UiTransform,\n  ComponentDefinition\n} from '@dcl/ecs'\nimport { LIVEKIT_MAX_SIZE } from '@dcl/ecs/dist/systems/crdt'\n\nexport const NOT_SYNC_COMPONENTS: ComponentDefinition<unknown>[] = [\n  VideoEvent,\n  TweenState,\n  AudioEvent,\n  EngineInfo,\n  GltfContainerLoadingState,\n  PointerEventsResult,\n  RaycastResult,\n  RealmInfo,\n  UiDropdown,\n  UiDropdownResult,\n  UiInput,\n  UiInputResult,\n  UiTransform,\n  UiText\n]\n\nexport const NOT_SYNC_COMPONENTS_IDS = NOT_SYNC_COMPONENTS.map(($) => $.componentId)\nexport const NOT_SYNC_COMPONENTS_NAMES: string[] = [\n  'asset-packs::Script' // ComponentName from: https://github.com/decentraland/asset-packs/blob/main/src/enums.ts\n]\n\nexport function shouldSyncComponent(component: ComponentDefinition<unknown>): boolean {\n  return !(\n    NOT_SYNC_COMPONENTS_IDS.includes(component.componentId) ||\n    NOT_SYNC_COMPONENTS_NAMES.includes(component.componentName)\n  )\n}\n\nexport function getDesyncedComponents(engine: IEngine): ComponentDefinition<unknown>[] {\n  return [...NOT_SYNC_COMPONENTS, ...NOT_SYNC_COMPONENTS_NAMES.map(($) => engine.getComponentOrNull($))].filter(\n    Boolean\n  ) as ComponentDefinition<unknown>[]\n}\n\nexport function engineToCrdt(engine: IEngine): Uint8Array[] {\n  const crdtBuffer = new ReadWriteByteBuffer()\n  const networkBuffer = new ReadWriteByteBuffer()\n  const NetworkEntity = engine.getComponent(_NetworkEntity.componentId) as INetowrkEntity\n  const chunks: Uint8Array[] = []\n\n  for (const itComponentDefinition of engine.componentsIter()) {\n    if (!shouldSyncComponent(itComponentDefinition)) {\n      continue\n    }\n    itComponentDefinition.dumpCrdtStateToBuffer(crdtBuffer, (entity) => {\n      const isNetworkEntity = NetworkEntity.has(entity)\n      return isNetworkEntity\n    })\n  }\n\n  let header: CrdtMessageHeader | null\n  while ((header = CrdtMessageProtocol.getHeader(crdtBuffer))) {\n    if (header.type === CrdtMessageType.PUT_COMPONENT) {\n      const message = PutComponentOperation.read(crdtBuffer)!\n      const networkEntity = NetworkEntity.getOrNull(message.entityId)\n\n      // Check if adding this message would exceed the size limit\n      const currentBufferSize = networkBuffer.toBinary().byteLength\n      const messageSize = message.data.byteLength\n\n      if ((currentBufferSize + messageSize) / 1024 > LIVEKIT_MAX_SIZE) {\n        // If the current buffer has content, save it as a chunk\n        if (currentBufferSize > 0) {\n          chunks.push(networkBuffer.toCopiedBinary())\n          networkBuffer.resetBuffer()\n        }\n\n        // If the message itself is larger than the limit, we need to handle it specially\n        // For now, we'll skip it to prevent infinite loops\n        if (messageSize / 1024 > LIVEKIT_MAX_SIZE) {\n          console.error(\n            `Message too large (${messageSize} bytes), skipping component ${message.componentId} for entity ${message.entityId}`\n          )\n          continue\n        }\n      }\n\n      if (networkEntity) {\n        PutNetworkComponentOperation.write(\n          networkEntity.entityId,\n          message.timestamp,\n          message.componentId,\n          networkEntity.networkId,\n          message.data,\n          networkBuffer\n        )\n      }\n    } else {\n      crdtBuffer.incrementReadOffset(header.length)\n    }\n  }\n\n  // Add any remaining data as the final chunk\n  if (networkBuffer.currentWriteOffset() > 0) {\n    chunks.push(networkBuffer.toBinary())\n  }\n\n  return chunks\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,15 +1,15 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dcl/sdk",
|
|
3
3
|
"description": "",
|
|
4
|
-
"version": "7.20.2-
|
|
4
|
+
"version": "7.20.2-22231111352.commit-d2f6f0a",
|
|
5
5
|
"author": "Decentraland",
|
|
6
6
|
"dependencies": {
|
|
7
|
-
"@dcl/ecs": "7.20.2-
|
|
7
|
+
"@dcl/ecs": "7.20.2-22231111352.commit-d2f6f0a",
|
|
8
8
|
"@dcl/ecs-math": "2.1.0",
|
|
9
9
|
"@dcl/explorer": "1.0.164509-20240802172549.commit-fb95b9b",
|
|
10
|
-
"@dcl/js-runtime": "7.20.2-
|
|
11
|
-
"@dcl/react-ecs": "7.20.2-
|
|
12
|
-
"@dcl/sdk-commands": "7.20.2-
|
|
10
|
+
"@dcl/js-runtime": "7.20.2-22231111352.commit-d2f6f0a",
|
|
11
|
+
"@dcl/react-ecs": "7.20.2-22231111352.commit-d2f6f0a",
|
|
12
|
+
"@dcl/sdk-commands": "7.20.2-22231111352.commit-d2f6f0a",
|
|
13
13
|
"text-encoding": "0.7.0"
|
|
14
14
|
},
|
|
15
15
|
"keywords": [],
|
|
@@ -35,5 +35,5 @@
|
|
|
35
35
|
},
|
|
36
36
|
"types": "./index.d.ts",
|
|
37
37
|
"typings": "./index.d.ts",
|
|
38
|
-
"commit": "
|
|
38
|
+
"commit": "d2f6f0a879331345b3f46766f176294cc562da7b"
|
|
39
39
|
}
|
|
@@ -1,12 +1,9 @@
|
|
|
1
1
|
import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer'
|
|
2
2
|
|
|
3
3
|
export enum CommsMessage {
|
|
4
|
-
CRDT =
|
|
5
|
-
REQ_CRDT_STATE =
|
|
6
|
-
RES_CRDT_STATE =
|
|
7
|
-
CRDT_SERVER = 4,
|
|
8
|
-
CRDT_AUTHORITATIVE = 5,
|
|
9
|
-
CUSTOM_EVENT = 6
|
|
4
|
+
CRDT = 1,
|
|
5
|
+
REQ_CRDT_STATE = 2,
|
|
6
|
+
RES_CRDT_STATE = 3
|
|
10
7
|
}
|
|
11
8
|
|
|
12
9
|
export function BinaryMessageBus<T extends CommsMessage>(
|
|
@@ -23,9 +20,7 @@ export function BinaryMessageBus<T extends CommsMessage>(
|
|
|
23
20
|
__processMessages: (messages: Uint8Array[]) => {
|
|
24
21
|
for (const message of messages) {
|
|
25
22
|
const commsMsg = decodeCommsMessage<T>(message)
|
|
26
|
-
if (!commsMsg)
|
|
27
|
-
continue
|
|
28
|
-
}
|
|
23
|
+
if (!commsMsg) continue
|
|
29
24
|
const { sender, messageType, data } = commsMsg
|
|
30
25
|
const fn = mapping.get(messageType)
|
|
31
26
|
if (fn) fn(data, sender)
|
package/src/network/index.ts
CHANGED
|
@@ -2,46 +2,9 @@ import { sendBinary } from '~system/CommunicationsController'
|
|
|
2
2
|
import { engine } from '@dcl/ecs'
|
|
3
3
|
import { addSyncTransport } from './message-bus-sync'
|
|
4
4
|
import { getUserData } from '~system/UserIdentity'
|
|
5
|
-
import { isServer as isServerApi } from '~system/EngineApi'
|
|
6
|
-
import { Atom } from '../atom'
|
|
7
|
-
|
|
8
|
-
// Create isServer atom for consistent state
|
|
9
|
-
const isServerAtom = Atom<boolean>(false)
|
|
10
|
-
void isServerApi({}).then((response) => {
|
|
11
|
-
isServerAtom.swap(!!response.isServer)
|
|
12
|
-
})
|
|
13
|
-
|
|
14
|
-
// Helper function to check if running on server
|
|
15
|
-
export function isServer(): boolean {
|
|
16
|
-
return isServerAtom.getOrNull() ?? false
|
|
17
|
-
}
|
|
18
5
|
|
|
19
6
|
// initialize sync transport for sdk engine
|
|
20
|
-
const {
|
|
21
|
-
|
|
22
|
-
syncEntity,
|
|
23
|
-
parentEntity,
|
|
24
|
-
getParent,
|
|
25
|
-
myProfile,
|
|
26
|
-
removeParent,
|
|
27
|
-
getFirstChild,
|
|
28
|
-
isStateSyncronized,
|
|
29
|
-
binaryMessageBus,
|
|
30
|
-
eventBus
|
|
31
|
-
} = addSyncTransport(engine, sendBinary, getUserData, isServerApi, 'network')
|
|
32
|
-
|
|
33
|
-
// Re-export the room messaging system
|
|
34
|
-
export { registerMessages, getRoom } from './events'
|
|
7
|
+
const { getChildren, syncEntity, parentEntity, getParent, myProfile, removeParent, getFirstChild, isStateSyncronized } =
|
|
8
|
+
addSyncTransport(engine, sendBinary, getUserData)
|
|
35
9
|
|
|
36
|
-
export {
|
|
37
|
-
getFirstChild,
|
|
38
|
-
getChildren,
|
|
39
|
-
syncEntity,
|
|
40
|
-
parentEntity,
|
|
41
|
-
getParent,
|
|
42
|
-
myProfile,
|
|
43
|
-
removeParent,
|
|
44
|
-
isStateSyncronized,
|
|
45
|
-
binaryMessageBus,
|
|
46
|
-
eventBus
|
|
47
|
-
}
|
|
10
|
+
export { getFirstChild, getChildren, syncEntity, parentEntity, getParent, myProfile, removeParent, isStateSyncronized }
|
|
@@ -1,53 +1,27 @@
|
|
|
1
|
-
import { IEngine, Transport, RealmInfo } from '@dcl/ecs'
|
|
1
|
+
import { IEngine, Transport, RealmInfo, PlayerIdentityData } from '@dcl/ecs'
|
|
2
2
|
import { type SendBinaryRequest, type SendBinaryResponse } from '~system/CommunicationsController'
|
|
3
3
|
|
|
4
4
|
import { syncFilter } from './filter'
|
|
5
5
|
import { engineToCrdt } from './state'
|
|
6
|
-
import { BinaryMessageBus, CommsMessage } from './binary-message-bus'
|
|
6
|
+
import { BinaryMessageBus, CommsMessage, decodeString, encodeString } from './binary-message-bus'
|
|
7
7
|
import { fetchProfile } from './utils'
|
|
8
8
|
import { entityUtils } from './entities'
|
|
9
|
-
import { createServerValidator } from './server'
|
|
10
9
|
import { GetUserDataRequest, GetUserDataResponse } from '~system/UserIdentity'
|
|
11
10
|
import { definePlayerHelper } from '../players'
|
|
12
11
|
import { serializeCrdtMessages } from '../internal/transports/logger'
|
|
13
|
-
import { IsServerRequest, IsServerResponse } from '~system/EngineApi'
|
|
14
|
-
import { Atom } from '../atom'
|
|
15
|
-
import { setGlobalRoom, Room } from './events/implementation'
|
|
16
12
|
|
|
17
13
|
export type IProfile = { networkId: number; userId: string }
|
|
18
14
|
// user that we asked for the inital crdt state
|
|
19
|
-
export const AUTH_SERVER_PEER_ID = 'authoritative-server'
|
|
20
|
-
export const DEBUG_NETWORK_MESSAGES = () => (globalThis as any).DEBUG_NETWORK_MESSAGES ?? false
|
|
21
|
-
|
|
22
|
-
// Test environment detection without 'as any'
|
|
23
|
-
const isTestEnvironment = (): boolean => {
|
|
24
|
-
try {
|
|
25
|
-
if (typeof globalThis === 'undefined') return false
|
|
26
|
-
const globalWithProcess = globalThis as unknown as { process?: { env?: { NODE_ENV?: string } } }
|
|
27
|
-
return globalWithProcess.process?.env?.NODE_ENV === 'test'
|
|
28
|
-
} catch {
|
|
29
|
-
return false
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
15
|
export function addSyncTransport(
|
|
34
16
|
engine: IEngine,
|
|
35
17
|
sendBinary: (msg: SendBinaryRequest) => Promise<SendBinaryResponse>,
|
|
36
|
-
getUserData: (value: GetUserDataRequest) => Promise<GetUserDataResponse
|
|
37
|
-
isServerFn: (request: IsServerRequest) => Promise<IsServerResponse>,
|
|
38
|
-
name: string
|
|
18
|
+
getUserData: (value: GetUserDataRequest) => Promise<GetUserDataResponse>
|
|
39
19
|
) {
|
|
20
|
+
const DEBUG_NETWORK_MESSAGES = () => (globalThis as any).DEBUG_NETWORK_MESSAGES ?? false
|
|
40
21
|
// Profile Info
|
|
41
22
|
const myProfile: IProfile = {} as IProfile
|
|
42
23
|
fetchProfile(myProfile!, getUserData)
|
|
43
24
|
|
|
44
|
-
const isServerAtom = Atom<boolean>()
|
|
45
|
-
const isRoomReadyAtom = Atom<boolean>(false)
|
|
46
|
-
|
|
47
|
-
void isServerFn({}).then(($: IsServerResponse) => {
|
|
48
|
-
return isServerAtom.swap(!!$.isServer)
|
|
49
|
-
})
|
|
50
|
-
|
|
51
25
|
// Entity utils
|
|
52
26
|
const entityDefinitions = entityUtils(engine, myProfile)
|
|
53
27
|
|
|
@@ -66,196 +40,108 @@ export function addSyncTransport(
|
|
|
66
40
|
const players = definePlayerHelper(engine)
|
|
67
41
|
|
|
68
42
|
let stateIsSyncronized = false
|
|
43
|
+
let transportInitialzed = false
|
|
69
44
|
|
|
70
|
-
/**
|
|
71
|
-
* We need to wait till 2 ticks that is when the engine is ready to send new messages.
|
|
72
|
-
* The first tick is for the client engine processing the CRDT messages,
|
|
73
|
-
* and the second one are the messages created by the main() function.
|
|
74
|
-
* So to avoid sending those messages, that all the clients have, through the network we put this validation here.
|
|
75
|
-
*/
|
|
76
|
-
let tick = 0
|
|
77
|
-
const TRANSPORT_INITIALIZED_NUMBER = isTestEnvironment() ? 0 : 2
|
|
78
45
|
// Add Sync Transport
|
|
79
46
|
const transport: Transport = {
|
|
80
47
|
filter: syncFilter(engine),
|
|
81
48
|
send: async (messages) => {
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
if (message.byteLength) {
|
|
49
|
+
for (const message of [messages].flat()) {
|
|
50
|
+
if (message.byteLength && transportInitialzed) {
|
|
85
51
|
DEBUG_NETWORK_MESSAGES() &&
|
|
86
52
|
console.log(...Array.from(serializeCrdtMessages('[NetworkMessage sent]:', message, engine)))
|
|
87
|
-
|
|
88
|
-
// Convert regular messages to network messages for broadcasting with chunking
|
|
89
|
-
for (const chunk of serverValidator.convertRegularToNetworkMessage(message)) {
|
|
90
|
-
binaryMessageBus.emit(CommsMessage.CRDT, chunk)
|
|
91
|
-
}
|
|
53
|
+
binaryMessageBus.emit(CommsMessage.CRDT, message)
|
|
92
54
|
}
|
|
93
55
|
}
|
|
94
56
|
const peerMessages = getMessagesToSend()
|
|
57
|
+
let totalSize = 0
|
|
58
|
+
for (const message of peerMessages) {
|
|
59
|
+
for (const data of message.data) {
|
|
60
|
+
totalSize += data.byteLength
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
if (totalSize) {
|
|
64
|
+
DEBUG_NETWORK_MESSAGES() && console.log('Sending network messages: ', totalSize / 1024, 'KB')
|
|
65
|
+
}
|
|
95
66
|
const response = await sendBinary({ data: [], peerData: peerMessages })
|
|
96
67
|
binaryMessageBus.__processMessages(response.data)
|
|
68
|
+
transportInitialzed = true
|
|
97
69
|
},
|
|
98
|
-
type:
|
|
70
|
+
type: 'network'
|
|
99
71
|
}
|
|
100
|
-
|
|
101
|
-
// Server validation setup
|
|
102
|
-
const serverValidator = createServerValidator({
|
|
103
|
-
engine,
|
|
104
|
-
binaryMessageBus
|
|
105
|
-
})
|
|
106
|
-
|
|
107
|
-
// Initialize Event Bus with registered schemas
|
|
108
|
-
const eventBus = new Room(engine, binaryMessageBus, isServerAtom, isRoomReadyAtom)
|
|
109
|
-
|
|
110
|
-
// Set global eventBus instance
|
|
111
|
-
setGlobalRoom(eventBus)
|
|
112
|
-
|
|
113
72
|
engine.addTransport(transport)
|
|
114
73
|
// End add sync transport
|
|
115
74
|
|
|
116
75
|
// Receive & Process CRDT_STATE
|
|
117
|
-
binaryMessageBus.on(CommsMessage.
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[Emiting:]', sender, Date.now())
|
|
121
|
-
binaryMessageBus.emit(CommsMessage.RES_CRDT_STATE, chunk, [sender])
|
|
122
|
-
}
|
|
123
|
-
})
|
|
124
|
-
binaryMessageBus.on(CommsMessage.RES_CRDT_STATE, async (data, sender) => {
|
|
125
|
-
requestingState = false
|
|
126
|
-
elapsedTimeSinceRequest = 0
|
|
127
|
-
if (isServerAtom.getOrNull() || sender !== AUTH_SERVER_PEER_ID) return
|
|
76
|
+
binaryMessageBus.on(CommsMessage.RES_CRDT_STATE, (value) => {
|
|
77
|
+
const { sender, data } = decodeCRDTState(value)
|
|
78
|
+
if (sender !== myProfile.userId) return
|
|
128
79
|
DEBUG_NETWORK_MESSAGES() && console.log('[Processing CRDT State]', data.byteLength / 1024, 'KB')
|
|
129
|
-
transport.onmessage!(
|
|
80
|
+
transport.onmessage!(data)
|
|
130
81
|
stateIsSyncronized = true
|
|
82
|
+
})
|
|
83
|
+
|
|
84
|
+
// Answer to REQ_CRDT_STATE
|
|
85
|
+
binaryMessageBus.on(CommsMessage.REQ_CRDT_STATE, async (_, userId) => {
|
|
86
|
+
DEBUG_NETWORK_MESSAGES() && console.log(`Sending CRDT State to: ${userId}`)
|
|
131
87
|
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
const realmInfo = RealmInfo.getOrNull(engine.RootEntity)
|
|
135
|
-
if (realmInfo && checkRoomReady(realmInfo)) {
|
|
136
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[isRoomReady] Marking room as ready after state sync')
|
|
137
|
-
isRoomReadyAtom.swap(true)
|
|
88
|
+
for (const chunk of engineToCrdt(engine)) {
|
|
89
|
+
binaryMessageBus.emit(CommsMessage.RES_CRDT_STATE, encodeCRDTState(userId, chunk), [userId])
|
|
138
90
|
}
|
|
139
91
|
})
|
|
140
92
|
|
|
141
|
-
//
|
|
142
|
-
binaryMessageBus.on(CommsMessage.CRDT, (value
|
|
143
|
-
const isServer = isServerAtom.getOrNull()
|
|
93
|
+
// Process CRDT messages here
|
|
94
|
+
binaryMessageBus.on(CommsMessage.CRDT, (value) => {
|
|
144
95
|
DEBUG_NETWORK_MESSAGES() &&
|
|
145
|
-
console.log(
|
|
146
|
-
|
|
147
|
-
...Array.from(serializeCrdtMessages('[NetworkMessage received]:', value, engine)),
|
|
148
|
-
isServer
|
|
149
|
-
)
|
|
150
|
-
if (isServer) {
|
|
151
|
-
transport.onmessage!(serverValidator.processServerMessages(value, sender))
|
|
152
|
-
} else if (sender === AUTH_SERVER_PEER_ID) {
|
|
153
|
-
// Process network messages from server and convert to regular messages
|
|
154
|
-
transport.onmessage!(serverValidator.processClientMessages(value, sender))
|
|
155
|
-
}
|
|
96
|
+
console.log(Array.from(serializeCrdtMessages('[NetworkMessage received]:', value, engine)))
|
|
97
|
+
transport.onmessage!(value)
|
|
156
98
|
})
|
|
157
99
|
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
if (sender !== AUTH_SERVER_PEER_ID) return
|
|
162
|
-
|
|
163
|
-
// DEBUG_NETWORK_MESSAGES() &&
|
|
164
|
-
console.log('[AUTHORITATIVE] Received authoritative message from server:', value.byteLength, 'bytes')
|
|
100
|
+
async function requestState(retryCount: number = 1) {
|
|
101
|
+
let players = Array.from(engine.getEntitiesWith(PlayerIdentityData))
|
|
102
|
+
DEBUG_NETWORK_MESSAGES() && console.log(`Requesting state. Players connected: ${players.length - 1}`)
|
|
165
103
|
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
if (authoritativeBuffer.byteLength > 0) {
|
|
170
|
-
// Apply authoritative message through normal transport, but the server's messages
|
|
171
|
-
// should be processed as authoritative with special timestamp handling
|
|
172
|
-
transport.onmessage!(authoritativeBuffer)
|
|
173
|
-
|
|
174
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[AUTHORITATIVE] Applied server authoritative message to local state')
|
|
104
|
+
if (!RealmInfo.getOrNull(engine.RootEntity)?.isConnectedSceneRoom) {
|
|
105
|
+
DEBUG_NETWORK_MESSAGES() && console.log(`Aborting Requesting state?. Disconnected`)
|
|
106
|
+
return
|
|
175
107
|
}
|
|
176
|
-
})
|
|
177
108
|
|
|
178
|
-
|
|
179
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[onEnterScene]', player.userId)
|
|
180
|
-
if (!isServerAtom.getOrNull() && myProfile.userId === player.userId) {
|
|
181
|
-
requestState()
|
|
182
|
-
}
|
|
183
|
-
})
|
|
109
|
+
binaryMessageBus.emit(CommsMessage.REQ_CRDT_STATE, new Uint8Array())
|
|
184
110
|
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
if (!realmInfo) return false
|
|
111
|
+
// Wait ~5s for the response.
|
|
112
|
+
await sleep(5000)
|
|
188
113
|
|
|
189
|
-
|
|
190
|
-
// Check if room instance exists
|
|
191
|
-
if (!eventBus) return false
|
|
114
|
+
players = Array.from(engine.getEntitiesWith(PlayerIdentityData))
|
|
192
115
|
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
116
|
+
if (!stateIsSyncronized) {
|
|
117
|
+
if (players.length > 1 && retryCount <= 2) {
|
|
118
|
+
DEBUG_NETWORK_MESSAGES() &&
|
|
119
|
+
console.log(`Requesting state again ${retryCount} (no response). Players connected: ${players.length - 1}`)
|
|
120
|
+
void requestState(retryCount + 1)
|
|
121
|
+
} else {
|
|
122
|
+
DEBUG_NETWORK_MESSAGES() && console.log('No active players. State syncronized')
|
|
123
|
+
stateIsSyncronized = true
|
|
124
|
+
}
|
|
196
125
|
}
|
|
197
126
|
}
|
|
198
127
|
|
|
128
|
+
players.onEnterScene((player) => {
|
|
129
|
+
DEBUG_NETWORK_MESSAGES() && console.log('[onEnterScene]', player.userId)
|
|
130
|
+
})
|
|
131
|
+
|
|
199
132
|
// Asks for the REQ_CRDT_STATE when its connected to comms
|
|
200
133
|
RealmInfo.onChange(engine.RootEntity, (value) => {
|
|
201
|
-
const isServer = isServerAtom.getOrNull()
|
|
202
|
-
|
|
203
134
|
if (!value?.isConnectedSceneRoom) {
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
DEBUG_NETWORK_MESSAGES() && console.log('Disconnected from comms')
|
|
207
|
-
isRoomReadyAtom.swap(false)
|
|
208
|
-
if (!isServer) {
|
|
209
|
-
stateIsSyncronized = false
|
|
210
|
-
}
|
|
211
|
-
}
|
|
135
|
+
DEBUG_NETWORK_MESSAGES() && console.log('Disconnected from comms')
|
|
136
|
+
stateIsSyncronized = false
|
|
212
137
|
}
|
|
213
138
|
|
|
214
139
|
if (value?.isConnectedSceneRoom) {
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
// For servers, mark as ready immediately when connected
|
|
218
|
-
// (servers don't need to sync state from anyone)
|
|
219
|
-
if (isServer && checkRoomReady(value) && isRoomReadyAtom.getOrNull() === false) {
|
|
220
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[isRoomReady] Server marking room as ready')
|
|
221
|
-
isRoomReadyAtom.swap(true)
|
|
222
|
-
}
|
|
223
|
-
// For clients, room will be marked ready after receiving CRDT state (above)
|
|
140
|
+
DEBUG_NETWORK_MESSAGES() && console.log('Connected to comms')
|
|
224
141
|
}
|
|
225
|
-
})
|
|
226
142
|
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
const STATE_REQUEST_RETRY_INTERVAL = 2.0 // seconds
|
|
230
|
-
|
|
231
|
-
/**
|
|
232
|
-
* Why we have to request the state if we have a server that can send us the state when we joined?
|
|
233
|
-
* The thing is that when the server detects a new JOIN_PARTICIPANT on livekit room, it sends automatically the state to that peer.
|
|
234
|
-
* But in unity, it takes more time, so that message is not being delivered to the client.
|
|
235
|
-
* So instead, when we are finally connected to the room, we request the state, and then the server answers with the state :)
|
|
236
|
-
*
|
|
237
|
-
* If no response is received within 2 seconds, the request is automatically retried.
|
|
238
|
-
*/
|
|
239
|
-
function requestState() {
|
|
240
|
-
if (isServerAtom.getOrNull()) return
|
|
241
|
-
if (RealmInfo.getOrNull(engine.RootEntity)?.isConnectedSceneRoom && !requestingState) {
|
|
242
|
-
requestingState = true
|
|
243
|
-
elapsedTimeSinceRequest = 0
|
|
244
|
-
DEBUG_NETWORK_MESSAGES() && console.log('Requesting state...')
|
|
245
|
-
binaryMessageBus.emit(CommsMessage.REQ_CRDT_STATE, new Uint8Array())
|
|
246
|
-
}
|
|
247
|
-
}
|
|
248
|
-
|
|
249
|
-
// System to retry state request if no response is received within the retry interval
|
|
250
|
-
engine.addSystem((dt: number) => {
|
|
251
|
-
if (requestingState && !stateIsSyncronized) {
|
|
252
|
-
elapsedTimeSinceRequest += dt
|
|
253
|
-
if (elapsedTimeSinceRequest >= STATE_REQUEST_RETRY_INTERVAL) {
|
|
254
|
-
DEBUG_NETWORK_MESSAGES() && console.log('State request timed out, retrying...')
|
|
255
|
-
elapsedTimeSinceRequest = 0
|
|
256
|
-
requestingState = false
|
|
257
|
-
requestState()
|
|
258
|
-
}
|
|
143
|
+
if (value?.isConnectedSceneRoom && !stateIsSyncronized) {
|
|
144
|
+
void requestState()
|
|
259
145
|
}
|
|
260
146
|
})
|
|
261
147
|
|
|
@@ -267,12 +153,56 @@ export function addSyncTransport(
|
|
|
267
153
|
return stateIsSyncronized
|
|
268
154
|
}
|
|
269
155
|
|
|
156
|
+
function sleep(ms: number) {
|
|
157
|
+
return new Promise<void>((resolve) => {
|
|
158
|
+
let timer = 0
|
|
159
|
+
function sleepSystem(dt: number) {
|
|
160
|
+
timer += dt
|
|
161
|
+
if (timer * 1000 >= ms) {
|
|
162
|
+
engine.removeSystem(sleepSystem)
|
|
163
|
+
resolve()
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
engine.addSystem(sleepSystem)
|
|
167
|
+
})
|
|
168
|
+
}
|
|
169
|
+
|
|
270
170
|
return {
|
|
271
171
|
...entityDefinitions,
|
|
272
172
|
myProfile,
|
|
273
|
-
isStateSyncronized
|
|
274
|
-
binaryMessageBus,
|
|
275
|
-
eventBus,
|
|
276
|
-
isRoomReadyAtom
|
|
173
|
+
isStateSyncronized
|
|
277
174
|
}
|
|
278
175
|
}
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Messages Protocol Encoding
|
|
179
|
+
*
|
|
180
|
+
* CRDT: Plain Uint8Array
|
|
181
|
+
*
|
|
182
|
+
* CRDT_STATE_RES { sender: string, data: Uint8Array}
|
|
183
|
+
*/
|
|
184
|
+
function decodeCRDTState(data: Uint8Array) {
|
|
185
|
+
let offset = 0
|
|
186
|
+
const r = new Uint8Array(data)
|
|
187
|
+
const view = new DataView(r.buffer)
|
|
188
|
+
const senderLength = view.getUint8(offset)
|
|
189
|
+
offset += 1
|
|
190
|
+
const sender = decodeString(data.subarray(1, senderLength + 1))
|
|
191
|
+
offset += senderLength
|
|
192
|
+
const state = r.subarray(offset)
|
|
193
|
+
|
|
194
|
+
return { sender, data: state }
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
function encodeCRDTState(address: string, data: Uint8Array) {
|
|
198
|
+
// address to uint8array
|
|
199
|
+
const addressBuffer = encodeString(address)
|
|
200
|
+
const addressOffset = 1
|
|
201
|
+
const messageLength = addressOffset + addressBuffer.byteLength + data.byteLength
|
|
202
|
+
|
|
203
|
+
const serializedMessage = new Uint8Array(messageLength)
|
|
204
|
+
serializedMessage.set(new Uint8Array([addressBuffer.byteLength]), 0)
|
|
205
|
+
serializedMessage.set(addressBuffer, 1)
|
|
206
|
+
serializedMessage.set(data, addressBuffer.byteLength + 1)
|
|
207
|
+
return serializedMessage
|
|
208
|
+
}
|
package/src/network/state.ts
CHANGED
|
@@ -25,7 +25,7 @@ import {
|
|
|
25
25
|
UiTransform,
|
|
26
26
|
ComponentDefinition
|
|
27
27
|
} from '@dcl/ecs'
|
|
28
|
-
import { LIVEKIT_MAX_SIZE } from '
|
|
28
|
+
import { LIVEKIT_MAX_SIZE } from '@dcl/ecs/dist/systems/crdt'
|
|
29
29
|
|
|
30
30
|
export const NOT_SYNC_COMPONENTS: ComponentDefinition<unknown>[] = [
|
|
31
31
|
VideoEvent,
|
|
@@ -72,9 +72,9 @@ export function engineToCrdt(engine: IEngine): Uint8Array[] {
|
|
|
72
72
|
if (!shouldSyncComponent(itComponentDefinition)) {
|
|
73
73
|
continue
|
|
74
74
|
}
|
|
75
|
-
|
|
76
75
|
itComponentDefinition.dumpCrdtStateToBuffer(crdtBuffer, (entity) => {
|
|
77
|
-
|
|
76
|
+
const isNetworkEntity = NetworkEntity.has(entity)
|
|
77
|
+
return isNetworkEntity
|
|
78
78
|
})
|
|
79
79
|
}
|
|
80
80
|
|
|
@@ -96,6 +96,7 @@ export function engineToCrdt(engine: IEngine): Uint8Array[] {
|
|
|
96
96
|
}
|
|
97
97
|
|
|
98
98
|
// If the message itself is larger than the limit, we need to handle it specially
|
|
99
|
+
// For now, we'll skip it to prevent infinite loops
|
|
99
100
|
if (messageSize / 1024 > LIVEKIT_MAX_SIZE) {
|
|
100
101
|
console.error(
|
|
101
102
|
`Message too large (${messageSize} bytes), skipping component ${message.componentId} for entity ${message.entityId}`
|
package/atom.d.ts
DELETED
|
@@ -1,19 +0,0 @@
|
|
|
1
|
-
declare class SimpleObservable<T> {
|
|
2
|
-
private observers;
|
|
3
|
-
private onceObservers;
|
|
4
|
-
add(callback: (value: T) => void): (value: T) => void;
|
|
5
|
-
addOnce(callback: (value: T) => void): void;
|
|
6
|
-
remove(callback: (value: T) => void): void;
|
|
7
|
-
notifyObservers(value: T): void;
|
|
8
|
-
}
|
|
9
|
-
declare const EMPTY: unique symbol;
|
|
10
|
-
type EMPTY = typeof EMPTY;
|
|
11
|
-
export type Atom<T> = {
|
|
12
|
-
deref(): Promise<T>;
|
|
13
|
-
getOrNull(): T | null;
|
|
14
|
-
observable: SimpleObservable<T>;
|
|
15
|
-
swap(value: T): T | void;
|
|
16
|
-
pipe(fn: (value: T) => void | Promise<void>): Promise<void>;
|
|
17
|
-
};
|
|
18
|
-
export declare function Atom<T>(initialValue?: T | EMPTY): Atom<T>;
|
|
19
|
-
export {};
|