@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.
Files changed (63) hide show
  1. package/network/binary-message-bus.d.ts +3 -6
  2. package/network/binary-message-bus.js +5 -9
  3. package/network/index.d.ts +2 -8
  4. package/network/index.js +3 -16
  5. package/network/message-bus-sync.d.ts +1 -14
  6. package/network/message-bus-sync.js +103 -166
  7. package/network/state.js +5 -3
  8. package/package.json +6 -6
  9. package/src/network/binary-message-bus.ts +4 -9
  10. package/src/network/index.ts +3 -40
  11. package/src/network/message-bus-sync.ts +110 -180
  12. package/src/network/state.ts +4 -3
  13. package/atom.d.ts +0 -19
  14. package/atom.js +0 -83
  15. package/future.d.ts +0 -8
  16. package/future.js +0 -26
  17. package/network/chunking.d.ts +0 -5
  18. package/network/chunking.js +0 -38
  19. package/network/events/implementation.d.ts +0 -93
  20. package/network/events/implementation.js +0 -230
  21. package/network/events/index.d.ts +0 -42
  22. package/network/events/index.js +0 -43
  23. package/network/events/protocol.d.ts +0 -27
  24. package/network/events/protocol.js +0 -66
  25. package/network/events/registry.d.ts +0 -8
  26. package/network/events/registry.js +0 -3
  27. package/network/server/index.d.ts +0 -14
  28. package/network/server/index.js +0 -219
  29. package/network/server/utils.d.ts +0 -18
  30. package/network/server/utils.js +0 -135
  31. package/server/env-var.d.ts +0 -15
  32. package/server/env-var.js +0 -31
  33. package/server/index.d.ts +0 -2
  34. package/server/index.js +0 -3
  35. package/server/storage/constants.d.ts +0 -23
  36. package/server/storage/constants.js +0 -2
  37. package/server/storage/index.d.ts +0 -22
  38. package/server/storage/index.js +0 -29
  39. package/server/storage/player.d.ts +0 -43
  40. package/server/storage/player.js +0 -92
  41. package/server/storage/scene.d.ts +0 -38
  42. package/server/storage/scene.js +0 -90
  43. package/server/storage-url.d.ts +0 -10
  44. package/server/storage-url.js +0 -29
  45. package/server/utils.d.ts +0 -35
  46. package/server/utils.js +0 -56
  47. package/src/atom.ts +0 -98
  48. package/src/future.ts +0 -38
  49. package/src/network/chunking.ts +0 -45
  50. package/src/network/events/implementation.ts +0 -286
  51. package/src/network/events/index.ts +0 -48
  52. package/src/network/events/protocol.ts +0 -94
  53. package/src/network/events/registry.ts +0 -18
  54. package/src/network/server/index.ts +0 -301
  55. package/src/network/server/utils.ts +0 -189
  56. package/src/server/env-var.ts +0 -36
  57. package/src/server/index.ts +0 -2
  58. package/src/server/storage/constants.ts +0 -22
  59. package/src/server/storage/index.ts +0 -44
  60. package/src/server/storage/player.ts +0 -156
  61. package/src/server/storage/scene.ts +0 -149
  62. package/src/server/storage-url.ts +0 -34
  63. 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 './server';
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
- return NetworkEntity.has(entity);
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,{"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 './server'\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\n    itComponentDefinition.dumpCrdtStateToBuffer(crdtBuffer, (entity) => {\n      return NetworkEntity.has(entity)\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        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"]}
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-22169778016.commit-030cbfe",
4
+ "version": "7.20.2-22231111352.commit-d2f6f0a",
5
5
  "author": "Decentraland",
6
6
  "dependencies": {
7
- "@dcl/ecs": "7.20.2-22169778016.commit-030cbfe",
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-22169778016.commit-030cbfe",
11
- "@dcl/react-ecs": "7.20.2-22169778016.commit-030cbfe",
12
- "@dcl/sdk-commands": "7.20.2-22169778016.commit-030cbfe",
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": "030cbfee8a40495d9916ade20280282e63e3cc8b"
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 = 7,
5
- REQ_CRDT_STATE = 8,
6
- RES_CRDT_STATE = 9,
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)
@@ -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
- getChildren,
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
- if (tick <= TRANSPORT_INITIALIZED_NUMBER) tick++
83
- for (const message of tick > TRANSPORT_INITIALIZED_NUMBER ? [messages].flat() : []) {
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: name
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.REQ_CRDT_STATE, async (data, sender) => {
118
- DEBUG_NETWORK_MESSAGES() && console.log('[REQ_CRDT_STATE]', sender, Date.now())
119
- for (const chunk of engineToCrdt(engine)) {
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!(serverValidator.processClientMessages(data, sender))
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
- // IMPORTANT: Only mark room as ready AFTER state is synchronized
133
- // This ensures comms is truly connected and working
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
- // received message from the network
142
- binaryMessageBus.on(CommsMessage.CRDT, (value, sender) => {
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
- transport.type,
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
- // received authoritative message from server - force apply to fix invalid local state
159
- binaryMessageBus.on(CommsMessage.CRDT_AUTHORITATIVE, (value, sender) => {
160
- // Only accept authoritative messages from authoritative server
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
- // Process authoritative messages by forcing them through normal CRDT processing
167
- // but with a timestamp that's guaranteed to be accepted
168
- const authoritativeBuffer = serverValidator.processClientMessages(value, sender, true)
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
- players.onEnterScene((player) => {
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
- // Helper to check room ready conditions
186
- function checkRoomReady(realmInfo: ReturnType<typeof RealmInfo.getOrNull>): boolean {
187
- if (!realmInfo) return false
111
+ // Wait ~5s for the response.
112
+ await sleep(5000)
188
113
 
189
- try {
190
- // Check if room instance exists
191
- if (!eventBus) return false
114
+ players = Array.from(engine.getEntitiesWith(PlayerIdentityData))
192
115
 
193
- return !!(realmInfo.commsAdapter && realmInfo.isConnectedSceneRoom && realmInfo.room)
194
- } catch {
195
- return false
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
- // Only react when actually transitioning from ready to not ready
205
- if (isRoomReadyAtom.getOrNull() === true) {
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
- requestState()
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
- let requestingState = false
228
- let elapsedTimeSinceRequest = 0
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
+ }
@@ -25,7 +25,7 @@ import {
25
25
  UiTransform,
26
26
  ComponentDefinition
27
27
  } from '@dcl/ecs'
28
- import { LIVEKIT_MAX_SIZE } from './server'
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
- return NetworkEntity.has(entity)
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 {};