@dcl/sdk 7.20.0 → 7.20.1-21975617794.commit-7189140

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/atom.d.ts +19 -0
  2. package/atom.js +83 -0
  3. package/future.d.ts +8 -0
  4. package/future.js +26 -0
  5. package/network/binary-message-bus.d.ts +6 -3
  6. package/network/binary-message-bus.js +9 -5
  7. package/network/chunking.d.ts +5 -0
  8. package/network/chunking.js +38 -0
  9. package/network/events/implementation.d.ts +93 -0
  10. package/network/events/implementation.js +230 -0
  11. package/network/events/index.d.ts +42 -0
  12. package/network/events/index.js +43 -0
  13. package/network/events/protocol.d.ts +27 -0
  14. package/network/events/protocol.js +66 -0
  15. package/network/events/registry.d.ts +8 -0
  16. package/network/events/registry.js +3 -0
  17. package/network/index.d.ts +8 -2
  18. package/network/index.js +16 -3
  19. package/network/message-bus-sync.d.ts +14 -1
  20. package/network/message-bus-sync.js +166 -103
  21. package/network/server/index.d.ts +14 -0
  22. package/network/server/index.js +219 -0
  23. package/network/server/utils.d.ts +18 -0
  24. package/network/server/utils.js +135 -0
  25. package/network/state.js +3 -5
  26. package/package.json +6 -6
  27. package/server/env-var.d.ts +15 -0
  28. package/server/env-var.js +31 -0
  29. package/server/index.d.ts +2 -0
  30. package/server/index.js +3 -0
  31. package/server/storage/constants.d.ts +1 -0
  32. package/server/storage/constants.js +2 -0
  33. package/server/storage/index.d.ts +21 -0
  34. package/server/storage/index.js +28 -0
  35. package/server/storage/player.d.ts +34 -0
  36. package/server/storage/player.js +61 -0
  37. package/server/storage/scene.d.ts +30 -0
  38. package/server/storage/scene.js +61 -0
  39. package/server/storage-url.d.ts +10 -0
  40. package/server/storage-url.js +29 -0
  41. package/server/utils.d.ts +35 -0
  42. package/server/utils.js +56 -0
  43. package/src/atom.ts +98 -0
  44. package/src/future.ts +38 -0
  45. package/src/network/binary-message-bus.ts +9 -4
  46. package/src/network/chunking.ts +45 -0
  47. package/src/network/events/implementation.ts +286 -0
  48. package/src/network/events/index.ts +48 -0
  49. package/src/network/events/protocol.ts +94 -0
  50. package/src/network/events/registry.ts +18 -0
  51. package/src/network/index.ts +40 -3
  52. package/src/network/message-bus-sync.ts +180 -110
  53. package/src/network/server/index.ts +301 -0
  54. package/src/network/server/utils.ts +189 -0
  55. package/src/network/state.ts +3 -4
  56. package/src/server/env-var.ts +36 -0
  57. package/src/server/index.ts +2 -0
  58. package/src/server/storage/constants.ts +1 -0
  59. package/src/server/storage/index.ts +42 -0
  60. package/src/server/storage/player.ts +106 -0
  61. package/src/server/storage/scene.ts +102 -0
  62. package/src/server/storage-url.ts +34 -0
  63. package/src/server/utils.ts +73 -0
@@ -0,0 +1,2 @@
1
+ export const MODULE_NAME = 'Storage';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NlcnZlci9zdG9yYWdlL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsU0FBUyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IE1PRFVMRV9OQU1FID0gJ1N0b3JhZ2UnXG4iXX0=
@@ -0,0 +1,21 @@
1
+ import { ISceneStorage } from './scene';
2
+ import { IPlayerStorage } from './player';
3
+ export { ISceneStorage } from './scene';
4
+ export { IPlayerStorage } from './player';
5
+ /**
6
+ * Storage interface with methods for scene-scoped and player-scoped storage.
7
+ */
8
+ export interface IStorage extends ISceneStorage {
9
+ /** Player-scoped storage for key-value pairs */
10
+ player: IPlayerStorage;
11
+ }
12
+ /**
13
+ * Storage provides methods to store and retrieve key-value data from the
14
+ * Server Side Storage service.
15
+ *
16
+ * - Use Storage.get/set/delete for scene-scoped storage
17
+ * - Use Storage.player.get/set/delete for player-scoped storage
18
+ *
19
+ * This module only works when running on server-side scenes.
20
+ */
21
+ export declare const Storage: IStorage;
@@ -0,0 +1,28 @@
1
+ import { createSceneStorage } from './scene';
2
+ import { createPlayerStorage } from './player';
3
+ /**
4
+ * Creates the Storage module with scene-scoped and player-scoped storage.
5
+ */
6
+ const createStorage = () => {
7
+ const sceneStorage = createSceneStorage();
8
+ const playerStorage = createPlayerStorage();
9
+ return {
10
+ // Spread scene storage methods at top level
11
+ get: sceneStorage.get,
12
+ set: sceneStorage.set,
13
+ delete: sceneStorage.delete,
14
+ // Keep player as nested property
15
+ player: playerStorage
16
+ };
17
+ };
18
+ /**
19
+ * Storage provides methods to store and retrieve key-value data from the
20
+ * Server Side Storage service.
21
+ *
22
+ * - Use Storage.get/set/delete for scene-scoped storage
23
+ * - Use Storage.player.get/set/delete for player-scoped storage
24
+ *
25
+ * This module only works when running on server-side scenes.
26
+ */
27
+ export const Storage = createStorage();
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VydmVyL3N0b3JhZ2UvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGtCQUFrQixFQUFpQixNQUFNLFNBQVMsQ0FBQTtBQUMzRCxPQUFPLEVBQUUsbUJBQW1CLEVBQWtCLE1BQU0sVUFBVSxDQUFBO0FBYzlEOztHQUVHO0FBQ0gsTUFBTSxhQUFhLEdBQUcsR0FBYSxFQUFFO0lBQ25DLE1BQU0sWUFBWSxHQUFHLGtCQUFrQixFQUFFLENBQUE7SUFDekMsTUFBTSxhQUFhLEdBQUcsbUJBQW1CLEVBQUUsQ0FBQTtJQUUzQyxPQUFPO1FBQ0wsNENBQTRDO1FBQzVDLEdBQUcsRUFBRSxZQUFZLENBQUMsR0FBRztRQUNyQixHQUFHLEVBQUUsWUFBWSxDQUFDLEdBQUc7UUFDckIsTUFBTSxFQUFFLFlBQVksQ0FBQyxNQUFNO1FBQzNCLGlDQUFpQztRQUNqQyxNQUFNLEVBQUUsYUFBYTtLQUN0QixDQUFBO0FBQ0gsQ0FBQyxDQUFBO0FBRUQ7Ozs7Ozs7O0dBUUc7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQWEsYUFBYSxFQUFFLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjcmVhdGVTY2VuZVN0b3JhZ2UsIElTY2VuZVN0b3JhZ2UgfSBmcm9tICcuL3NjZW5lJ1xuaW1wb3J0IHsgY3JlYXRlUGxheWVyU3RvcmFnZSwgSVBsYXllclN0b3JhZ2UgfSBmcm9tICcuL3BsYXllcidcblxuLy8gUmUtZXhwb3J0IGludGVyZmFjZXNcbmV4cG9ydCB7IElTY2VuZVN0b3JhZ2UgfSBmcm9tICcuL3NjZW5lJ1xuZXhwb3J0IHsgSVBsYXllclN0b3JhZ2UgfSBmcm9tICcuL3BsYXllcidcblxuLyoqXG4gKiBTdG9yYWdlIGludGVyZmFjZSB3aXRoIG1ldGhvZHMgZm9yIHNjZW5lLXNjb3BlZCBhbmQgcGxheWVyLXNjb3BlZCBzdG9yYWdlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIElTdG9yYWdlIGV4dGVuZHMgSVNjZW5lU3RvcmFnZSB7XG4gIC8qKiBQbGF5ZXItc2NvcGVkIHN0b3JhZ2UgZm9yIGtleS12YWx1ZSBwYWlycyAqL1xuICBwbGF5ZXI6IElQbGF5ZXJTdG9yYWdlXG59XG5cbi8qKlxuICogQ3JlYXRlcyB0aGUgU3RvcmFnZSBtb2R1bGUgd2l0aCBzY2VuZS1zY29wZWQgYW5kIHBsYXllci1zY29wZWQgc3RvcmFnZS5cbiAqL1xuY29uc3QgY3JlYXRlU3RvcmFnZSA9ICgpOiBJU3RvcmFnZSA9PiB7XG4gIGNvbnN0IHNjZW5lU3RvcmFnZSA9IGNyZWF0ZVNjZW5lU3RvcmFnZSgpXG4gIGNvbnN0IHBsYXllclN0b3JhZ2UgPSBjcmVhdGVQbGF5ZXJTdG9yYWdlKClcblxuICByZXR1cm4ge1xuICAgIC8vIFNwcmVhZCBzY2VuZSBzdG9yYWdlIG1ldGhvZHMgYXQgdG9wIGxldmVsXG4gICAgZ2V0OiBzY2VuZVN0b3JhZ2UuZ2V0LFxuICAgIHNldDogc2NlbmVTdG9yYWdlLnNldCxcbiAgICBkZWxldGU6IHNjZW5lU3RvcmFnZS5kZWxldGUsXG4gICAgLy8gS2VlcCBwbGF5ZXIgYXMgbmVzdGVkIHByb3BlcnR5XG4gICAgcGxheWVyOiBwbGF5ZXJTdG9yYWdlXG4gIH1cbn1cblxuLyoqXG4gKiBTdG9yYWdlIHByb3ZpZGVzIG1ldGhvZHMgdG8gc3RvcmUgYW5kIHJldHJpZXZlIGtleS12YWx1ZSBkYXRhIGZyb20gdGhlXG4gKiBTZXJ2ZXIgU2lkZSBTdG9yYWdlIHNlcnZpY2UuXG4gKlxuICogLSBVc2UgU3RvcmFnZS5nZXQvc2V0L2RlbGV0ZSBmb3Igc2NlbmUtc2NvcGVkIHN0b3JhZ2VcbiAqIC0gVXNlIFN0b3JhZ2UucGxheWVyLmdldC9zZXQvZGVsZXRlIGZvciBwbGF5ZXItc2NvcGVkIHN0b3JhZ2VcbiAqXG4gKiBUaGlzIG1vZHVsZSBvbmx5IHdvcmtzIHdoZW4gcnVubmluZyBvbiBzZXJ2ZXItc2lkZSBzY2VuZXMuXG4gKi9cbmV4cG9ydCBjb25zdCBTdG9yYWdlOiBJU3RvcmFnZSA9IGNyZWF0ZVN0b3JhZ2UoKVxuIl19
@@ -0,0 +1,34 @@
1
+ /**
2
+ * Player-scoped storage interface for key-value pairs from the Server Side Storage service.
3
+ * This is NOT filesystem storage - data is stored in the remote storage service.
4
+ */
5
+ export interface IPlayerStorage {
6
+ /**
7
+ * Retrieves a value from a player's storage by key from the Server Side Storage service.
8
+ * @param address - The player's wallet address
9
+ * @param key - The key to retrieve
10
+ * @returns A promise that resolves to the parsed JSON value, or null if not found
11
+ */
12
+ get<T = unknown>(address: string, key: string): Promise<T | null>;
13
+ /**
14
+ * Stores a value in a player's storage in the Server Side Storage service.
15
+ * @param address - The player's wallet address
16
+ * @param key - The key to store the value under
17
+ * @param value - The value to store (will be JSON serialized)
18
+ * @returns A promise that resolves to true if successful, false otherwise
19
+ */
20
+ set<T = unknown>(address: string, key: string, value: T): Promise<boolean>;
21
+ /**
22
+ * Deletes a value from a player's storage in the Server Side Storage service.
23
+ * @param address - The player's wallet address
24
+ * @param key - The key to delete
25
+ * @returns A promise that resolves to true if deleted, false if not found
26
+ */
27
+ delete(address: string, key: string): Promise<boolean>;
28
+ }
29
+ /**
30
+ * Creates player-scoped storage that provides methods to interact with
31
+ * player-specific key-value pairs from the Server Side Storage service.
32
+ * This module only works when running on server-side scenes.
33
+ */
34
+ export declare const createPlayerStorage: () => IPlayerStorage;
@@ -0,0 +1,61 @@
1
+ import { getStorageServerUrl } from '../storage-url';
2
+ import { assertIsServer, wrapSignedFetch } from '../utils';
3
+ import { MODULE_NAME } from './constants';
4
+ /**
5
+ * Creates player-scoped storage that provides methods to interact with
6
+ * player-specific key-value pairs from the Server Side Storage service.
7
+ * This module only works when running on server-side scenes.
8
+ */
9
+ export const createPlayerStorage = () => {
10
+ return {
11
+ async get(address, key) {
12
+ assertIsServer(MODULE_NAME);
13
+ const baseUrl = await getStorageServerUrl();
14
+ const url = `${baseUrl}/players/${encodeURIComponent(address)}/values/${encodeURIComponent(key)}`;
15
+ const [error, data] = await wrapSignedFetch({ url });
16
+ if (error) {
17
+ console.error(`Failed to get player storage value '${key}' for '${address}': ${error}`);
18
+ return null;
19
+ }
20
+ return data?.value ?? null;
21
+ },
22
+ async set(address, key, value) {
23
+ assertIsServer(MODULE_NAME);
24
+ const baseUrl = await getStorageServerUrl();
25
+ const url = `${baseUrl}/players/${encodeURIComponent(address)}/values/${encodeURIComponent(key)}`;
26
+ const [error] = await wrapSignedFetch({
27
+ url,
28
+ init: {
29
+ method: 'PUT',
30
+ headers: {
31
+ 'content-type': 'application/json'
32
+ },
33
+ body: JSON.stringify({ value })
34
+ }
35
+ });
36
+ if (error) {
37
+ console.error(`Failed to set player storage value '${key}' for '${address}': ${error}`);
38
+ return false;
39
+ }
40
+ return true;
41
+ },
42
+ async delete(address, key) {
43
+ assertIsServer(MODULE_NAME);
44
+ const baseUrl = await getStorageServerUrl();
45
+ const url = `${baseUrl}/players/${encodeURIComponent(address)}/values/${encodeURIComponent(key)}`;
46
+ const [error] = await wrapSignedFetch({
47
+ url,
48
+ init: {
49
+ method: 'DELETE',
50
+ headers: {}
51
+ }
52
+ });
53
+ if (error) {
54
+ console.error(`Failed to delete player storage value '${key}' for '${address}': ${error}`);
55
+ return false;
56
+ }
57
+ return true;
58
+ }
59
+ };
60
+ };
61
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGxheWVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NlcnZlci9zdG9yYWdlL3BsYXllci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQTtBQUNwRCxPQUFPLEVBQUUsY0FBYyxFQUFFLGVBQWUsRUFBRSxNQUFNLFVBQVUsQ0FBQTtBQUMxRCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBaUN6Qzs7OztHQUlHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sbUJBQW1CLEdBQUcsR0FBbUIsRUFBRTtJQUN0RCxPQUFPO1FBQ0wsS0FBSyxDQUFDLEdBQUcsQ0FBYyxPQUFlLEVBQUUsR0FBVztZQUNqRCxjQUFjLENBQUMsV0FBVyxDQUFDLENBQUE7WUFFM0IsTUFBTSxPQUFPLEdBQUcsTUFBTSxtQkFBbUIsRUFBRSxDQUFBO1lBQzNDLE1BQU0sR0FBRyxHQUFHLEdBQUcsT0FBTyxZQUFZLGtCQUFrQixDQUFDLE9BQU8sQ0FBQyxXQUFXLGtCQUFrQixDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUE7WUFFakcsTUFBTSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsR0FBRyxNQUFNLGVBQWUsQ0FBZSxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUE7WUFFbEUsSUFBSSxLQUFLLEVBQUU7Z0JBQ1QsT0FBTyxDQUFDLEtBQUssQ0FBQyx1Q0FBdUMsR0FBRyxVQUFVLE9BQU8sTUFBTSxLQUFLLEVBQUUsQ0FBQyxDQUFBO2dCQUN2RixPQUFPLElBQUksQ0FBQTthQUNaO1lBRUQsT0FBTyxJQUFJLEVBQUUsS0FBSyxJQUFJLElBQUksQ0FBQTtRQUM1QixDQUFDO1FBRUQsS0FBSyxDQUFDLEdBQUcsQ0FBYyxPQUFlLEVBQUUsR0FBVyxFQUFFLEtBQVE7WUFDM0QsY0FBYyxDQUFDLFdBQVcsQ0FBQyxDQUFBO1lBRTNCLE1BQU0sT0FBTyxHQUFHLE1BQU0sbUJBQW1CLEVBQUUsQ0FBQTtZQUMzQyxNQUFNLEdBQUcsR0FBRyxHQUFHLE9BQU8sWUFBWSxrQkFBa0IsQ0FBQyxPQUFPLENBQUMsV0FBVyxrQkFBa0IsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFBO1lBRWpHLE1BQU0sQ0FBQyxLQUFLLENBQUMsR0FBRyxNQUFNLGVBQWUsQ0FBQztnQkFDcEMsR0FBRztnQkFDSCxJQUFJLEVBQUU7b0JBQ0osTUFBTSxFQUFFLEtBQUs7b0JBQ2IsT0FBTyxFQUFFO3dCQUNQLGNBQWMsRUFBRSxrQkFBa0I7cUJBQ25DO29CQUNELElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLEVBQUUsS0FBSyxFQUFFLENBQUM7aUJBQ2hDO2FBQ0YsQ0FBQyxDQUFBO1lBRUYsSUFBSSxLQUFLLEVBQUU7Z0JBQ1QsT0FBTyxDQUFDLEtBQUssQ0FBQyx1Q0FBdUMsR0FBRyxVQUFVLE9BQU8sTUFBTSxLQUFLLEVBQUUsQ0FBQyxDQUFBO2dCQUN2RixPQUFPLEtBQUssQ0FBQTthQUNiO1lBRUQsT0FBTyxJQUFJLENBQUE7UUFDYixDQUFDO1FBRUQsS0FBSyxDQUFDLE1BQU0sQ0FBQyxPQUFlLEVBQUUsR0FBVztZQUN2QyxjQUFjLENBQUMsV0FBVyxDQUFDLENBQUE7WUFFM0IsTUFBTSxPQUFPLEdBQUcsTUFBTSxtQkFBbUIsRUFBRSxDQUFBO1lBQzNDLE1BQU0sR0FBRyxHQUFHLEdBQUcsT0FBTyxZQUFZLGtCQUFrQixDQUFDLE9BQU8sQ0FBQyxXQUFXLGtCQUFrQixDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUE7WUFFakcsTUFBTSxDQUFDLEtBQUssQ0FBQyxHQUFHLE1BQU0sZUFBZSxDQUFDO2dCQUNwQyxHQUFHO2dCQUNILElBQUksRUFBRTtvQkFDSixNQUFNLEVBQUUsUUFBUTtvQkFDaEIsT0FBTyxFQUFFLEVBQUU7aUJBQ1o7YUFDRixDQUFDLENBQUE7WUFFRixJQUFJLEtBQUssRUFBRTtnQkFDVCxPQUFPLENBQUMsS0FBSyxDQUFDLDBDQUEwQyxHQUFHLFVBQVUsT0FBTyxNQUFNLEtBQUssRUFBRSxDQUFDLENBQUE7Z0JBQzFGLE9BQU8sS0FBSyxDQUFBO2FBQ2I7WUFFRCxPQUFPLElBQUksQ0FBQTtRQUNiLENBQUM7S0FDRixDQUFBO0FBQ0gsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZ2V0U3RvcmFnZVNlcnZlclVybCB9IGZyb20gJy4uL3N0b3JhZ2UtdXJsJ1xuaW1wb3J0IHsgYXNzZXJ0SXNTZXJ2ZXIsIHdyYXBTaWduZWRGZXRjaCB9IGZyb20gJy4uL3V0aWxzJ1xuaW1wb3J0IHsgTU9EVUxFX05BTUUgfSBmcm9tICcuL2NvbnN0YW50cydcblxuLyoqXG4gKiBQbGF5ZXItc2NvcGVkIHN0b3JhZ2UgaW50ZXJmYWNlIGZvciBrZXktdmFsdWUgcGFpcnMgZnJvbSB0aGUgU2VydmVyIFNpZGUgU3RvcmFnZSBzZXJ2aWNlLlxuICogVGhpcyBpcyBOT1QgZmlsZXN5c3RlbSBzdG9yYWdlIC0gZGF0YSBpcyBzdG9yZWQgaW4gdGhlIHJlbW90ZSBzdG9yYWdlIHNlcnZpY2UuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSVBsYXllclN0b3JhZ2Uge1xuICAvKipcbiAgICogUmV0cmlldmVzIGEgdmFsdWUgZnJvbSBhIHBsYXllcidzIHN0b3JhZ2UgYnkga2V5IGZyb20gdGhlIFNlcnZlciBTaWRlIFN0b3JhZ2Ugc2VydmljZS5cbiAgICogQHBhcmFtIGFkZHJlc3MgLSBUaGUgcGxheWVyJ3Mgd2FsbGV0IGFkZHJlc3NcbiAgICogQHBhcmFtIGtleSAtIFRoZSBrZXkgdG8gcmV0cmlldmVcbiAgICogQHJldHVybnMgQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIHBhcnNlZCBKU09OIHZhbHVlLCBvciBudWxsIGlmIG5vdCBmb3VuZFxuICAgKi9cbiAgZ2V0PFQgPSB1bmtub3duPihhZGRyZXNzOiBzdHJpbmcsIGtleTogc3RyaW5nKTogUHJvbWlzZTxUIHwgbnVsbD5cblxuICAvKipcbiAgICogU3RvcmVzIGEgdmFsdWUgaW4gYSBwbGF5ZXIncyBzdG9yYWdlIGluIHRoZSBTZXJ2ZXIgU2lkZSBTdG9yYWdlIHNlcnZpY2UuXG4gICAqIEBwYXJhbSBhZGRyZXNzIC0gVGhlIHBsYXllcidzIHdhbGxldCBhZGRyZXNzXG4gICAqIEBwYXJhbSBrZXkgLSBUaGUga2V5IHRvIHN0b3JlIHRoZSB2YWx1ZSB1bmRlclxuICAgKiBAcGFyYW0gdmFsdWUgLSBUaGUgdmFsdWUgdG8gc3RvcmUgKHdpbGwgYmUgSlNPTiBzZXJpYWxpemVkKVxuICAgKiBAcmV0dXJucyBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0cnVlIGlmIHN1Y2Nlc3NmdWwsIGZhbHNlIG90aGVyd2lzZVxuICAgKi9cbiAgc2V0PFQgPSB1bmtub3duPihhZGRyZXNzOiBzdHJpbmcsIGtleTogc3RyaW5nLCB2YWx1ZTogVCk6IFByb21pc2U8Ym9vbGVhbj5cblxuICAvKipcbiAgICogRGVsZXRlcyBhIHZhbHVlIGZyb20gYSBwbGF5ZXIncyBzdG9yYWdlIGluIHRoZSBTZXJ2ZXIgU2lkZSBTdG9yYWdlIHNlcnZpY2UuXG4gICAqIEBwYXJhbSBhZGRyZXNzIC0gVGhlIHBsYXllcidzIHdhbGxldCBhZGRyZXNzXG4gICAqIEBwYXJhbSBrZXkgLSBUaGUga2V5IHRvIGRlbGV0ZVxuICAgKiBAcmV0dXJucyBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0cnVlIGlmIGRlbGV0ZWQsIGZhbHNlIGlmIG5vdCBmb3VuZFxuICAgKi9cbiAgZGVsZXRlKGFkZHJlc3M6IHN0cmluZywga2V5OiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+XG59XG5cbi8qKlxuICogQ3JlYXRlcyBwbGF5ZXItc2NvcGVkIHN0b3JhZ2UgdGhhdCBwcm92aWRlcyBtZXRob2RzIHRvIGludGVyYWN0IHdpdGhcbiAqIHBsYXllci1zcGVjaWZpYyBrZXktdmFsdWUgcGFpcnMgZnJvbSB0aGUgU2VydmVyIFNpZGUgU3RvcmFnZSBzZXJ2aWNlLlxuICogVGhpcyBtb2R1bGUgb25seSB3b3JrcyB3aGVuIHJ1bm5pbmcgb24gc2VydmVyLXNpZGUgc2NlbmVzLlxuICovXG5leHBvcnQgY29uc3QgY3JlYXRlUGxheWVyU3RvcmFnZSA9ICgpOiBJUGxheWVyU3RvcmFnZSA9PiB7XG4gIHJldHVybiB7XG4gICAgYXN5bmMgZ2V0PFQgPSB1bmtub3duPihhZGRyZXNzOiBzdHJpbmcsIGtleTogc3RyaW5nKTogUHJvbWlzZTxUIHwgbnVsbD4ge1xuICAgICAgYXNzZXJ0SXNTZXJ2ZXIoTU9EVUxFX05BTUUpXG5cbiAgICAgIGNvbnN0IGJhc2VVcmwgPSBhd2FpdCBnZXRTdG9yYWdlU2VydmVyVXJsKClcbiAgICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L3BsYXllcnMvJHtlbmNvZGVVUklDb21wb25lbnQoYWRkcmVzcyl9L3ZhbHVlcy8ke2VuY29kZVVSSUNvbXBvbmVudChrZXkpfWBcblxuICAgICAgY29uc3QgW2Vycm9yLCBkYXRhXSA9IGF3YWl0IHdyYXBTaWduZWRGZXRjaDx7IHZhbHVlOiBUIH0+KHsgdXJsIH0pXG5cbiAgICAgIGlmIChlcnJvcikge1xuICAgICAgICBjb25zb2xlLmVycm9yKGBGYWlsZWQgdG8gZ2V0IHBsYXllciBzdG9yYWdlIHZhbHVlICcke2tleX0nIGZvciAnJHthZGRyZXNzfSc6ICR7ZXJyb3J9YClcbiAgICAgICAgcmV0dXJuIG51bGxcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIGRhdGE/LnZhbHVlID8/IG51bGxcbiAgICB9LFxuXG4gICAgYXN5bmMgc2V0PFQgPSB1bmtub3duPihhZGRyZXNzOiBzdHJpbmcsIGtleTogc3RyaW5nLCB2YWx1ZTogVCk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgICAgYXNzZXJ0SXNTZXJ2ZXIoTU9EVUxFX05BTUUpXG5cbiAgICAgIGNvbnN0IGJhc2VVcmwgPSBhd2FpdCBnZXRTdG9yYWdlU2VydmVyVXJsKClcbiAgICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L3BsYXllcnMvJHtlbmNvZGVVUklDb21wb25lbnQoYWRkcmVzcyl9L3ZhbHVlcy8ke2VuY29kZVVSSUNvbXBvbmVudChrZXkpfWBcblxuICAgICAgY29uc3QgW2Vycm9yXSA9IGF3YWl0IHdyYXBTaWduZWRGZXRjaCh7XG4gICAgICAgIHVybCxcbiAgICAgICAgaW5pdDoge1xuICAgICAgICAgIG1ldGhvZDogJ1BVVCcsXG4gICAgICAgICAgaGVhZGVyczoge1xuICAgICAgICAgICAgJ2NvbnRlbnQtdHlwZSc6ICdhcHBsaWNhdGlvbi9qc29uJ1xuICAgICAgICAgIH0sXG4gICAgICAgICAgYm9keTogSlNPTi5zdHJpbmdpZnkoeyB2YWx1ZSB9KVxuICAgICAgICB9XG4gICAgICB9KVxuXG4gICAgICBpZiAoZXJyb3IpIHtcbiAgICAgICAgY29uc29sZS5lcnJvcihgRmFpbGVkIHRvIHNldCBwbGF5ZXIgc3RvcmFnZSB2YWx1ZSAnJHtrZXl9JyBmb3IgJyR7YWRkcmVzc30nOiAke2Vycm9yfWApXG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuXG4gICAgICByZXR1cm4gdHJ1ZVxuICAgIH0sXG5cbiAgICBhc3luYyBkZWxldGUoYWRkcmVzczogc3RyaW5nLCBrZXk6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgICAgYXNzZXJ0SXNTZXJ2ZXIoTU9EVUxFX05BTUUpXG5cbiAgICAgIGNvbnN0IGJhc2VVcmwgPSBhd2FpdCBnZXRTdG9yYWdlU2VydmVyVXJsKClcbiAgICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L3BsYXllcnMvJHtlbmNvZGVVUklDb21wb25lbnQoYWRkcmVzcyl9L3ZhbHVlcy8ke2VuY29kZVVSSUNvbXBvbmVudChrZXkpfWBcblxuICAgICAgY29uc3QgW2Vycm9yXSA9IGF3YWl0IHdyYXBTaWduZWRGZXRjaCh7XG4gICAgICAgIHVybCxcbiAgICAgICAgaW5pdDoge1xuICAgICAgICAgIG1ldGhvZDogJ0RFTEVURScsXG4gICAgICAgICAgaGVhZGVyczoge31cbiAgICAgICAgfVxuICAgICAgfSlcblxuICAgICAgaWYgKGVycm9yKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoYEZhaWxlZCB0byBkZWxldGUgcGxheWVyIHN0b3JhZ2UgdmFsdWUgJyR7a2V5fScgZm9yICcke2FkZHJlc3N9JzogJHtlcnJvcn1gKVxuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIHRydWVcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,30 @@
1
+ /**
2
+ * Scene-scoped storage interface for key-value pairs from the Server Side Storage service.
3
+ * This is NOT filesystem storage - data is stored in the remote storage service.
4
+ */
5
+ export interface ISceneStorage {
6
+ /**
7
+ * Retrieves a value from scene storage by key from the Server Side Storage service.
8
+ * @param key - The key to retrieve
9
+ * @returns A promise that resolves to the parsed JSON value, or null if not found
10
+ */
11
+ get<T = unknown>(key: string): Promise<T | null>;
12
+ /**
13
+ * Stores a value in scene storage in the Server Side Storage service.
14
+ * @param key - The key to store the value under
15
+ * @param value - The value to store (will be JSON serialized)
16
+ */
17
+ set<T = unknown>(key: string, value: T): Promise<boolean>;
18
+ /**
19
+ * Deletes a value from scene storage in the Server Side Storage service.
20
+ * @param key - The key to delete
21
+ * @returns A promise that resolves to true if deleted, false if not found
22
+ */
23
+ delete(key: string): Promise<boolean>;
24
+ }
25
+ /**
26
+ * Creates scene-scoped storage that provides methods to interact with
27
+ * scene-specific key-value pairs from the Server Side Storage service.
28
+ * This module only works when running on server-side scenes.
29
+ */
30
+ export declare const createSceneStorage: () => ISceneStorage;
@@ -0,0 +1,61 @@
1
+ import { getStorageServerUrl } from '../storage-url';
2
+ import { assertIsServer, wrapSignedFetch } from '../utils';
3
+ import { MODULE_NAME } from './constants';
4
+ /**
5
+ * Creates scene-scoped storage that provides methods to interact with
6
+ * scene-specific key-value pairs from the Server Side Storage service.
7
+ * This module only works when running on server-side scenes.
8
+ */
9
+ export const createSceneStorage = () => {
10
+ return {
11
+ async get(key) {
12
+ assertIsServer(MODULE_NAME);
13
+ const baseUrl = await getStorageServerUrl();
14
+ const url = `${baseUrl}/values/${encodeURIComponent(key)}`;
15
+ const [error, data] = await wrapSignedFetch({ url });
16
+ if (error) {
17
+ console.error(`Failed to get storage value '${key}': ${error}`);
18
+ return null;
19
+ }
20
+ return data?.value ?? null;
21
+ },
22
+ async set(key, value) {
23
+ assertIsServer(MODULE_NAME);
24
+ const baseUrl = await getStorageServerUrl();
25
+ const url = `${baseUrl}/values/${encodeURIComponent(key)}`;
26
+ const [error] = await wrapSignedFetch({
27
+ url,
28
+ init: {
29
+ method: 'PUT',
30
+ headers: {
31
+ 'content-type': 'application/json'
32
+ },
33
+ body: JSON.stringify({ value })
34
+ }
35
+ });
36
+ if (error) {
37
+ console.error(`Failed to set storage value '${key}': ${error}`);
38
+ return false;
39
+ }
40
+ return true;
41
+ },
42
+ async delete(key) {
43
+ assertIsServer(MODULE_NAME);
44
+ const baseUrl = await getStorageServerUrl();
45
+ const url = `${baseUrl}/values/${encodeURIComponent(key)}`;
46
+ const [error] = await wrapSignedFetch({
47
+ url,
48
+ init: {
49
+ method: 'DELETE',
50
+ headers: {}
51
+ }
52
+ });
53
+ if (error) {
54
+ console.error(`Failed to delete storage value '${key}': ${error}`);
55
+ return false;
56
+ }
57
+ return true;
58
+ }
59
+ };
60
+ };
61
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NlbmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VydmVyL3N0b3JhZ2Uvc2NlbmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUE7QUFDcEQsT0FBTyxFQUFFLGNBQWMsRUFBRSxlQUFlLEVBQUUsTUFBTSxVQUFVLENBQUE7QUFDMUQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQTZCekM7Ozs7R0FJRztBQUNILE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLEdBQWtCLEVBQUU7SUFDcEQsT0FBTztRQUNMLEtBQUssQ0FBQyxHQUFHLENBQWMsR0FBVztZQUNoQyxjQUFjLENBQUMsV0FBVyxDQUFDLENBQUE7WUFFM0IsTUFBTSxPQUFPLEdBQUcsTUFBTSxtQkFBbUIsRUFBRSxDQUFBO1lBQzNDLE1BQU0sR0FBRyxHQUFHLEdBQUcsT0FBTyxXQUFXLGtCQUFrQixDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUE7WUFFMUQsTUFBTSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsR0FBRyxNQUFNLGVBQWUsQ0FBZSxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUE7WUFFbEUsSUFBSSxLQUFLLEVBQUU7Z0JBQ1QsT0FBTyxDQUFDLEtBQUssQ0FBQyxnQ0FBZ0MsR0FBRyxNQUFNLEtBQUssRUFBRSxDQUFDLENBQUE7Z0JBQy9ELE9BQU8sSUFBSSxDQUFBO2FBQ1o7WUFFRCxPQUFPLElBQUksRUFBRSxLQUFLLElBQUksSUFBSSxDQUFBO1FBQzVCLENBQUM7UUFFRCxLQUFLLENBQUMsR0FBRyxDQUFjLEdBQVcsRUFBRSxLQUFRO1lBQzFDLGNBQWMsQ0FBQyxXQUFXLENBQUMsQ0FBQTtZQUUzQixNQUFNLE9BQU8sR0FBRyxNQUFNLG1CQUFtQixFQUFFLENBQUE7WUFDM0MsTUFBTSxHQUFHLEdBQUcsR0FBRyxPQUFPLFdBQVcsa0JBQWtCLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQTtZQUUxRCxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsTUFBTSxlQUFlLENBQUM7Z0JBQ3BDLEdBQUc7Z0JBQ0gsSUFBSSxFQUFFO29CQUNKLE1BQU0sRUFBRSxLQUFLO29CQUNiLE9BQU8sRUFBRTt3QkFDUCxjQUFjLEVBQUUsa0JBQWtCO3FCQUNuQztvQkFDRCxJQUFJLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxFQUFFLEtBQUssRUFBRSxDQUFDO2lCQUNoQzthQUNGLENBQUMsQ0FBQTtZQUVGLElBQUksS0FBSyxFQUFFO2dCQUNULE9BQU8sQ0FBQyxLQUFLLENBQUMsZ0NBQWdDLEdBQUcsTUFBTSxLQUFLLEVBQUUsQ0FBQyxDQUFBO2dCQUMvRCxPQUFPLEtBQUssQ0FBQTthQUNiO1lBRUQsT0FBTyxJQUFJLENBQUE7UUFDYixDQUFDO1FBRUQsS0FBSyxDQUFDLE1BQU0sQ0FBQyxHQUFXO1lBQ3RCLGNBQWMsQ0FBQyxXQUFXLENBQUMsQ0FBQTtZQUUzQixNQUFNLE9BQU8sR0FBRyxNQUFNLG1CQUFtQixFQUFFLENBQUE7WUFDM0MsTUFBTSxHQUFHLEdBQUcsR0FBRyxPQUFPLFdBQVcsa0JBQWtCLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQTtZQUUxRCxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsTUFBTSxlQUFlLENBQUM7Z0JBQ3BDLEdBQUc7Z0JBQ0gsSUFBSSxFQUFFO29CQUNKLE1BQU0sRUFBRSxRQUFRO29CQUNoQixPQUFPLEVBQUUsRUFBRTtpQkFDWjthQUNGLENBQUMsQ0FBQTtZQUVGLElBQUksS0FBSyxFQUFFO2dCQUNULE9BQU8sQ0FBQyxLQUFLLENBQUMsbUNBQW1DLEdBQUcsTUFBTSxLQUFLLEVBQUUsQ0FBQyxDQUFBO2dCQUNsRSxPQUFPLEtBQUssQ0FBQTthQUNiO1lBRUQsT0FBTyxJQUFJLENBQUE7UUFDYixDQUFDO0tBQ0YsQ0FBQTtBQUNILENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGdldFN0b3JhZ2VTZXJ2ZXJVcmwgfSBmcm9tICcuLi9zdG9yYWdlLXVybCdcbmltcG9ydCB7IGFzc2VydElzU2VydmVyLCB3cmFwU2lnbmVkRmV0Y2ggfSBmcm9tICcuLi91dGlscydcbmltcG9ydCB7IE1PRFVMRV9OQU1FIH0gZnJvbSAnLi9jb25zdGFudHMnXG5cbi8qKlxuICogU2NlbmUtc2NvcGVkIHN0b3JhZ2UgaW50ZXJmYWNlIGZvciBrZXktdmFsdWUgcGFpcnMgZnJvbSB0aGUgU2VydmVyIFNpZGUgU3RvcmFnZSBzZXJ2aWNlLlxuICogVGhpcyBpcyBOT1QgZmlsZXN5c3RlbSBzdG9yYWdlIC0gZGF0YSBpcyBzdG9yZWQgaW4gdGhlIHJlbW90ZSBzdG9yYWdlIHNlcnZpY2UuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSVNjZW5lU3RvcmFnZSB7XG4gIC8qKlxuICAgKiBSZXRyaWV2ZXMgYSB2YWx1ZSBmcm9tIHNjZW5lIHN0b3JhZ2UgYnkga2V5IGZyb20gdGhlIFNlcnZlciBTaWRlIFN0b3JhZ2Ugc2VydmljZS5cbiAgICogQHBhcmFtIGtleSAtIFRoZSBrZXkgdG8gcmV0cmlldmVcbiAgICogQHJldHVybnMgQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIHBhcnNlZCBKU09OIHZhbHVlLCBvciBudWxsIGlmIG5vdCBmb3VuZFxuICAgKi9cbiAgZ2V0PFQgPSB1bmtub3duPihrZXk6IHN0cmluZyk6IFByb21pc2U8VCB8IG51bGw+XG5cbiAgLyoqXG4gICAqIFN0b3JlcyBhIHZhbHVlIGluIHNjZW5lIHN0b3JhZ2UgaW4gdGhlIFNlcnZlciBTaWRlIFN0b3JhZ2Ugc2VydmljZS5cbiAgICogQHBhcmFtIGtleSAtIFRoZSBrZXkgdG8gc3RvcmUgdGhlIHZhbHVlIHVuZGVyXG4gICAqIEBwYXJhbSB2YWx1ZSAtIFRoZSB2YWx1ZSB0byBzdG9yZSAod2lsbCBiZSBKU09OIHNlcmlhbGl6ZWQpXG4gICAqL1xuICBzZXQ8VCA9IHVua25vd24+KGtleTogc3RyaW5nLCB2YWx1ZTogVCk6IFByb21pc2U8Ym9vbGVhbj5cblxuICAvKipcbiAgICogRGVsZXRlcyBhIHZhbHVlIGZyb20gc2NlbmUgc3RvcmFnZSBpbiB0aGUgU2VydmVyIFNpZGUgU3RvcmFnZSBzZXJ2aWNlLlxuICAgKiBAcGFyYW0ga2V5IC0gVGhlIGtleSB0byBkZWxldGVcbiAgICogQHJldHVybnMgQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdHJ1ZSBpZiBkZWxldGVkLCBmYWxzZSBpZiBub3QgZm91bmRcbiAgICovXG4gIGRlbGV0ZShrZXk6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj5cbn1cblxuLyoqXG4gKiBDcmVhdGVzIHNjZW5lLXNjb3BlZCBzdG9yYWdlIHRoYXQgcHJvdmlkZXMgbWV0aG9kcyB0byBpbnRlcmFjdCB3aXRoXG4gKiBzY2VuZS1zcGVjaWZpYyBrZXktdmFsdWUgcGFpcnMgZnJvbSB0aGUgU2VydmVyIFNpZGUgU3RvcmFnZSBzZXJ2aWNlLlxuICogVGhpcyBtb2R1bGUgb25seSB3b3JrcyB3aGVuIHJ1bm5pbmcgb24gc2VydmVyLXNpZGUgc2NlbmVzLlxuICovXG5leHBvcnQgY29uc3QgY3JlYXRlU2NlbmVTdG9yYWdlID0gKCk6IElTY2VuZVN0b3JhZ2UgPT4ge1xuICByZXR1cm4ge1xuICAgIGFzeW5jIGdldDxUID0gdW5rbm93bj4oa2V5OiBzdHJpbmcpOiBQcm9taXNlPFQgfCBudWxsPiB7XG4gICAgICBhc3NlcnRJc1NlcnZlcihNT0RVTEVfTkFNRSlcblxuICAgICAgY29uc3QgYmFzZVVybCA9IGF3YWl0IGdldFN0b3JhZ2VTZXJ2ZXJVcmwoKVxuICAgICAgY29uc3QgdXJsID0gYCR7YmFzZVVybH0vdmFsdWVzLyR7ZW5jb2RlVVJJQ29tcG9uZW50KGtleSl9YFxuXG4gICAgICBjb25zdCBbZXJyb3IsIGRhdGFdID0gYXdhaXQgd3JhcFNpZ25lZEZldGNoPHsgdmFsdWU6IFQgfT4oeyB1cmwgfSlcblxuICAgICAgaWYgKGVycm9yKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoYEZhaWxlZCB0byBnZXQgc3RvcmFnZSB2YWx1ZSAnJHtrZXl9JzogJHtlcnJvcn1gKVxuICAgICAgICByZXR1cm4gbnVsbFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gZGF0YT8udmFsdWUgPz8gbnVsbFxuICAgIH0sXG5cbiAgICBhc3luYyBzZXQ8VCA9IHVua25vd24+KGtleTogc3RyaW5nLCB2YWx1ZTogVCk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgICAgYXNzZXJ0SXNTZXJ2ZXIoTU9EVUxFX05BTUUpXG5cbiAgICAgIGNvbnN0IGJhc2VVcmwgPSBhd2FpdCBnZXRTdG9yYWdlU2VydmVyVXJsKClcbiAgICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L3ZhbHVlcy8ke2VuY29kZVVSSUNvbXBvbmVudChrZXkpfWBcblxuICAgICAgY29uc3QgW2Vycm9yXSA9IGF3YWl0IHdyYXBTaWduZWRGZXRjaCh7XG4gICAgICAgIHVybCxcbiAgICAgICAgaW5pdDoge1xuICAgICAgICAgIG1ldGhvZDogJ1BVVCcsXG4gICAgICAgICAgaGVhZGVyczoge1xuICAgICAgICAgICAgJ2NvbnRlbnQtdHlwZSc6ICdhcHBsaWNhdGlvbi9qc29uJ1xuICAgICAgICAgIH0sXG4gICAgICAgICAgYm9keTogSlNPTi5zdHJpbmdpZnkoeyB2YWx1ZSB9KVxuICAgICAgICB9XG4gICAgICB9KVxuXG4gICAgICBpZiAoZXJyb3IpIHtcbiAgICAgICAgY29uc29sZS5lcnJvcihgRmFpbGVkIHRvIHNldCBzdG9yYWdlIHZhbHVlICcke2tleX0nOiAke2Vycm9yfWApXG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuXG4gICAgICByZXR1cm4gdHJ1ZVxuICAgIH0sXG5cbiAgICBhc3luYyBkZWxldGUoa2V5OiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+IHtcbiAgICAgIGFzc2VydElzU2VydmVyKE1PRFVMRV9OQU1FKVxuXG4gICAgICBjb25zdCBiYXNlVXJsID0gYXdhaXQgZ2V0U3RvcmFnZVNlcnZlclVybCgpXG4gICAgICBjb25zdCB1cmwgPSBgJHtiYXNlVXJsfS92YWx1ZXMvJHtlbmNvZGVVUklDb21wb25lbnQoa2V5KX1gXG5cbiAgICAgIGNvbnN0IFtlcnJvcl0gPSBhd2FpdCB3cmFwU2lnbmVkRmV0Y2goe1xuICAgICAgICB1cmwsXG4gICAgICAgIGluaXQ6IHtcbiAgICAgICAgICBtZXRob2Q6ICdERUxFVEUnLFxuICAgICAgICAgIGhlYWRlcnM6IHt9XG4gICAgICAgIH1cbiAgICAgIH0pXG5cbiAgICAgIGlmIChlcnJvcikge1xuICAgICAgICBjb25zb2xlLmVycm9yKGBGYWlsZWQgdG8gZGVsZXRlIHN0b3JhZ2UgdmFsdWUgJyR7a2V5fSc6ICR7ZXJyb3J9YClcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICB9XG5cbiAgICAgIHJldHVybiB0cnVlXG4gICAgfVxuICB9XG59XG4iXX0=
@@ -0,0 +1,10 @@
1
+ /**
2
+ * Determines the correct storage server URL based on the current realm.
3
+ *
4
+ * - If `isPreview` is true, uses the realm's baseUrl (localhost)
5
+ * - If the realm's baseUrl contains `.zone`, uses storage.decentraland.zone
6
+ * - Otherwise, uses storage.decentraland.org (production)
7
+ *
8
+ * @returns The storage server base URL
9
+ */
10
+ export declare function getStorageServerUrl(): Promise<string>;
@@ -0,0 +1,29 @@
1
+ import { getRealm } from '~system/Runtime';
2
+ const STORAGE_SERVER_ORG = 'https://storage.decentraland.org';
3
+ const STORAGE_SERVER_ZONE = 'https://storage.decentraland.zone';
4
+ /**
5
+ * Determines the correct storage server URL based on the current realm.
6
+ *
7
+ * - If `isPreview` is true, uses the realm's baseUrl (localhost)
8
+ * - If the realm's baseUrl contains `.zone`, uses storage.decentraland.zone
9
+ * - Otherwise, uses storage.decentraland.org (production)
10
+ *
11
+ * @returns The storage server base URL
12
+ */
13
+ export async function getStorageServerUrl() {
14
+ const { realmInfo } = await getRealm({});
15
+ if (!realmInfo) {
16
+ throw new Error('Unable to retrieve realm information');
17
+ }
18
+ // Local development / preview mode
19
+ if (realmInfo.isPreview) {
20
+ return realmInfo.baseUrl;
21
+ }
22
+ // Staging / testing environment
23
+ if (realmInfo.baseUrl.includes('.zone')) {
24
+ return STORAGE_SERVER_ZONE;
25
+ }
26
+ // Production environment
27
+ return STORAGE_SERVER_ORG;
28
+ }
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZS11cmwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvc2VydmVyL3N0b3JhZ2UtdXJsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQTtBQUUxQyxNQUFNLGtCQUFrQixHQUFHLGtDQUFrQyxDQUFBO0FBQzdELE1BQU0sbUJBQW1CLEdBQUcsbUNBQW1DLENBQUE7QUFFL0Q7Ozs7Ozs7O0dBUUc7QUFDSCxNQUFNLENBQUMsS0FBSyxVQUFVLG1CQUFtQjtJQUN2QyxNQUFNLEVBQUUsU0FBUyxFQUFFLEdBQUcsTUFBTSxRQUFRLENBQUMsRUFBRSxDQUFDLENBQUE7SUFFeEMsSUFBSSxDQUFDLFNBQVMsRUFBRTtRQUNkLE1BQU0sSUFBSSxLQUFLLENBQUMsc0NBQXNDLENBQUMsQ0FBQTtLQUN4RDtJQUVELG1DQUFtQztJQUNuQyxJQUFJLFNBQVMsQ0FBQyxTQUFTLEVBQUU7UUFDdkIsT0FBTyxTQUFTLENBQUMsT0FBTyxDQUFBO0tBQ3pCO0lBRUQsZ0NBQWdDO0lBQ2hDLElBQUksU0FBUyxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLEVBQUU7UUFDdkMsT0FBTyxtQkFBbUIsQ0FBQTtLQUMzQjtJQUVELHlCQUF5QjtJQUN6QixPQUFPLGtCQUFrQixDQUFBO0FBQzNCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBnZXRSZWFsbSB9IGZyb20gJ35zeXN0ZW0vUnVudGltZSdcblxuY29uc3QgU1RPUkFHRV9TRVJWRVJfT1JHID0gJ2h0dHBzOi8vc3RvcmFnZS5kZWNlbnRyYWxhbmQub3JnJ1xuY29uc3QgU1RPUkFHRV9TRVJWRVJfWk9ORSA9ICdodHRwczovL3N0b3JhZ2UuZGVjZW50cmFsYW5kLnpvbmUnXG5cbi8qKlxuICogRGV0ZXJtaW5lcyB0aGUgY29ycmVjdCBzdG9yYWdlIHNlcnZlciBVUkwgYmFzZWQgb24gdGhlIGN1cnJlbnQgcmVhbG0uXG4gKlxuICogLSBJZiBgaXNQcmV2aWV3YCBpcyB0cnVlLCB1c2VzIHRoZSByZWFsbSdzIGJhc2VVcmwgKGxvY2FsaG9zdClcbiAqIC0gSWYgdGhlIHJlYWxtJ3MgYmFzZVVybCBjb250YWlucyBgLnpvbmVgLCB1c2VzIHN0b3JhZ2UuZGVjZW50cmFsYW5kLnpvbmVcbiAqIC0gT3RoZXJ3aXNlLCB1c2VzIHN0b3JhZ2UuZGVjZW50cmFsYW5kLm9yZyAocHJvZHVjdGlvbilcbiAqXG4gKiBAcmV0dXJucyBUaGUgc3RvcmFnZSBzZXJ2ZXIgYmFzZSBVUkxcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldFN0b3JhZ2VTZXJ2ZXJVcmwoKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgY29uc3QgeyByZWFsbUluZm8gfSA9IGF3YWl0IGdldFJlYWxtKHt9KVxuXG4gIGlmICghcmVhbG1JbmZvKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKCdVbmFibGUgdG8gcmV0cmlldmUgcmVhbG0gaW5mb3JtYXRpb24nKVxuICB9XG5cbiAgLy8gTG9jYWwgZGV2ZWxvcG1lbnQgLyBwcmV2aWV3IG1vZGVcbiAgaWYgKHJlYWxtSW5mby5pc1ByZXZpZXcpIHtcbiAgICByZXR1cm4gcmVhbG1JbmZvLmJhc2VVcmxcbiAgfVxuXG4gIC8vIFN0YWdpbmcgLyB0ZXN0aW5nIGVudmlyb25tZW50XG4gIGlmIChyZWFsbUluZm8uYmFzZVVybC5pbmNsdWRlcygnLnpvbmUnKSkge1xuICAgIHJldHVybiBTVE9SQUdFX1NFUlZFUl9aT05FXG4gIH1cblxuICAvLyBQcm9kdWN0aW9uIGVudmlyb25tZW50XG4gIHJldHVybiBTVE9SQUdFX1NFUlZFUl9PUkdcbn1cbiJdfQ==
@@ -0,0 +1,35 @@
1
+ /// <reference types="@dcl/js-runtime" />
2
+ import { SignedFetchRequest } from '~system/SignedFetch';
3
+ /**
4
+ * Validates that the code is running on a server-side scene.
5
+ * Throws an error if called from a client-side context.
6
+ *
7
+ * @param moduleName - The name of the module for the error message
8
+ * @throws Error if not running on a server-side scene
9
+ */
10
+ export declare function assertIsServer(moduleName: string): void;
11
+ /**
12
+ * Result type for operations that can fail.
13
+ * Returns a tuple of [error, null] on failure or [null, data] on success.
14
+ */
15
+ export type Result<T, E = string> = [E, null] | [null, T];
16
+ /**
17
+ * Extended result type that includes HTTP status code information.
18
+ */
19
+ export type FetchResult<T> = [string, null, number?] | [null, T, number];
20
+ /**
21
+ * Wraps a promise to catch errors and return a Result tuple.
22
+ * This allows for cleaner error handling without try-catch blocks.
23
+ *
24
+ * @param promise - The promise to wrap
25
+ * @returns A tuple of [error, null] on failure or [null, data] on success
26
+ */
27
+ export declare function tryCatch<T, E = Error>(promise: Promise<T>): Promise<Result<T, E>>;
28
+ /**
29
+ * Wraps signedFetch with automatic error handling and JSON parsing.
30
+ * Returns a FetchResult tuple with parsed JSON data or error message and status code.
31
+ *
32
+ * @param signedFetchBody - The signedFetch request configuration
33
+ * @returns A tuple of [error, null, statusCode?] on failure or [null, data, statusCode] on success
34
+ */
35
+ export declare function wrapSignedFetch<T = unknown>(signedFetchBody: SignedFetchRequest): Promise<FetchResult<T>>;
@@ -0,0 +1,56 @@
1
+ import { signedFetch } from '~system/SignedFetch';
2
+ import { isServer } from '../network';
3
+ /**
4
+ * Validates that the code is running on a server-side scene.
5
+ * Throws an error if called from a client-side context.
6
+ *
7
+ * @param moduleName - The name of the module for the error message
8
+ * @throws Error if not running on a server-side scene
9
+ */
10
+ export function assertIsServer(moduleName) {
11
+ if (!isServer()) {
12
+ throw new Error(`${moduleName} is only available on server-side scenes`);
13
+ }
14
+ }
15
+ /**
16
+ * Wraps a promise to catch errors and return a Result tuple.
17
+ * This allows for cleaner error handling without try-catch blocks.
18
+ *
19
+ * @param promise - The promise to wrap
20
+ * @returns A tuple of [error, null] on failure or [null, data] on success
21
+ */
22
+ export async function tryCatch(promise) {
23
+ try {
24
+ const data = await promise;
25
+ return [null, data];
26
+ }
27
+ catch (error) {
28
+ return [error, null];
29
+ }
30
+ }
31
+ /**
32
+ * Wraps signedFetch with automatic error handling and JSON parsing.
33
+ * Returns a FetchResult tuple with parsed JSON data or error message and status code.
34
+ *
35
+ * @param signedFetchBody - The signedFetch request configuration
36
+ * @returns A tuple of [error, null, statusCode?] on failure or [null, data, statusCode] on success
37
+ */
38
+ export async function wrapSignedFetch(signedFetchBody) {
39
+ const [error, response] = await tryCatch(signedFetch(signedFetchBody));
40
+ if (error) {
41
+ console.error(`Error in ${signedFetchBody.url} endpoint`, { error });
42
+ return [error.message, null, undefined];
43
+ }
44
+ if (!response.ok) {
45
+ const errorMessage = `${response.status} ${response.statusText}`;
46
+ console.error(`Error in ${signedFetchBody.url} endpoint`, { response });
47
+ return [errorMessage, null, response.status];
48
+ }
49
+ const [parseError, body] = await tryCatch(JSON.parse(response.body || '{}'));
50
+ if (parseError) {
51
+ console.error(`Failed to parse response from ${signedFetchBody.url}`);
52
+ return ['Failed to parse response', null, response.status];
53
+ }
54
+ return [null, (body ?? {}), response.status];
55
+ }
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvc2VydmVyL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxXQUFXLEVBQXNCLE1BQU0scUJBQXFCLENBQUE7QUFDckUsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLFlBQVksQ0FBQTtBQUVyQzs7Ozs7O0dBTUc7QUFDSCxNQUFNLFVBQVUsY0FBYyxDQUFDLFVBQWtCO0lBQy9DLElBQUksQ0FBQyxRQUFRLEVBQUUsRUFBRTtRQUNmLE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxVQUFVLDBDQUEwQyxDQUFDLENBQUE7S0FDekU7QUFDSCxDQUFDO0FBYUQ7Ozs7OztHQU1HO0FBQ0gsTUFBTSxDQUFDLEtBQUssVUFBVSxRQUFRLENBQWUsT0FBbUI7SUFDOUQsSUFBSTtRQUNGLE1BQU0sSUFBSSxHQUFHLE1BQU0sT0FBTyxDQUFBO1FBQzFCLE9BQU8sQ0FBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLENBQUE7S0FDcEI7SUFBQyxPQUFPLEtBQUssRUFBRTtRQUNkLE9BQU8sQ0FBQyxLQUFVLEVBQUUsSUFBSSxDQUFDLENBQUE7S0FDMUI7QUFDSCxDQUFDO0FBRUQ7Ozs7OztHQU1HO0FBQ0gsTUFBTSxDQUFDLEtBQUssVUFBVSxlQUFlLENBQWMsZUFBbUM7SUFDcEYsTUFBTSxDQUFDLEtBQUssRUFBRSxRQUFRLENBQUMsR0FBRyxNQUFNLFFBQVEsQ0FBQyxXQUFXLENBQUMsZUFBZSxDQUFDLENBQUMsQ0FBQTtJQUV0RSxJQUFJLEtBQUssRUFBRTtRQUNULE9BQU8sQ0FBQyxLQUFLLENBQUMsWUFBWSxlQUFlLENBQUMsR0FBRyxXQUFXLEVBQUUsRUFBRSxLQUFLLEVBQUUsQ0FBQyxDQUFBO1FBQ3BFLE9BQU8sQ0FBQyxLQUFLLENBQUMsT0FBTyxFQUFFLElBQUksRUFBRSxTQUFTLENBQUMsQ0FBQTtLQUN4QztJQUVELElBQUksQ0FBQyxRQUFRLENBQUMsRUFBRSxFQUFFO1FBQ2hCLE1BQU0sWUFBWSxHQUFHLEdBQUcsUUFBUSxDQUFDLE1BQU0sSUFBSSxRQUFRLENBQUMsVUFBVSxFQUFFLENBQUE7UUFDaEUsT0FBTyxDQUFDLEtBQUssQ0FBQyxZQUFZLGVBQWUsQ0FBQyxHQUFHLFdBQVcsRUFBRSxFQUFFLFFBQVEsRUFBRSxDQUFDLENBQUE7UUFDdkUsT0FBTyxDQUFDLFlBQVksRUFBRSxJQUFJLEVBQUUsUUFBUSxDQUFDLE1BQU0sQ0FBQyxDQUFBO0tBQzdDO0lBRUQsTUFBTSxDQUFDLFVBQVUsRUFBRSxJQUFJLENBQUMsR0FBRyxNQUFNLFFBQVEsQ0FBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxJQUFJLElBQUksSUFBSSxDQUFDLENBQUMsQ0FBQTtJQUUvRSxJQUFJLFVBQVUsRUFBRTtRQUNkLE9BQU8sQ0FBQyxLQUFLLENBQUMsaUNBQWlDLGVBQWUsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFBO1FBQ3JFLE9BQU8sQ0FBQywwQkFBMEIsRUFBRSxJQUFJLEVBQUUsUUFBUSxDQUFDLE1BQU0sQ0FBQyxDQUFBO0tBQzNEO0lBRUQsT0FBTyxDQUFDLElBQUksRUFBRSxDQUFDLElBQUksSUFBSSxFQUFFLENBQU0sRUFBRSxRQUFRLENBQUMsTUFBTSxDQUFDLENBQUE7QUFDbkQsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHNpZ25lZEZldGNoLCBTaWduZWRGZXRjaFJlcXVlc3QgfSBmcm9tICd+c3lzdGVtL1NpZ25lZEZldGNoJ1xuaW1wb3J0IHsgaXNTZXJ2ZXIgfSBmcm9tICcuLi9uZXR3b3JrJ1xuXG4vKipcbiAqIFZhbGlkYXRlcyB0aGF0IHRoZSBjb2RlIGlzIHJ1bm5pbmcgb24gYSBzZXJ2ZXItc2lkZSBzY2VuZS5cbiAqIFRocm93cyBhbiBlcnJvciBpZiBjYWxsZWQgZnJvbSBhIGNsaWVudC1zaWRlIGNvbnRleHQuXG4gKlxuICogQHBhcmFtIG1vZHVsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgbW9kdWxlIGZvciB0aGUgZXJyb3IgbWVzc2FnZVxuICogQHRocm93cyBFcnJvciBpZiBub3QgcnVubmluZyBvbiBhIHNlcnZlci1zaWRlIHNjZW5lXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBhc3NlcnRJc1NlcnZlcihtb2R1bGVOYW1lOiBzdHJpbmcpOiB2b2lkIHtcbiAgaWYgKCFpc1NlcnZlcigpKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGAke21vZHVsZU5hbWV9IGlzIG9ubHkgYXZhaWxhYmxlIG9uIHNlcnZlci1zaWRlIHNjZW5lc2ApXG4gIH1cbn1cblxuLyoqXG4gKiBSZXN1bHQgdHlwZSBmb3Igb3BlcmF0aW9ucyB0aGF0IGNhbiBmYWlsLlxuICogUmV0dXJucyBhIHR1cGxlIG9mIFtlcnJvciwgbnVsbF0gb24gZmFpbHVyZSBvciBbbnVsbCwgZGF0YV0gb24gc3VjY2Vzcy5cbiAqL1xuZXhwb3J0IHR5cGUgUmVzdWx0PFQsIEUgPSBzdHJpbmc+ID0gW0UsIG51bGxdIHwgW251bGwsIFRdXG5cbi8qKlxuICogRXh0ZW5kZWQgcmVzdWx0IHR5cGUgdGhhdCBpbmNsdWRlcyBIVFRQIHN0YXR1cyBjb2RlIGluZm9ybWF0aW9uLlxuICovXG5leHBvcnQgdHlwZSBGZXRjaFJlc3VsdDxUPiA9IFtzdHJpbmcsIG51bGwsIG51bWJlcj9dIHwgW251bGwsIFQsIG51bWJlcl1cblxuLyoqXG4gKiBXcmFwcyBhIHByb21pc2UgdG8gY2F0Y2ggZXJyb3JzIGFuZCByZXR1cm4gYSBSZXN1bHQgdHVwbGUuXG4gKiBUaGlzIGFsbG93cyBmb3IgY2xlYW5lciBlcnJvciBoYW5kbGluZyB3aXRob3V0IHRyeS1jYXRjaCBibG9ja3MuXG4gKlxuICogQHBhcmFtIHByb21pc2UgLSBUaGUgcHJvbWlzZSB0byB3cmFwXG4gKiBAcmV0dXJucyBBIHR1cGxlIG9mIFtlcnJvciwgbnVsbF0gb24gZmFpbHVyZSBvciBbbnVsbCwgZGF0YV0gb24gc3VjY2Vzc1xuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gdHJ5Q2F0Y2g8VCwgRSA9IEVycm9yPihwcm9taXNlOiBQcm9taXNlPFQ+KTogUHJvbWlzZTxSZXN1bHQ8VCwgRT4+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBkYXRhID0gYXdhaXQgcHJvbWlzZVxuICAgIHJldHVybiBbbnVsbCwgZGF0YV1cbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICByZXR1cm4gW2Vycm9yIGFzIEUsIG51bGxdXG4gIH1cbn1cblxuLyoqXG4gKiBXcmFwcyBzaWduZWRGZXRjaCB3aXRoIGF1dG9tYXRpYyBlcnJvciBoYW5kbGluZyBhbmQgSlNPTiBwYXJzaW5nLlxuICogUmV0dXJucyBhIEZldGNoUmVzdWx0IHR1cGxlIHdpdGggcGFyc2VkIEpTT04gZGF0YSBvciBlcnJvciBtZXNzYWdlIGFuZCBzdGF0dXMgY29kZS5cbiAqXG4gKiBAcGFyYW0gc2lnbmVkRmV0Y2hCb2R5IC0gVGhlIHNpZ25lZEZldGNoIHJlcXVlc3QgY29uZmlndXJhdGlvblxuICogQHJldHVybnMgQSB0dXBsZSBvZiBbZXJyb3IsIG51bGwsIHN0YXR1c0NvZGU/XSBvbiBmYWlsdXJlIG9yIFtudWxsLCBkYXRhLCBzdGF0dXNDb2RlXSBvbiBzdWNjZXNzXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB3cmFwU2lnbmVkRmV0Y2g8VCA9IHVua25vd24+KHNpZ25lZEZldGNoQm9keTogU2lnbmVkRmV0Y2hSZXF1ZXN0KTogUHJvbWlzZTxGZXRjaFJlc3VsdDxUPj4ge1xuICBjb25zdCBbZXJyb3IsIHJlc3BvbnNlXSA9IGF3YWl0IHRyeUNhdGNoKHNpZ25lZEZldGNoKHNpZ25lZEZldGNoQm9keSkpXG5cbiAgaWYgKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihgRXJyb3IgaW4gJHtzaWduZWRGZXRjaEJvZHkudXJsfSBlbmRwb2ludGAsIHsgZXJyb3IgfSlcbiAgICByZXR1cm4gW2Vycm9yLm1lc3NhZ2UsIG51bGwsIHVuZGVmaW5lZF1cbiAgfVxuXG4gIGlmICghcmVzcG9uc2Uub2spIHtcbiAgICBjb25zdCBlcnJvck1lc3NhZ2UgPSBgJHtyZXNwb25zZS5zdGF0dXN9ICR7cmVzcG9uc2Uuc3RhdHVzVGV4dH1gXG4gICAgY29uc29sZS5lcnJvcihgRXJyb3IgaW4gJHtzaWduZWRGZXRjaEJvZHkudXJsfSBlbmRwb2ludGAsIHsgcmVzcG9uc2UgfSlcbiAgICByZXR1cm4gW2Vycm9yTWVzc2FnZSwgbnVsbCwgcmVzcG9uc2Uuc3RhdHVzXVxuICB9XG5cbiAgY29uc3QgW3BhcnNlRXJyb3IsIGJvZHldID0gYXdhaXQgdHJ5Q2F0Y2g8VD4oSlNPTi5wYXJzZShyZXNwb25zZS5ib2R5IHx8ICd7fScpKVxuXG4gIGlmIChwYXJzZUVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihgRmFpbGVkIHRvIHBhcnNlIHJlc3BvbnNlIGZyb20gJHtzaWduZWRGZXRjaEJvZHkudXJsfWApXG4gICAgcmV0dXJuIFsnRmFpbGVkIHRvIHBhcnNlIHJlc3BvbnNlJywgbnVsbCwgcmVzcG9uc2Uuc3RhdHVzXVxuICB9XG5cbiAgcmV0dXJuIFtudWxsLCAoYm9keSA/PyB7fSkgYXMgVCwgcmVzcG9uc2Uuc3RhdHVzXVxufVxuIl19
package/src/atom.ts ADDED
@@ -0,0 +1,98 @@
1
+ import future, { IFuture } from './future'
2
+
3
+ // atom value wrapper like in clojure
4
+
5
+ // Simple Observable implementation to replace Babylon.js dependency
6
+ class SimpleObservable<T> {
7
+ private observers: ((value: T) => void)[] = []
8
+ private onceObservers: ((value: T) => void)[] = []
9
+
10
+ add(callback: (value: T) => void): (value: T) => void {
11
+ this.observers.push(callback)
12
+ return callback
13
+ }
14
+
15
+ addOnce(callback: (value: T) => void): void {
16
+ this.onceObservers.push(callback)
17
+ }
18
+
19
+ remove(callback: (value: T) => void): void {
20
+ const index = this.observers.indexOf(callback)
21
+ if (index > -1) {
22
+ this.observers.splice(index, 1)
23
+ }
24
+ }
25
+
26
+ notifyObservers(value: T): void {
27
+ this.observers.forEach((observer) => observer(value))
28
+
29
+ if (this.onceObservers.length > 0) {
30
+ this.onceObservers.forEach((observer) => observer(value))
31
+ this.onceObservers.length = 0
32
+ }
33
+ }
34
+ }
35
+
36
+ const EMPTY = Symbol('empty')
37
+ type EMPTY = typeof EMPTY
38
+
39
+ export type Atom<T> = {
40
+ deref(): Promise<T>
41
+ getOrNull(): T | null
42
+ observable: SimpleObservable<T>
43
+ swap(value: T): T | void
44
+ pipe(fn: (value: T) => void | Promise<void>): Promise<void>
45
+ }
46
+
47
+ export function Atom<T>(initialValue: T | EMPTY = EMPTY): Atom<T> {
48
+ const observable = new SimpleObservable<T>()
49
+ let value: T | EMPTY = initialValue
50
+ const valueFutures: IFuture<T>[] = []
51
+
52
+ observable.addOnce((value) => {
53
+ valueFutures.forEach(($) => $.resolve(value))
54
+ valueFutures.length = 0
55
+ })
56
+
57
+ return {
58
+ async pipe(fn) {
59
+ observable.add(async (t) => {
60
+ try {
61
+ await fn(t)
62
+ } catch (err) {
63
+ console.error(err)
64
+ }
65
+ })
66
+ if (value !== EMPTY) {
67
+ try {
68
+ await fn(value)
69
+ } catch (err) {
70
+ console.error(err)
71
+ }
72
+ }
73
+ },
74
+ deref() {
75
+ if (value === EMPTY) {
76
+ const ret = future<T>()
77
+ valueFutures.push(ret)
78
+ return ret
79
+ }
80
+ return Promise.resolve(value)
81
+ },
82
+ getOrNull() {
83
+ if (value === EMPTY) {
84
+ return null
85
+ }
86
+ return value
87
+ },
88
+ observable,
89
+ swap(newValue) {
90
+ const oldValue = value
91
+ if (newValue !== value) {
92
+ value = newValue
93
+ observable.notifyObservers(value)
94
+ }
95
+ return oldValue === EMPTY ? undefined : oldValue
96
+ }
97
+ }
98
+ }
package/src/future.ts ADDED
@@ -0,0 +1,38 @@
1
+ export type IFuture<T> = Promise<T> & {
2
+ resolve: (x: T) => void
3
+ reject: (x: Error) => void
4
+ finally: (fn: () => void) => void
5
+ isPending: boolean
6
+ }
7
+
8
+ export function future<T = any>(): IFuture<T> {
9
+ let resolver: (x: T) => void
10
+ let rejecter: (x: Error) => void
11
+
12
+ const promise: any = new Promise((ok, err) => {
13
+ resolver = (x: T) => {
14
+ ok(x)
15
+ promise.isPending = false
16
+ }
17
+ rejecter = (x: Error) => {
18
+ err(x)
19
+ promise.isPending = false
20
+ }
21
+ }).catch((e) => Promise.reject(e))
22
+
23
+ promise.resolve = resolver!
24
+ promise.reject = rejecter!
25
+
26
+ if (!('finally' in promise)) {
27
+ promise.finally = (fn: any) => {
28
+ promise.then(fn)
29
+ promise.catch(fn)
30
+ }
31
+ }
32
+
33
+ promise.isPending = true
34
+
35
+ return promise as IFuture<T>
36
+ }
37
+
38
+ export default future
@@ -1,9 +1,12 @@
1
1
  import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer'
2
2
 
3
3
  export enum CommsMessage {
4
- CRDT = 1,
5
- REQ_CRDT_STATE = 2,
6
- RES_CRDT_STATE = 3
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
7
10
  }
8
11
 
9
12
  export function BinaryMessageBus<T extends CommsMessage>(
@@ -20,7 +23,9 @@ export function BinaryMessageBus<T extends CommsMessage>(
20
23
  __processMessages: (messages: Uint8Array[]) => {
21
24
  for (const message of messages) {
22
25
  const commsMsg = decodeCommsMessage<T>(message)
23
- if (!commsMsg) continue
26
+ if (!commsMsg) {
27
+ continue
28
+ }
24
29
  const { sender, messageType, data } = commsMsg
25
30
  const fn = mapping.get(messageType)
26
31
  if (fn) fn(data, sender)
@@ -0,0 +1,45 @@
1
+ import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer'
2
+ import { readMessages } from './server/utils'
3
+
4
+ /**
5
+ * Chunks CRDT messages from a Uint8Array buffer, respecting message boundaries
6
+ * Uses the comprehensive readMessages function that handles all message types
7
+ */
8
+ export function chunkCrdtMessages(data: Uint8Array, maxSizeKB: number = 12): Uint8Array[] {
9
+ if (data.length === 0) {
10
+ return []
11
+ }
12
+
13
+ const networkBuffer = new ReadWriteByteBuffer()
14
+ const chunks: Uint8Array[] = []
15
+
16
+ for (const message of readMessages(data)) {
17
+ // Check if adding this message would exceed the size limit
18
+ const currentBufferSize = networkBuffer.toBinary().byteLength
19
+ const messageSize = message.messageBuffer.byteLength
20
+
21
+ if ((currentBufferSize + messageSize) / 1024 > maxSizeKB) {
22
+ // If the current buffer has content, save it as a chunk
23
+ if (currentBufferSize > 0) {
24
+ chunks.push(networkBuffer.toCopiedBinary())
25
+ networkBuffer.resetBuffer()
26
+ }
27
+
28
+ // If the message itself is larger than the limit, skip it
29
+ if (messageSize / 1024 > maxSizeKB) {
30
+ console.error(`Message too large (${messageSize} bytes), skipping CRDT message`)
31
+ continue
32
+ }
33
+ }
34
+
35
+ // Add message to current buffer
36
+ networkBuffer.writeBuffer(message.messageBuffer, false)
37
+ }
38
+
39
+ // Add any remaining data as the final chunk
40
+ if (networkBuffer.currentWriteOffset() > 0) {
41
+ chunks.push(networkBuffer.toBinary())
42
+ }
43
+
44
+ return chunks
45
+ }