@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
|
@@ -1,66 +0,0 @@
|
|
|
1
|
-
import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer';
|
|
2
|
-
import { Schemas } from '@dcl/ecs';
|
|
3
|
-
import { EventSchemas } from './registry';
|
|
4
|
-
// Event envelope that wraps all events with metadata
|
|
5
|
-
const EventEnvelope = Schemas.Map({
|
|
6
|
-
eventType: Schemas.String,
|
|
7
|
-
timestamp: Schemas.Int64
|
|
8
|
-
});
|
|
9
|
-
/**
|
|
10
|
-
* Encode an event into a binary buffer
|
|
11
|
-
* @param eventType - The type of event from the registry
|
|
12
|
-
* @param data - The event data matching the schema
|
|
13
|
-
* @param registry - Optional custom registry (defaults to EventSchemas)
|
|
14
|
-
* @returns Binary buffer containing the encoded event
|
|
15
|
-
*/
|
|
16
|
-
export function encodeEvent(eventType, data, registry = EventSchemas) {
|
|
17
|
-
const buffer = new ReadWriteByteBuffer();
|
|
18
|
-
// Write envelope with event type and timestamp
|
|
19
|
-
EventEnvelope.serialize({
|
|
20
|
-
eventType: eventType,
|
|
21
|
-
timestamp: Date.now()
|
|
22
|
-
}, buffer);
|
|
23
|
-
// Get the schema for this event type
|
|
24
|
-
const schema = registry[eventType];
|
|
25
|
-
if (!schema) {
|
|
26
|
-
throw new Error(`Unknown event type: ${String(eventType)}`);
|
|
27
|
-
}
|
|
28
|
-
// Write the typed payload
|
|
29
|
-
schema.serialize(data, buffer);
|
|
30
|
-
return buffer.toBinary();
|
|
31
|
-
}
|
|
32
|
-
/**
|
|
33
|
-
* Decode a binary buffer into an event
|
|
34
|
-
* @param data - Binary buffer containing the encoded event
|
|
35
|
-
* @param registry - Optional custom registry (defaults to EventSchemas)
|
|
36
|
-
* @returns Decoded event with type, payload, and timestamp
|
|
37
|
-
*/
|
|
38
|
-
export function decodeEvent(data, registry = EventSchemas) {
|
|
39
|
-
const buffer = new ReadWriteByteBuffer();
|
|
40
|
-
buffer.writeBuffer(data, false);
|
|
41
|
-
// Read envelope
|
|
42
|
-
const envelope = EventEnvelope.deserialize(buffer);
|
|
43
|
-
const eventType = envelope.eventType;
|
|
44
|
-
// Get the schema for this event type
|
|
45
|
-
const schema = registry[eventType];
|
|
46
|
-
if (!schema) {
|
|
47
|
-
throw new Error(`Unknown event type: ${String(eventType)}`);
|
|
48
|
-
}
|
|
49
|
-
// Read the typed payload
|
|
50
|
-
const payload = schema.deserialize(buffer);
|
|
51
|
-
return {
|
|
52
|
-
eventType,
|
|
53
|
-
payload,
|
|
54
|
-
timestamp: envelope.timestamp
|
|
55
|
-
};
|
|
56
|
-
}
|
|
57
|
-
/**
|
|
58
|
-
* Validate if an event type exists in the registry
|
|
59
|
-
* @param eventType - The event type to check
|
|
60
|
-
* @param registry - Optional custom registry (defaults to EventSchemas)
|
|
61
|
-
* @returns True if the event type exists
|
|
62
|
-
*/
|
|
63
|
-
export function isValidEventType(eventType, registry = EventSchemas) {
|
|
64
|
-
return eventType in registry;
|
|
65
|
-
}
|
|
66
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdG9jb2wuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbmV0d29yay9ldmVudHMvcHJvdG9jb2wudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sd0NBQXdDLENBQUE7QUFDNUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLFVBQVUsQ0FBQTtBQUNsQyxPQUFPLEVBQUUsWUFBWSxFQUFtQyxNQUFNLFlBQVksQ0FBQTtBQUUxRSxxREFBcUQ7QUFDckQsTUFBTSxhQUFhLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQztJQUNoQyxTQUFTLEVBQUUsT0FBTyxDQUFDLE1BQU07SUFDekIsU0FBUyxFQUFFLE9BQU8sQ0FBQyxLQUFLO0NBQ3pCLENBQUMsQ0FBQTtBQUVGOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxXQUFXLENBQ3pCLFNBQVksRUFDWixJQUFzQixFQUN0QixXQUFjLFlBQWlCO0lBRS9CLE1BQU0sTUFBTSxHQUFHLElBQUksbUJBQW1CLEVBQUUsQ0FBQTtJQUV4QywrQ0FBK0M7SUFDL0MsYUFBYSxDQUFDLFNBQVMsQ0FDckI7UUFDRSxTQUFTLEVBQUUsU0FBbUI7UUFDOUIsU0FBUyxFQUFFLElBQUksQ0FBQyxHQUFHLEVBQUU7S0FDdEIsRUFDRCxNQUFNLENBQ1AsQ0FBQTtJQUVELHFDQUFxQztJQUNyQyxNQUFNLE1BQU0sR0FBRyxRQUFRLENBQUMsU0FBUyxDQUFDLENBQUE7SUFDbEMsSUFBSSxDQUFDLE1BQU0sRUFBRTtRQUNYLE1BQU0sSUFBSSxLQUFLLENBQUMsdUJBQXVCLE1BQU0sQ0FBQyxTQUFTLENBQUMsRUFBRSxDQUFDLENBQUE7S0FDNUQ7SUFFRCwwQkFBMEI7SUFDMUIsTUFBTSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUE7SUFFOUIsT0FBTyxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUE7QUFDMUIsQ0FBQztBQUVEOzs7OztHQUtHO0FBQ0gsTUFBTSxVQUFVLFdBQVcsQ0FDekIsSUFBZ0IsRUFDaEIsV0FBYyxZQUFpQjtJQU0vQixNQUFNLE1BQU0sR0FBRyxJQUFJLG1CQUFtQixFQUFFLENBQUE7SUFDeEMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsS0FBSyxDQUFDLENBQUE7SUFFL0IsZ0JBQWdCO0lBQ2hCLE1BQU0sUUFBUSxHQUFHLGFBQWEsQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDbEQsTUFBTSxTQUFTLEdBQUcsUUFBUSxDQUFDLFNBQW9CLENBQUE7SUFFL0MscUNBQXFDO0lBQ3JDLE1BQU0sTUFBTSxHQUFHLFFBQVEsQ0FBQyxTQUFTLENBQUMsQ0FBQTtJQUNsQyxJQUFJLENBQUMsTUFBTSxFQUFFO1FBQ1gsTUFBTSxJQUFJLEtBQUssQ0FBQyx1QkFBdUIsTUFBTSxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUMsQ0FBQTtLQUM1RDtJQUVELHlCQUF5QjtJQUN6QixNQUFNLE9BQU8sR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFBO0lBRTFDLE9BQU87UUFDTCxTQUFTO1FBQ1QsT0FBTztRQUNQLFNBQVMsRUFBRSxRQUFRLENBQUMsU0FBUztLQUM5QixDQUFBO0FBQ0gsQ0FBQztBQUVEOzs7OztHQUtHO0FBQ0gsTUFBTSxVQUFVLGdCQUFnQixDQUM5QixTQUFpQixFQUNqQixXQUFjLFlBQWlCO0lBRS9CLE9BQU8sU0FBUyxJQUFJLFFBQVEsQ0FBQTtBQUM5QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVhZFdyaXRlQnl0ZUJ1ZmZlciB9IGZyb20gJ0BkY2wvZWNzL2Rpc3Qvc2VyaWFsaXphdGlvbi9CeXRlQnVmZmVyJ1xuaW1wb3J0IHsgU2NoZW1hcyB9IGZyb20gJ0BkY2wvZWNzJ1xuaW1wb3J0IHsgRXZlbnRTY2hlbWFzLCBFdmVudFR5cGVzLCBFdmVudFNjaGVtYVJlZ2lzdHJ5IH0gZnJvbSAnLi9yZWdpc3RyeSdcblxuLy8gRXZlbnQgZW52ZWxvcGUgdGhhdCB3cmFwcyBhbGwgZXZlbnRzIHdpdGggbWV0YWRhdGFcbmNvbnN0IEV2ZW50RW52ZWxvcGUgPSBTY2hlbWFzLk1hcCh7XG4gIGV2ZW50VHlwZTogU2NoZW1hcy5TdHJpbmcsXG4gIHRpbWVzdGFtcDogU2NoZW1hcy5JbnQ2NFxufSlcblxuLyoqXG4gKiBFbmNvZGUgYW4gZXZlbnQgaW50byBhIGJpbmFyeSBidWZmZXJcbiAqIEBwYXJhbSBldmVudFR5cGUgLSBUaGUgdHlwZSBvZiBldmVudCBmcm9tIHRoZSByZWdpc3RyeVxuICogQHBhcmFtIGRhdGEgLSBUaGUgZXZlbnQgZGF0YSBtYXRjaGluZyB0aGUgc2NoZW1hXG4gKiBAcGFyYW0gcmVnaXN0cnkgLSBPcHRpb25hbCBjdXN0b20gcmVnaXN0cnkgKGRlZmF1bHRzIHRvIEV2ZW50U2NoZW1hcylcbiAqIEByZXR1cm5zIEJpbmFyeSBidWZmZXIgY29udGFpbmluZyB0aGUgZW5jb2RlZCBldmVudFxuICovXG5leHBvcnQgZnVuY3Rpb24gZW5jb2RlRXZlbnQ8VCBleHRlbmRzIEV2ZW50U2NoZW1hUmVnaXN0cnkgPSB0eXBlb2YgRXZlbnRTY2hlbWFzLCBLIGV4dGVuZHMga2V5b2YgVCA9IGtleW9mIFQ+KFxuICBldmVudFR5cGU6IEssXG4gIGRhdGE6IEV2ZW50VHlwZXM8VD5bS10sXG4gIHJlZ2lzdHJ5OiBUID0gRXZlbnRTY2hlbWFzIGFzIFRcbik6IFVpbnQ4QXJyYXkge1xuICBjb25zdCBidWZmZXIgPSBuZXcgUmVhZFdyaXRlQnl0ZUJ1ZmZlcigpXG5cbiAgLy8gV3JpdGUgZW52ZWxvcGUgd2l0aCBldmVudCB0eXBlIGFuZCB0aW1lc3RhbXBcbiAgRXZlbnRFbnZlbG9wZS5zZXJpYWxpemUoXG4gICAge1xuICAgICAgZXZlbnRUeXBlOiBldmVudFR5cGUgYXMgc3RyaW5nLFxuICAgICAgdGltZXN0YW1wOiBEYXRlLm5vdygpXG4gICAgfSxcbiAgICBidWZmZXJcbiAgKVxuXG4gIC8vIEdldCB0aGUgc2NoZW1hIGZvciB0aGlzIGV2ZW50IHR5cGVcbiAgY29uc3Qgc2NoZW1hID0gcmVnaXN0cnlbZXZlbnRUeXBlXVxuICBpZiAoIXNjaGVtYSkge1xuICAgIHRocm93IG5ldyBFcnJvcihgVW5rbm93biBldmVudCB0eXBlOiAke1N0cmluZyhldmVudFR5cGUpfWApXG4gIH1cblxuICAvLyBXcml0ZSB0aGUgdHlwZWQgcGF5bG9hZFxuICBzY2hlbWEuc2VyaWFsaXplKGRhdGEsIGJ1ZmZlcilcblxuICByZXR1cm4gYnVmZmVyLnRvQmluYXJ5KClcbn1cblxuLyoqXG4gKiBEZWNvZGUgYSBiaW5hcnkgYnVmZmVyIGludG8gYW4gZXZlbnRcbiAqIEBwYXJhbSBkYXRhIC0gQmluYXJ5IGJ1ZmZlciBjb250YWluaW5nIHRoZSBlbmNvZGVkIGV2ZW50XG4gKiBAcGFyYW0gcmVnaXN0cnkgLSBPcHRpb25hbCBjdXN0b20gcmVnaXN0cnkgKGRlZmF1bHRzIHRvIEV2ZW50U2NoZW1hcylcbiAqIEByZXR1cm5zIERlY29kZWQgZXZlbnQgd2l0aCB0eXBlLCBwYXlsb2FkLCBhbmQgdGltZXN0YW1wXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBkZWNvZGVFdmVudDxUIGV4dGVuZHMgRXZlbnRTY2hlbWFSZWdpc3RyeSA9IHR5cGVvZiBFdmVudFNjaGVtYXM+KFxuICBkYXRhOiBVaW50OEFycmF5LFxuICByZWdpc3RyeTogVCA9IEV2ZW50U2NoZW1hcyBhcyBUXG4pOiB7XG4gIGV2ZW50VHlwZToga2V5b2YgVFxuICBwYXlsb2FkOiBFdmVudFR5cGVzPFQ+W2tleW9mIFRdXG4gIHRpbWVzdGFtcDogbnVtYmVyXG59IHtcbiAgY29uc3QgYnVmZmVyID0gbmV3IFJlYWRXcml0ZUJ5dGVCdWZmZXIoKVxuICBidWZmZXIud3JpdGVCdWZmZXIoZGF0YSwgZmFsc2UpXG5cbiAgLy8gUmVhZCBlbnZlbG9wZVxuICBjb25zdCBlbnZlbG9wZSA9IEV2ZW50RW52ZWxvcGUuZGVzZXJpYWxpemUoYnVmZmVyKVxuICBjb25zdCBldmVudFR5cGUgPSBlbnZlbG9wZS5ldmVudFR5cGUgYXMga2V5b2YgVFxuXG4gIC8vIEdldCB0aGUgc2NoZW1hIGZvciB0aGlzIGV2ZW50IHR5cGVcbiAgY29uc3Qgc2NoZW1hID0gcmVnaXN0cnlbZXZlbnRUeXBlXVxuICBpZiAoIXNjaGVtYSkge1xuICAgIHRocm93IG5ldyBFcnJvcihgVW5rbm93biBldmVudCB0eXBlOiAke1N0cmluZyhldmVudFR5cGUpfWApXG4gIH1cblxuICAvLyBSZWFkIHRoZSB0eXBlZCBwYXlsb2FkXG4gIGNvbnN0IHBheWxvYWQgPSBzY2hlbWEuZGVzZXJpYWxpemUoYnVmZmVyKVxuXG4gIHJldHVybiB7XG4gICAgZXZlbnRUeXBlLFxuICAgIHBheWxvYWQsXG4gICAgdGltZXN0YW1wOiBlbnZlbG9wZS50aW1lc3RhbXBcbiAgfVxufVxuXG4vKipcbiAqIFZhbGlkYXRlIGlmIGFuIGV2ZW50IHR5cGUgZXhpc3RzIGluIHRoZSByZWdpc3RyeVxuICogQHBhcmFtIGV2ZW50VHlwZSAtIFRoZSBldmVudCB0eXBlIHRvIGNoZWNrXG4gKiBAcGFyYW0gcmVnaXN0cnkgLSBPcHRpb25hbCBjdXN0b20gcmVnaXN0cnkgKGRlZmF1bHRzIHRvIEV2ZW50U2NoZW1hcylcbiAqIEByZXR1cm5zIFRydWUgaWYgdGhlIGV2ZW50IHR5cGUgZXhpc3RzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc1ZhbGlkRXZlbnRUeXBlPFQgZXh0ZW5kcyBFdmVudFNjaGVtYVJlZ2lzdHJ5ID0gdHlwZW9mIEV2ZW50U2NoZW1hcz4oXG4gIGV2ZW50VHlwZTogc3RyaW5nLFxuICByZWdpc3RyeTogVCA9IEV2ZW50U2NoZW1hcyBhcyBUXG4pOiBldmVudFR5cGUgaXMgRXh0cmFjdDxrZXlvZiBULCBzdHJpbmc+IHtcbiAgcmV0dXJuIGV2ZW50VHlwZSBpbiByZWdpc3RyeVxufVxuIl19
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
import { ISchema } from '@dcl/ecs';
|
|
2
|
-
export type EventSchemaRegistry = Record<string, ISchema>;
|
|
3
|
-
export type EventTypes<T extends EventSchemaRegistry = EventSchemaRegistry> = {
|
|
4
|
-
[K in keyof T]: T[K] extends ISchema<infer U> ? U : never;
|
|
5
|
-
};
|
|
6
|
-
export type RegisteredEvents = EventSchemaRegistry;
|
|
7
|
-
export declare const EventSchemas: EventSchemaRegistry;
|
|
8
|
-
export type ValidateEventRegistry<T extends EventSchemaRegistry> = T;
|
|
@@ -1,3 +0,0 @@
|
|
|
1
|
-
// Default empty registry
|
|
2
|
-
export const EventSchemas = {};
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVnaXN0cnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbmV0d29yay9ldmVudHMvcmVnaXN0cnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBYUEseUJBQXlCO0FBQ3pCLE1BQU0sQ0FBQyxNQUFNLFlBQVksR0FBRyxFQUFzQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSVNjaGVtYSB9IGZyb20gJ0BkY2wvZWNzJ1xuXG4vLyBCYXNlIHR5cGUgZm9yIGV2ZW50IHNjaGVtYSByZWdpc3RyeVxuZXhwb3J0IHR5cGUgRXZlbnRTY2hlbWFSZWdpc3RyeSA9IFJlY29yZDxzdHJpbmcsIElTY2hlbWE+XG5cbi8vIFR5cGUgZXh0cmFjdGlvbiBmcm9tIHNjaGVtYXNcbmV4cG9ydCB0eXBlIEV2ZW50VHlwZXM8VCBleHRlbmRzIEV2ZW50U2NoZW1hUmVnaXN0cnkgPSBFdmVudFNjaGVtYVJlZ2lzdHJ5PiA9IHtcbiAgW0sgaW4ga2V5b2YgVF06IFRbS10gZXh0ZW5kcyBJU2NoZW1hPGluZmVyIFU+ID8gVSA6IG5ldmVyXG59XG5cbi8vIEdsb2JhbCBpbnRlcmZhY2UgdGhhdCB1c2VycyBjYW4gYXVnbWVudCB3aXRoIHRoZWlyIG93biBldmVudHNcbmV4cG9ydCB0eXBlIFJlZ2lzdGVyZWRFdmVudHMgPSBFdmVudFNjaGVtYVJlZ2lzdHJ5XG5cbi8vIERlZmF1bHQgZW1wdHkgcmVnaXN0cnlcbmV4cG9ydCBjb25zdCBFdmVudFNjaGVtYXMgPSB7fSBhcyBSZWdpc3RlcmVkRXZlbnRzXG5cbi8vIEhlbHBlciB0byBlbnN1cmUgdXNlciBldmVudHMgY29uZm9ybSB0byB0aGUgcmVnaXN0cnkgdHlwZVxuZXhwb3J0IHR5cGUgVmFsaWRhdGVFdmVudFJlZ2lzdHJ5PFQgZXh0ZW5kcyBFdmVudFNjaGVtYVJlZ2lzdHJ5PiA9IFRcbiJdfQ==
|
|
@@ -1,14 +0,0 @@
|
|
|
1
|
-
import { IEngine, Entity } from '@dcl/ecs';
|
|
2
|
-
import * as utils from './utils';
|
|
3
|
-
import { type BinaryMessageBus } from '../binary-message-bus';
|
|
4
|
-
export declare const LIVEKIT_MAX_SIZE = 12;
|
|
5
|
-
export interface ServerValidationConfig {
|
|
6
|
-
engine: IEngine;
|
|
7
|
-
binaryMessageBus: ReturnType<typeof BinaryMessageBus>;
|
|
8
|
-
}
|
|
9
|
-
export declare function createServerValidator(config: ServerValidationConfig): {
|
|
10
|
-
findExistingNetworkEntity: (message: utils.NetworkMessage) => Entity | null;
|
|
11
|
-
processClientMessages: (value: Uint8Array, sender: string, forceCorrections?: boolean) => Uint8Array;
|
|
12
|
-
processServerMessages: (value: Uint8Array, sender: string) => Uint8Array;
|
|
13
|
-
convertRegularToNetworkMessage: (regularMessage: Uint8Array) => Uint8Array[];
|
|
14
|
-
};
|
package/network/server/index.js
DELETED
|
@@ -1,219 +0,0 @@
|
|
|
1
|
-
import { CrdtMessageType, ProcessMessageResultType, PutNetworkComponentOperation } from '@dcl/ecs';
|
|
2
|
-
import * as components from '@dcl/ecs/dist/components';
|
|
3
|
-
import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer';
|
|
4
|
-
import { CommsMessage } from '../binary-message-bus';
|
|
5
|
-
import { chunkCrdtMessages } from '../chunking';
|
|
6
|
-
import * as utils from './utils';
|
|
7
|
-
import { AUTH_SERVER_PEER_ID, DEBUG_NETWORK_MESSAGES } from '../message-bus-sync';
|
|
8
|
-
export const LIVEKIT_MAX_SIZE = 12;
|
|
9
|
-
export function createServerValidator(config) {
|
|
10
|
-
const { engine, binaryMessageBus } = config;
|
|
11
|
-
// Initialize components for network operations and transform fixing
|
|
12
|
-
const NetworkEntity = components.NetworkEntity(engine);
|
|
13
|
-
const CreatedBy = components.CreatedBy(engine);
|
|
14
|
-
const NetworkParent = components.NetworkParent(engine);
|
|
15
|
-
// Type guard to check if component supports corrections (both LWW and GrowOnlySet)
|
|
16
|
-
function supportsCorrections(component) {
|
|
17
|
-
return ((component.componentType === 0 /* ComponentType.LastWriteWinElementSet */ ||
|
|
18
|
-
component.componentType === 1 /* ComponentType.GrowOnlyValueSet */) &&
|
|
19
|
-
'getCrdtState' in component);
|
|
20
|
-
}
|
|
21
|
-
function findExistingNetworkEntity(message) {
|
|
22
|
-
// Look for existing network entity mapping (don't create new ones)
|
|
23
|
-
for (const [entityId, networkData] of engine.getEntitiesWith(NetworkEntity)) {
|
|
24
|
-
if (networkData.networkId === message.networkId && networkData.entityId === message.entityId) {
|
|
25
|
-
return entityId;
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
// Return null if not found
|
|
29
|
-
return null;
|
|
30
|
-
}
|
|
31
|
-
function findOrCreateNetworkEntity(message, sender, isServer) {
|
|
32
|
-
// Look for existing network entity mapping first
|
|
33
|
-
const existingEntity = findExistingNetworkEntity(message);
|
|
34
|
-
if (existingEntity) {
|
|
35
|
-
return existingEntity;
|
|
36
|
-
}
|
|
37
|
-
// Create new entity and network mapping
|
|
38
|
-
const newEntityId = engine.addEntity();
|
|
39
|
-
NetworkEntity.createOrReplace(newEntityId, {
|
|
40
|
-
networkId: message.networkId,
|
|
41
|
-
entityId: message.entityId
|
|
42
|
-
});
|
|
43
|
-
if (isServer) {
|
|
44
|
-
CreatedBy.createOrReplace(newEntityId, { address: sender });
|
|
45
|
-
}
|
|
46
|
-
DEBUG_NETWORK_MESSAGES() &&
|
|
47
|
-
console.log(`[DEBUG] Created new entity ${newEntityId} for network ${message.networkId}:${message.entityId}`);
|
|
48
|
-
return newEntityId;
|
|
49
|
-
}
|
|
50
|
-
function convertNetworkToRegularMessage(networkMessage, localEntityId, forceCorrections = false) {
|
|
51
|
-
const buffer = new ReadWriteByteBuffer();
|
|
52
|
-
try {
|
|
53
|
-
// Use the well-tested networkMessageToLocal utility with transform fixing for Unity
|
|
54
|
-
const message = utils.networkMessageToLocal(networkMessage, localEntityId, buffer, NetworkParent, forceCorrections);
|
|
55
|
-
return { ...message, messageBuffer: buffer.toBinary() };
|
|
56
|
-
}
|
|
57
|
-
catch (error) {
|
|
58
|
-
DEBUG_NETWORK_MESSAGES() && console.error('Error converting network message:', error);
|
|
59
|
-
return null;
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
function validateMessagePermissions(message, sender, _localEntityId) {
|
|
63
|
-
// Basic checks
|
|
64
|
-
if (!sender || sender === AUTH_SERVER_PEER_ID) {
|
|
65
|
-
return false; // Server shouldn't send messages to itself
|
|
66
|
-
}
|
|
67
|
-
if (message.type === CrdtMessageType.DELETE_ENTITY) {
|
|
68
|
-
// TODO: how to handle this case ?
|
|
69
|
-
}
|
|
70
|
-
if (message.type === CrdtMessageType.PUT_COMPONENT || message.type === CrdtMessageType.DELETE_COMPONENT) {
|
|
71
|
-
const component = engine.getComponent(message.componentId);
|
|
72
|
-
const buf = 'data' in message ? new ReadWriteByteBuffer(message.data) : null;
|
|
73
|
-
const value = buf ? component.schema.deserialize(buf) : null;
|
|
74
|
-
const dryRunCRDT = component.__dry_run_updateFromCrdt(message);
|
|
75
|
-
const validCRDT = [
|
|
76
|
-
ProcessMessageResultType.StateUpdatedData,
|
|
77
|
-
ProcessMessageResultType.StateUpdatedTimestamp,
|
|
78
|
-
ProcessMessageResultType.EntityDeleted
|
|
79
|
-
].includes(dryRunCRDT);
|
|
80
|
-
const createdBy = CreatedBy.getOrNull(message.entityId);
|
|
81
|
-
const validMessage = validCRDT &&
|
|
82
|
-
component.__run_validateBeforeChange(message.entityId, value, sender, createdBy?.address ?? AUTH_SERVER_PEER_ID);
|
|
83
|
-
return !!validMessage;
|
|
84
|
-
}
|
|
85
|
-
// For now, basic validation - in the future this will check component sync permissions
|
|
86
|
-
// TODO: Check if sender owns the entity
|
|
87
|
-
// TODO: Check component sync mode ('all' | 'owner' | 'server')
|
|
88
|
-
// TODO: Run component custom validation
|
|
89
|
-
return true;
|
|
90
|
-
}
|
|
91
|
-
function broadcastBatchedMessages(messages, excludeSender) {
|
|
92
|
-
if (messages.length === 0)
|
|
93
|
-
return;
|
|
94
|
-
// Build the complete buffer with all messages
|
|
95
|
-
const networkBuffer = new ReadWriteByteBuffer();
|
|
96
|
-
for (const message of messages) {
|
|
97
|
-
// Skip oversized messages upfront
|
|
98
|
-
if (message.messageBuffer.byteLength / 1024 > LIVEKIT_MAX_SIZE) {
|
|
99
|
-
console.error(`Message too large (${message.messageBuffer.byteLength} bytes), skipping message from ${excludeSender}`);
|
|
100
|
-
continue;
|
|
101
|
-
}
|
|
102
|
-
networkBuffer.writeBuffer(message.messageBuffer, false);
|
|
103
|
-
}
|
|
104
|
-
// Use the chunking function to split into proper chunks
|
|
105
|
-
const chunks = chunkCrdtMessages(networkBuffer.toBinary(), LIVEKIT_MAX_SIZE);
|
|
106
|
-
for (const chunk of chunks) {
|
|
107
|
-
binaryMessageBus.emit(CommsMessage.CRDT, chunk);
|
|
108
|
-
}
|
|
109
|
-
DEBUG_NETWORK_MESSAGES() &&
|
|
110
|
-
console.log(`Total: ${messages.length} messages in ${chunks.length} chunks from ${excludeSender}`);
|
|
111
|
-
}
|
|
112
|
-
function sendCorrectionToSender(networkMessage, sender, localEntityId) {
|
|
113
|
-
try {
|
|
114
|
-
// Only handle component messages (PUT/DELETE), not entity deletion
|
|
115
|
-
if (networkMessage.type === CrdtMessageType.DELETE_ENTITY_NETWORK) {
|
|
116
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[AUTHORITATIVE] Cannot send authoritative message for entity deletion');
|
|
117
|
-
return;
|
|
118
|
-
}
|
|
119
|
-
// Safe to access componentId and timestamp now
|
|
120
|
-
const component = engine.getComponent(networkMessage.componentId);
|
|
121
|
-
// Only proceed if component supports authoritative messages (LWW or GrowOnlySet)
|
|
122
|
-
if (!supportsCorrections(component)) {
|
|
123
|
-
DEBUG_NETWORK_MESSAGES() && console.log('[AUTHORITATIVE] Component does not support authoritative messages');
|
|
124
|
-
return;
|
|
125
|
-
}
|
|
126
|
-
const serverCRDTState = component.getCrdtState(localEntityId);
|
|
127
|
-
if (serverCRDTState) {
|
|
128
|
-
// Create authoritative message using PUT_COMPONENT_NETWORK
|
|
129
|
-
// Each client will convert this to AUTHORITATIVE_PUT_COMPONENT with proper entity mapping
|
|
130
|
-
const correctionBuffer = new ReadWriteByteBuffer();
|
|
131
|
-
PutNetworkComponentOperation.write(networkMessage.entityId, // Use original network entity ID
|
|
132
|
-
serverCRDTState.timestamp, networkMessage.componentId, networkMessage.networkId, serverCRDTState.data, correctionBuffer);
|
|
133
|
-
// Send authoritative message directly to the sender
|
|
134
|
-
binaryMessageBus.emit(CommsMessage.CRDT_AUTHORITATIVE, correctionBuffer.toBinary(), [sender]);
|
|
135
|
-
DEBUG_NETWORK_MESSAGES() &&
|
|
136
|
-
console.log(`[AUTHORITATIVE] Sent authoritative message to ${sender} for entity ${localEntityId} component ${networkMessage.componentId} with timestamp ${networkMessage.timestamp}`);
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
catch (error) {
|
|
140
|
-
DEBUG_NETWORK_MESSAGES() && console.error('Error sending correction:', error);
|
|
141
|
-
}
|
|
142
|
-
}
|
|
143
|
-
return {
|
|
144
|
-
findExistingNetworkEntity,
|
|
145
|
-
// transform Network messages to CRDT Common Messages.
|
|
146
|
-
processClientMessages: function processClientMessages(value, sender, forceCorrections = false) {
|
|
147
|
-
// console.log(`[CLIENT] Processing message from ${sender}, ${value.length} bytes`)
|
|
148
|
-
// Collect all regular messages in a single buffer for batched application
|
|
149
|
-
const combinedBuffer = new ReadWriteByteBuffer();
|
|
150
|
-
// Clients process network messages from server and convert them to regular messages
|
|
151
|
-
for (const message of utils.readMessages(value)) {
|
|
152
|
-
// Only process network messages in client message handler
|
|
153
|
-
if (utils.isNetworkMessage(message)) {
|
|
154
|
-
const networkMessage = message;
|
|
155
|
-
// Find or create network entity mapping
|
|
156
|
-
const localEntityId = findOrCreateNetworkEntity(networkMessage, sender, false);
|
|
157
|
-
// Convert network message to regular message or correction message
|
|
158
|
-
const regularMessage = convertNetworkToRegularMessage(networkMessage, localEntityId, forceCorrections);
|
|
159
|
-
if (regularMessage?.messageBuffer.byteLength) {
|
|
160
|
-
combinedBuffer.writeBuffer(regularMessage.messageBuffer, false);
|
|
161
|
-
}
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
return combinedBuffer.toBinary();
|
|
165
|
-
},
|
|
166
|
-
// Sever Code: process message, handle permissions, and broadcast if needed.
|
|
167
|
-
processServerMessages: function processServerMessages(value, sender) {
|
|
168
|
-
// console.log(`[SERVER] Processing message from ${sender}, ${value.length} bytes`)
|
|
169
|
-
// Collect all valid messages for batched broadcasting
|
|
170
|
-
const messagesToBroadcast = [];
|
|
171
|
-
const regularMessagesBuffer = new ReadWriteByteBuffer();
|
|
172
|
-
for (const message of utils.readMessages(value)) {
|
|
173
|
-
try {
|
|
174
|
-
// Only process network messages in server message handler
|
|
175
|
-
if (utils.isNetworkMessage(message)) {
|
|
176
|
-
const networkMessage = message;
|
|
177
|
-
// 1. Find or create network entity mapping
|
|
178
|
-
const localEntityId = findOrCreateNetworkEntity(networkMessage, sender, true);
|
|
179
|
-
// 2. Convert network message to regular message and collect for local application
|
|
180
|
-
const regularMessage = convertNetworkToRegularMessage(networkMessage, localEntityId);
|
|
181
|
-
// 3. Basic permission validation
|
|
182
|
-
if (!validateMessagePermissions(regularMessage, sender, localEntityId)) {
|
|
183
|
-
// Send correction back to sender with server's authoritative state
|
|
184
|
-
sendCorrectionToSender(networkMessage, sender, localEntityId);
|
|
185
|
-
continue;
|
|
186
|
-
}
|
|
187
|
-
// 4. Collect valid message for batched broadcasting
|
|
188
|
-
messagesToBroadcast.push(networkMessage);
|
|
189
|
-
if (regularMessage?.messageBuffer.byteLength) {
|
|
190
|
-
regularMessagesBuffer.writeBuffer(regularMessage.messageBuffer, false);
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
catch (error) {
|
|
195
|
-
DEBUG_NETWORK_MESSAGES() && console.error('Error processing server message:', error);
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
// Batch broadcast all valid messages together
|
|
199
|
-
broadcastBatchedMessages(messagesToBroadcast, sender);
|
|
200
|
-
return regularMessagesBuffer.toBinary();
|
|
201
|
-
},
|
|
202
|
-
// engine changes that needs to be broadcasted.
|
|
203
|
-
convertRegularToNetworkMessage: function convertRegularToNetworkMessage(regularMessage) {
|
|
204
|
-
const groupedBuffer = new ReadWriteByteBuffer();
|
|
205
|
-
// First pass: Convert all regular messages to network format and group them into one big buffer
|
|
206
|
-
for (const message of utils.readMessages(regularMessage)) {
|
|
207
|
-
// Only convert regular messages that have network data
|
|
208
|
-
const networkData = NetworkEntity.getOrNull(message.entityId);
|
|
209
|
-
if (networkData && !utils.isNetworkMessage(message)) {
|
|
210
|
-
utils.localMessageToNetwork(message, networkData, groupedBuffer);
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
// Second pass: Use the new chunking function that respects message boundaries
|
|
214
|
-
const totalData = groupedBuffer.toBinary();
|
|
215
|
-
return chunkCrdtMessages(totalData, LIVEKIT_MAX_SIZE);
|
|
216
|
-
}
|
|
217
|
-
};
|
|
218
|
-
}
|
|
219
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/network/server/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  IEngine,\n  Entity,\n  CrdtMessageType,\n  CrdtMessageBody,\n  ProcessMessageResultType,\n  ComponentType,\n  PutNetworkComponentOperation\n} from '@dcl/ecs'\nimport * as components from '@dcl/ecs/dist/components'\nimport { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer'\nimport { CommsMessage } from '../binary-message-bus'\nimport { chunkCrdtMessages } from '../chunking'\nimport * as utils from './utils'\nimport { AUTH_SERVER_PEER_ID, DEBUG_NETWORK_MESSAGES } from '../message-bus-sync'\nimport { type BinaryMessageBus } from '../binary-message-bus'\nimport {\n  LastWriteWinElementSetComponentDefinition,\n  GrowOnlyValueSetComponentDefinition,\n  ComponentDefinition,\n  InternalBaseComponent\n} from '@dcl/ecs/dist/engine/component'\n\nexport const LIVEKIT_MAX_SIZE = 12\n\nexport interface ServerValidationConfig {\n  engine: IEngine\n  binaryMessageBus: ReturnType<typeof BinaryMessageBus>\n}\n\nexport function createServerValidator(config: ServerValidationConfig) {\n  const { engine, binaryMessageBus } = config\n\n  // Initialize components for network operations and transform fixing\n  const NetworkEntity = components.NetworkEntity(engine)\n  const CreatedBy = components.CreatedBy(engine)\n  const NetworkParent = components.NetworkParent(engine)\n\n  // Type guard to check if component supports corrections (both LWW and GrowOnlySet)\n  function supportsCorrections<T>(\n    component: ComponentDefinition<T>\n  ): component is LastWriteWinElementSetComponentDefinition<T> | GrowOnlyValueSetComponentDefinition<T> {\n    return (\n      (component.componentType === ComponentType.LastWriteWinElementSet ||\n        component.componentType === ComponentType.GrowOnlyValueSet) &&\n      'getCrdtState' in component\n    )\n  }\n\n  function findExistingNetworkEntity(message: utils.NetworkMessage): Entity | null {\n    // Look for existing network entity mapping (don't create new ones)\n    for (const [entityId, networkData] of engine.getEntitiesWith(NetworkEntity)) {\n      if (networkData.networkId === message.networkId && networkData.entityId === message.entityId) {\n        return entityId\n      }\n    }\n    // Return null if not found\n    return null\n  }\n\n  function findOrCreateNetworkEntity(message: utils.NetworkMessage, sender: string, isServer: boolean): Entity {\n    // Look for existing network entity mapping first\n    const existingEntity = findExistingNetworkEntity(message)\n\n    if (existingEntity) {\n      return existingEntity\n    }\n\n    // Create new entity and network mapping\n    const newEntityId = engine.addEntity()\n    NetworkEntity.createOrReplace(newEntityId, {\n      networkId: message.networkId,\n      entityId: message.entityId\n    })\n\n    if (isServer) {\n      CreatedBy.createOrReplace(newEntityId, { address: sender })\n    }\n\n    DEBUG_NETWORK_MESSAGES() &&\n      console.log(`[DEBUG] Created new entity ${newEntityId} for network ${message.networkId}:${message.entityId}`)\n    return newEntityId\n  }\n\n  function convertNetworkToRegularMessage(\n    networkMessage: utils.NetworkMessage,\n    localEntityId: Entity,\n    forceCorrections = false\n  ): (CrdtMessageBody & { messageBuffer: Uint8Array }) | null {\n    const buffer = new ReadWriteByteBuffer()\n\n    try {\n      // Use the well-tested networkMessageToLocal utility with transform fixing for Unity\n      const message = utils.networkMessageToLocal(\n        networkMessage,\n        localEntityId,\n        buffer,\n        NetworkParent,\n        forceCorrections\n      )\n      return { ...message, messageBuffer: buffer.toBinary() }\n    } catch (error) {\n      DEBUG_NETWORK_MESSAGES() && console.error('Error converting network message:', error)\n      return null\n    }\n  }\n\n  function validateMessagePermissions(message: utils.RegularMessage, sender: string, _localEntityId: Entity): boolean {\n    // Basic checks\n    if (!sender || sender === AUTH_SERVER_PEER_ID) {\n      return false // Server shouldn't send messages to itself\n    }\n\n    if (message.type === CrdtMessageType.DELETE_ENTITY) {\n      // TODO: how to handle this case ?\n    }\n\n    if (message.type === CrdtMessageType.PUT_COMPONENT || message.type === CrdtMessageType.DELETE_COMPONENT) {\n      const component = engine.getComponent(message.componentId) as InternalBaseComponent<unknown>\n      const buf = 'data' in message ? new ReadWriteByteBuffer(message.data) : null\n      const value = buf ? component.schema.deserialize(buf) : null\n      const dryRunCRDT = component.__dry_run_updateFromCrdt(message)\n      const validCRDT = [\n        ProcessMessageResultType.StateUpdatedData,\n        ProcessMessageResultType.StateUpdatedTimestamp,\n        ProcessMessageResultType.EntityDeleted\n      ].includes(dryRunCRDT)\n      const createdBy = CreatedBy.getOrNull(message.entityId)\n      const validMessage =\n        validCRDT &&\n        component.__run_validateBeforeChange(message.entityId, value, sender, createdBy?.address ?? AUTH_SERVER_PEER_ID)\n\n      return !!validMessage\n    }\n\n    // For now, basic validation - in the future this will check component sync permissions\n    // TODO: Check if sender owns the entity\n    // TODO: Check component sync mode ('all' | 'owner' | 'server')\n    // TODO: Run component custom validation\n    return true\n  }\n\n  function broadcastBatchedMessages(messages: utils.NetworkMessage[], excludeSender: string) {\n    if (messages.length === 0) return\n\n    // Build the complete buffer with all messages\n    const networkBuffer = new ReadWriteByteBuffer()\n    for (const message of messages) {\n      // Skip oversized messages upfront\n      if (message.messageBuffer.byteLength / 1024 > LIVEKIT_MAX_SIZE) {\n        console.error(\n          `Message too large (${message.messageBuffer.byteLength} bytes), skipping message from ${excludeSender}`\n        )\n        continue\n      }\n      networkBuffer.writeBuffer(message.messageBuffer, false)\n    }\n\n    // Use the chunking function to split into proper chunks\n    const chunks = chunkCrdtMessages(networkBuffer.toBinary(), LIVEKIT_MAX_SIZE)\n\n    for (const chunk of chunks) {\n      binaryMessageBus.emit(CommsMessage.CRDT, chunk)\n    }\n    DEBUG_NETWORK_MESSAGES() &&\n      console.log(`Total: ${messages.length} messages in ${chunks.length} chunks from ${excludeSender}`)\n  }\n\n  function sendCorrectionToSender(networkMessage: utils.NetworkMessage, sender: string, localEntityId: Entity) {\n    try {\n      // Only handle component messages (PUT/DELETE), not entity deletion\n      if (networkMessage.type === CrdtMessageType.DELETE_ENTITY_NETWORK) {\n        DEBUG_NETWORK_MESSAGES() && console.log('[AUTHORITATIVE] Cannot send authoritative message for entity deletion')\n        return\n      }\n\n      // Safe to access componentId and timestamp now\n      const component = engine.getComponent(networkMessage.componentId)\n\n      // Only proceed if component supports authoritative messages (LWW or GrowOnlySet)\n      if (!supportsCorrections(component)) {\n        DEBUG_NETWORK_MESSAGES() && console.log('[AUTHORITATIVE] Component does not support authoritative messages')\n        return\n      }\n\n      const serverCRDTState = component.getCrdtState(localEntityId)\n\n      if (serverCRDTState) {\n        // Create authoritative message using PUT_COMPONENT_NETWORK\n        // Each client will convert this to AUTHORITATIVE_PUT_COMPONENT with proper entity mapping\n        const correctionBuffer = new ReadWriteByteBuffer()\n        PutNetworkComponentOperation.write(\n          networkMessage.entityId, // Use original network entity ID\n          serverCRDTState.timestamp,\n          networkMessage.componentId,\n          networkMessage.networkId,\n          serverCRDTState.data,\n          correctionBuffer\n        )\n        // Send authoritative message directly to the sender\n        binaryMessageBus.emit(CommsMessage.CRDT_AUTHORITATIVE, correctionBuffer.toBinary(), [sender])\n\n        DEBUG_NETWORK_MESSAGES() &&\n          console.log(\n            `[AUTHORITATIVE] Sent authoritative message to ${sender} for entity ${localEntityId} component ${networkMessage.componentId} with timestamp ${networkMessage.timestamp}`\n          )\n      }\n    } catch (error) {\n      DEBUG_NETWORK_MESSAGES() && console.error('Error sending correction:', error)\n    }\n  }\n\n  return {\n    findExistingNetworkEntity,\n    // transform Network messages to CRDT Common Messages.\n    processClientMessages: function processClientMessages(value: Uint8Array, sender: string, forceCorrections = false) {\n      // console.log(`[CLIENT] Processing message from ${sender}, ${value.length} bytes`)\n\n      // Collect all regular messages in a single buffer for batched application\n      const combinedBuffer = new ReadWriteByteBuffer()\n\n      // Clients process network messages from server and convert them to regular messages\n      for (const message of utils.readMessages(value)) {\n        // Only process network messages in client message handler\n        if (utils.isNetworkMessage(message)) {\n          const networkMessage = message as utils.NetworkMessage\n\n          // Find or create network entity mapping\n          const localEntityId = findOrCreateNetworkEntity(networkMessage, sender, false)\n\n          // Convert network message to regular message or correction message\n          const regularMessage = convertNetworkToRegularMessage(networkMessage, localEntityId, forceCorrections)\n\n          if (regularMessage?.messageBuffer.byteLength) {\n            combinedBuffer.writeBuffer(regularMessage.messageBuffer, false)\n          }\n        }\n      }\n      return combinedBuffer.toBinary()\n    },\n    // Sever Code: process message, handle permissions, and broadcast if needed.\n    processServerMessages: function processServerMessages(value: Uint8Array, sender: string) {\n      // console.log(`[SERVER] Processing message from ${sender}, ${value.length} bytes`)\n\n      // Collect all valid messages for batched broadcasting\n      const messagesToBroadcast: utils.NetworkMessage[] = []\n      const regularMessagesBuffer = new ReadWriteByteBuffer()\n\n      for (const message of utils.readMessages(value)) {\n        try {\n          // Only process network messages in server message handler\n          if (utils.isNetworkMessage(message)) {\n            const networkMessage = message as utils.NetworkMessage\n            // 1. Find or create network entity mapping\n            const localEntityId = findOrCreateNetworkEntity(networkMessage, sender, true)\n\n            // 2. Convert network message to regular message and collect for local application\n            const regularMessage = convertNetworkToRegularMessage(networkMessage, localEntityId)\n\n            // 3. Basic permission validation\n            if (!validateMessagePermissions(regularMessage as any, sender, localEntityId)) {\n              // Send correction back to sender with server's authoritative state\n              sendCorrectionToSender(networkMessage, sender, localEntityId)\n              continue\n            }\n\n            // 4. Collect valid message for batched broadcasting\n            messagesToBroadcast.push(networkMessage)\n\n            if (regularMessage?.messageBuffer.byteLength) {\n              regularMessagesBuffer.writeBuffer(regularMessage.messageBuffer, false)\n            }\n          }\n        } catch (error) {\n          DEBUG_NETWORK_MESSAGES() && console.error('Error processing server message:', error)\n        }\n      }\n      // Batch broadcast all valid messages together\n      broadcastBatchedMessages(messagesToBroadcast, sender)\n      return regularMessagesBuffer.toBinary()\n    },\n    // engine changes that needs to be broadcasted.\n    convertRegularToNetworkMessage: function convertRegularToNetworkMessage(regularMessage: Uint8Array): Uint8Array[] {\n      const groupedBuffer = new ReadWriteByteBuffer()\n\n      // First pass: Convert all regular messages to network format and group them into one big buffer\n      for (const message of utils.readMessages(regularMessage)) {\n        // Only convert regular messages that have network data\n        const networkData = NetworkEntity.getOrNull(message.entityId)\n\n        if (networkData && !utils.isNetworkMessage(message)) {\n          utils.localMessageToNetwork(message, networkData, groupedBuffer)\n        }\n      }\n\n      // Second pass: Use the new chunking function that respects message boundaries\n      const totalData = groupedBuffer.toBinary()\n      return chunkCrdtMessages(totalData, LIVEKIT_MAX_SIZE)\n    }\n  }\n}\n"]}
|
|
@@ -1,18 +0,0 @@
|
|
|
1
|
-
import { Entity } from '@dcl/ecs/dist/engine';
|
|
2
|
-
import { NetworkParent } from '@dcl/ecs';
|
|
3
|
-
import { ReceiveMessage } from '@dcl/ecs/dist/runtime/types';
|
|
4
|
-
import { ReceiveNetworkMessage } from '@dcl/ecs/dist/systems/crdt/types';
|
|
5
|
-
import { ByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer';
|
|
6
|
-
import { CrdtMessageBody, DeleteComponentMessage, DeleteComponentNetworkMessage, DeleteEntityMessage, DeleteEntityNetworkMessage, PutComponentMessage, AuthoritativePutComponentMessage, PutNetworkComponentMessage } from '@dcl/ecs/dist/serialization/crdt/types';
|
|
7
|
-
import { INetowrkEntityType } from '@dcl/ecs/dist/components/types';
|
|
8
|
-
export type NetworkMessage = (PutNetworkComponentMessage | DeleteComponentNetworkMessage | DeleteEntityNetworkMessage) & {
|
|
9
|
-
messageBuffer: Uint8Array;
|
|
10
|
-
};
|
|
11
|
-
export type RegularMessage = (PutComponentMessage | AuthoritativePutComponentMessage | DeleteComponentMessage | DeleteEntityMessage) & {
|
|
12
|
-
messageBuffer: Uint8Array;
|
|
13
|
-
};
|
|
14
|
-
export declare function readMessages(data: Uint8Array): (NetworkMessage | RegularMessage)[];
|
|
15
|
-
export declare function isNetworkMessage(message: ReceiveMessage): message is ReceiveNetworkMessage;
|
|
16
|
-
export declare function networkMessageToLocal(message: ReceiveNetworkMessage, localEntityId: Entity, destinationBuffer: ByteBuffer, networkParentComponent?: typeof NetworkParent, forceCorrections?: boolean): CrdtMessageBody;
|
|
17
|
-
export declare function localMessageToNetwork(message: ReceiveMessage, network: INetowrkEntityType, destinationBuffer: ByteBuffer): void;
|
|
18
|
-
export declare function fixTransformParent(message: ReceiveMessage, parent?: Entity): Uint8Array;
|
package/network/server/utils.js
DELETED
|
@@ -1,135 +0,0 @@
|
|
|
1
|
-
import { CrdtMessageProtocol } from '@dcl/ecs';
|
|
2
|
-
import { ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer';
|
|
3
|
-
import { AuthoritativePutComponentOperation, PutComponentOperation } from '@dcl/ecs/dist/serialization/crdt';
|
|
4
|
-
import { CrdtMessageType } from '@dcl/ecs/dist/serialization/crdt/types';
|
|
5
|
-
import { DeleteComponent } from '@dcl/ecs/dist/serialization/crdt/deleteComponent';
|
|
6
|
-
import { DeleteEntity } from '@dcl/ecs/dist/serialization/crdt/deleteEntity';
|
|
7
|
-
import { PutNetworkComponentOperation } from '@dcl/ecs/dist/serialization/crdt/network/putComponentNetwork';
|
|
8
|
-
import { DeleteComponentNetwork } from '@dcl/ecs/dist/serialization/crdt/network/deleteComponentNetwork';
|
|
9
|
-
import { DeleteEntityNetwork } from '@dcl/ecs/dist/serialization/crdt/network/deleteEntityNetwork';
|
|
10
|
-
import { TransformSchema, COMPONENT_ID as TransformComponentId } from '@dcl/ecs/dist/components/manual/Transform';
|
|
11
|
-
export function readMessages(data) {
|
|
12
|
-
const buffer = new ReadWriteByteBuffer(data);
|
|
13
|
-
const messages = [];
|
|
14
|
-
let header;
|
|
15
|
-
while ((header = CrdtMessageProtocol.getHeader(buffer))) {
|
|
16
|
-
const offset = buffer.currentReadOffset();
|
|
17
|
-
let message = undefined;
|
|
18
|
-
// Network messages
|
|
19
|
-
if (header.type === CrdtMessageType.DELETE_COMPONENT_NETWORK) {
|
|
20
|
-
message = DeleteComponentNetwork.read(buffer);
|
|
21
|
-
}
|
|
22
|
-
else if (header.type === CrdtMessageType.PUT_COMPONENT_NETWORK) {
|
|
23
|
-
message = PutNetworkComponentOperation.read(buffer);
|
|
24
|
-
}
|
|
25
|
-
else if (header.type === CrdtMessageType.DELETE_ENTITY_NETWORK) {
|
|
26
|
-
message = DeleteEntityNetwork.read(buffer);
|
|
27
|
-
}
|
|
28
|
-
// Regular messages
|
|
29
|
-
else if (header.type === CrdtMessageType.PUT_COMPONENT) {
|
|
30
|
-
message = PutComponentOperation.read(buffer);
|
|
31
|
-
}
|
|
32
|
-
else if (header.type === CrdtMessageType.AUTHORITATIVE_PUT_COMPONENT) {
|
|
33
|
-
message = AuthoritativePutComponentOperation.read(buffer);
|
|
34
|
-
}
|
|
35
|
-
else if (header.type === CrdtMessageType.DELETE_COMPONENT) {
|
|
36
|
-
message = DeleteComponent.read(buffer);
|
|
37
|
-
}
|
|
38
|
-
else if (header.type === CrdtMessageType.DELETE_ENTITY) {
|
|
39
|
-
message = DeleteEntity.read(buffer);
|
|
40
|
-
}
|
|
41
|
-
else {
|
|
42
|
-
// consume unknown messages
|
|
43
|
-
buffer.incrementReadOffset(header.length);
|
|
44
|
-
}
|
|
45
|
-
if (message) {
|
|
46
|
-
messages.push({
|
|
47
|
-
...message,
|
|
48
|
-
messageBuffer: buffer.buffer().subarray(offset, buffer.currentReadOffset())
|
|
49
|
-
});
|
|
50
|
-
}
|
|
51
|
-
}
|
|
52
|
-
return messages;
|
|
53
|
-
}
|
|
54
|
-
export function isNetworkMessage(message) {
|
|
55
|
-
return [
|
|
56
|
-
CrdtMessageType.DELETE_COMPONENT_NETWORK,
|
|
57
|
-
CrdtMessageType.DELETE_ENTITY_NETWORK,
|
|
58
|
-
CrdtMessageType.PUT_COMPONENT_NETWORK
|
|
59
|
-
].includes(message.type);
|
|
60
|
-
}
|
|
61
|
-
export function networkMessageToLocal(message, localEntityId, destinationBuffer,
|
|
62
|
-
// Optional network parent component for transform fixing
|
|
63
|
-
networkParentComponent,
|
|
64
|
-
// Force corrections - converts PUT_COMPONENT_NETWORK to authoritative_PUT_COMPONENT
|
|
65
|
-
forceCorrections = false) {
|
|
66
|
-
if (message.type === CrdtMessageType.PUT_COMPONENT_NETWORK) {
|
|
67
|
-
let messageData = message.data;
|
|
68
|
-
// Fix transform parent if needed for Unity/engine processing
|
|
69
|
-
if (message.componentId === TransformComponentId && networkParentComponent) {
|
|
70
|
-
const parentNetwork = networkParentComponent.getOrNull(localEntityId);
|
|
71
|
-
messageData = fixTransformParent(message, parentNetwork?.entityId);
|
|
72
|
-
}
|
|
73
|
-
if (forceCorrections) {
|
|
74
|
-
// Use AUTHORITATIVE_PUT_COMPONENT for forced state updates
|
|
75
|
-
AuthoritativePutComponentOperation.write(localEntityId, message.timestamp, message.componentId, messageData, destinationBuffer);
|
|
76
|
-
return {
|
|
77
|
-
type: CrdtMessageType.AUTHORITATIVE_PUT_COMPONENT,
|
|
78
|
-
componentId: message.componentId,
|
|
79
|
-
timestamp: message.timestamp,
|
|
80
|
-
data: messageData,
|
|
81
|
-
entityId: localEntityId
|
|
82
|
-
};
|
|
83
|
-
}
|
|
84
|
-
else {
|
|
85
|
-
// Normal PUT_COMPONENT conversion
|
|
86
|
-
PutComponentOperation.write(localEntityId, message.timestamp, message.componentId, messageData, destinationBuffer);
|
|
87
|
-
return {
|
|
88
|
-
type: CrdtMessageType.PUT_COMPONENT,
|
|
89
|
-
componentId: message.componentId,
|
|
90
|
-
timestamp: message.timestamp,
|
|
91
|
-
data: messageData,
|
|
92
|
-
entityId: localEntityId
|
|
93
|
-
};
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
else if (message.type === CrdtMessageType.DELETE_COMPONENT_NETWORK) {
|
|
97
|
-
DeleteComponent.write(localEntityId, message.componentId, message.timestamp, destinationBuffer);
|
|
98
|
-
return {
|
|
99
|
-
type: CrdtMessageType.DELETE_COMPONENT,
|
|
100
|
-
componentId: message.componentId,
|
|
101
|
-
timestamp: message.timestamp,
|
|
102
|
-
entityId: localEntityId
|
|
103
|
-
};
|
|
104
|
-
}
|
|
105
|
-
else if (message.type === CrdtMessageType.DELETE_ENTITY_NETWORK) {
|
|
106
|
-
DeleteEntity.write(localEntityId, destinationBuffer);
|
|
107
|
-
return {
|
|
108
|
-
type: CrdtMessageType.DELETE_ENTITY,
|
|
109
|
-
entityId: localEntityId
|
|
110
|
-
};
|
|
111
|
-
}
|
|
112
|
-
throw 1;
|
|
113
|
-
}
|
|
114
|
-
export function localMessageToNetwork(message, network, destinationBuffer) {
|
|
115
|
-
if (message.type === CrdtMessageType.PUT_COMPONENT) {
|
|
116
|
-
PutNetworkComponentOperation.write(network.entityId, message.timestamp, message.componentId, network.networkId, message.data, destinationBuffer);
|
|
117
|
-
}
|
|
118
|
-
else if (message.type === CrdtMessageType.DELETE_COMPONENT) {
|
|
119
|
-
DeleteComponentNetwork.write(network.entityId, message.componentId, message.timestamp, network.networkId, destinationBuffer);
|
|
120
|
-
}
|
|
121
|
-
else if (message.type === CrdtMessageType.DELETE_ENTITY) {
|
|
122
|
-
DeleteEntityNetwork.write(network.entityId, network.networkId, destinationBuffer);
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
export function fixTransformParent(message, parent) {
|
|
126
|
-
const buffer = new ReadWriteByteBuffer();
|
|
127
|
-
const transform = 'data' in message && TransformSchema.deserialize(new ReadWriteByteBuffer(message.data));
|
|
128
|
-
if (!transform)
|
|
129
|
-
throw new Error('Invalid parent transform');
|
|
130
|
-
// Generate new transform raw data with the parent
|
|
131
|
-
const newTransform = { ...transform, parent };
|
|
132
|
-
TransformSchema.serialize(newTransform, buffer);
|
|
133
|
-
return buffer.toBinary();
|
|
134
|
-
}
|
|
135
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/network/server/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { Entity } from '@dcl/ecs/dist/engine'\nimport { CrdtMessageProtocol, NetworkParent } from '@dcl/ecs'\nimport { ReceiveMessage } from '@dcl/ecs/dist/runtime/types'\nimport { ReceiveNetworkMessage } from '@dcl/ecs/dist/systems/crdt/types'\nimport { ByteBuffer, ReadWriteByteBuffer } from '@dcl/ecs/dist/serialization/ByteBuffer'\nimport { AuthoritativePutComponentOperation, PutComponentOperation } from '@dcl/ecs/dist/serialization/crdt'\nimport {\n  CrdtMessage,\n  CrdtMessageBody,\n  CrdtMessageHeader,\n  CrdtMessageType,\n  DeleteComponentMessage,\n  DeleteComponentNetworkMessage,\n  DeleteEntityMessage,\n  DeleteEntityNetworkMessage,\n  PutComponentMessage,\n  AuthoritativePutComponentMessage,\n  PutNetworkComponentMessage\n} from '@dcl/ecs/dist/serialization/crdt/types'\nimport { DeleteComponent } from '@dcl/ecs/dist/serialization/crdt/deleteComponent'\nimport { DeleteEntity } from '@dcl/ecs/dist/serialization/crdt/deleteEntity'\nimport { INetowrkEntityType } from '@dcl/ecs/dist/components/types'\nimport { PutNetworkComponentOperation } from '@dcl/ecs/dist/serialization/crdt/network/putComponentNetwork'\nimport { DeleteComponentNetwork } from '@dcl/ecs/dist/serialization/crdt/network/deleteComponentNetwork'\nimport { DeleteEntityNetwork } from '@dcl/ecs/dist/serialization/crdt/network/deleteEntityNetwork'\nimport { TransformSchema, COMPONENT_ID as TransformComponentId } from '@dcl/ecs/dist/components/manual/Transform'\n\nexport type NetworkMessage = (\n  | PutNetworkComponentMessage\n  | DeleteComponentNetworkMessage\n  | DeleteEntityNetworkMessage\n) & { messageBuffer: Uint8Array }\n\nexport type RegularMessage = (\n  | PutComponentMessage\n  | AuthoritativePutComponentMessage\n  | DeleteComponentMessage\n  | DeleteEntityMessage\n) & {\n  messageBuffer: Uint8Array\n}\nexport function readMessages(data: Uint8Array): (NetworkMessage | RegularMessage)[] {\n  const buffer = new ReadWriteByteBuffer(data)\n  const messages: (NetworkMessage | RegularMessage)[] = []\n  let header: CrdtMessageHeader | null\n  while ((header = CrdtMessageProtocol.getHeader(buffer))) {\n    const offset = buffer.currentReadOffset()\n    let message: CrdtMessage | undefined = undefined\n\n    // Network messages\n    if (header.type === CrdtMessageType.DELETE_COMPONENT_NETWORK) {\n      message = DeleteComponentNetwork.read(buffer)!\n    } else if (header.type === CrdtMessageType.PUT_COMPONENT_NETWORK) {\n      message = PutNetworkComponentOperation.read(buffer)!\n    } else if (header.type === CrdtMessageType.DELETE_ENTITY_NETWORK) {\n      message = DeleteEntityNetwork.read(buffer)!\n    }\n    // Regular messages\n    else if (header.type === CrdtMessageType.PUT_COMPONENT) {\n      message = PutComponentOperation.read(buffer)!\n    } else if (header.type === CrdtMessageType.AUTHORITATIVE_PUT_COMPONENT) {\n      message = AuthoritativePutComponentOperation.read(buffer)!\n    } else if (header.type === CrdtMessageType.DELETE_COMPONENT) {\n      message = DeleteComponent.read(buffer)!\n    } else if (header.type === CrdtMessageType.DELETE_ENTITY) {\n      message = DeleteEntity.read(buffer)!\n    } else {\n      // consume unknown messages\n      buffer.incrementReadOffset(header.length)\n    }\n\n    if (message) {\n      messages.push({\n        ...message,\n        messageBuffer: buffer.buffer().subarray(offset, buffer.currentReadOffset())\n      })\n    }\n  }\n  return messages\n}\n\nexport function isNetworkMessage(message: ReceiveMessage): message is ReceiveNetworkMessage {\n  return [\n    CrdtMessageType.DELETE_COMPONENT_NETWORK,\n    CrdtMessageType.DELETE_ENTITY_NETWORK,\n    CrdtMessageType.PUT_COMPONENT_NETWORK\n  ].includes(message.type)\n}\n\nexport function networkMessageToLocal(\n  message: ReceiveNetworkMessage,\n  localEntityId: Entity,\n  destinationBuffer: ByteBuffer,\n  // Optional network parent component for transform fixing\n  networkParentComponent?: typeof NetworkParent,\n  // Force corrections - converts PUT_COMPONENT_NETWORK to authoritative_PUT_COMPONENT\n  forceCorrections = false\n): CrdtMessageBody {\n  if (message.type === CrdtMessageType.PUT_COMPONENT_NETWORK) {\n    let messageData = message.data\n\n    // Fix transform parent if needed for Unity/engine processing\n    if (message.componentId === TransformComponentId && networkParentComponent) {\n      const parentNetwork = networkParentComponent.getOrNull(localEntityId)\n      messageData = fixTransformParent(message, parentNetwork?.entityId)\n    }\n    if (forceCorrections) {\n      // Use AUTHORITATIVE_PUT_COMPONENT for forced state updates\n      AuthoritativePutComponentOperation.write(\n        localEntityId,\n        message.timestamp,\n        message.componentId,\n        messageData,\n        destinationBuffer\n      )\n      return {\n        type: CrdtMessageType.AUTHORITATIVE_PUT_COMPONENT,\n        componentId: message.componentId,\n        timestamp: message.timestamp,\n        data: messageData,\n        entityId: localEntityId\n      }\n    } else {\n      // Normal PUT_COMPONENT conversion\n      PutComponentOperation.write(localEntityId, message.timestamp, message.componentId, messageData, destinationBuffer)\n      return {\n        type: CrdtMessageType.PUT_COMPONENT,\n        componentId: message.componentId,\n        timestamp: message.timestamp,\n        data: messageData,\n        entityId: localEntityId\n      }\n    }\n  } else if (message.type === CrdtMessageType.DELETE_COMPONENT_NETWORK) {\n    DeleteComponent.write(localEntityId, message.componentId, message.timestamp, destinationBuffer)\n    return {\n      type: CrdtMessageType.DELETE_COMPONENT,\n      componentId: message.componentId,\n      timestamp: message.timestamp,\n      entityId: localEntityId\n    }\n  } else if (message.type === CrdtMessageType.DELETE_ENTITY_NETWORK) {\n    DeleteEntity.write(localEntityId, destinationBuffer)\n    return {\n      type: CrdtMessageType.DELETE_ENTITY,\n      entityId: localEntityId\n    }\n  }\n  throw 1\n}\n\nexport function localMessageToNetwork(\n  message: ReceiveMessage,\n  network: INetowrkEntityType,\n  destinationBuffer: ByteBuffer\n) {\n  if (message.type === CrdtMessageType.PUT_COMPONENT) {\n    PutNetworkComponentOperation.write(\n      network.entityId,\n      message.timestamp,\n      message.componentId,\n      network.networkId,\n      message.data,\n      destinationBuffer\n    )\n  } else if (message.type === CrdtMessageType.DELETE_COMPONENT) {\n    DeleteComponentNetwork.write(\n      network.entityId,\n      message.componentId,\n      message.timestamp,\n      network.networkId,\n      destinationBuffer\n    )\n  } else if (message.type === CrdtMessageType.DELETE_ENTITY) {\n    DeleteEntityNetwork.write(network.entityId, network.networkId, destinationBuffer)\n  }\n}\n\nexport function fixTransformParent(message: ReceiveMessage, parent?: Entity): Uint8Array {\n  const buffer = new ReadWriteByteBuffer()\n  const transform = 'data' in message && TransformSchema.deserialize(new ReadWriteByteBuffer(message.data))\n\n  if (!transform) throw new Error('Invalid parent transform')\n\n  // Generate new transform raw data with the parent\n  const newTransform = { ...transform, parent }\n  TransformSchema.serialize(newTransform, buffer)\n  return buffer.toBinary()\n}\n"]}
|
package/server/env-var.d.ts
DELETED
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* EnvVar provides methods to fetch environment variables from the
|
|
3
|
-
* Server Side Storage service. This module only works when running
|
|
4
|
-
* on server-side scenes.
|
|
5
|
-
*/
|
|
6
|
-
export declare const EnvVar: {
|
|
7
|
-
/**
|
|
8
|
-
* Fetches a specific environment variable by key as plain text.
|
|
9
|
-
*
|
|
10
|
-
* @param key - The name of the environment variable to fetch
|
|
11
|
-
* @returns A promise that resolves to the plain text value, or empty string if not found
|
|
12
|
-
* @throws Error if not running on a server-side scene
|
|
13
|
-
*/
|
|
14
|
-
get(key: string): Promise<string>;
|
|
15
|
-
};
|
package/server/env-var.js
DELETED
|
@@ -1,31 +0,0 @@
|
|
|
1
|
-
import { getStorageServerUrl } from './storage-url';
|
|
2
|
-
import { assertIsServer, wrapSignedFetch } from './utils';
|
|
3
|
-
const MODULE_NAME = 'EnvVar';
|
|
4
|
-
/**
|
|
5
|
-
* EnvVar provides methods to fetch environment variables from the
|
|
6
|
-
* Server Side Storage service. This module only works when running
|
|
7
|
-
* on server-side scenes.
|
|
8
|
-
*/
|
|
9
|
-
export const EnvVar = {
|
|
10
|
-
/**
|
|
11
|
-
* Fetches a specific environment variable by key as plain text.
|
|
12
|
-
*
|
|
13
|
-
* @param key - The name of the environment variable to fetch
|
|
14
|
-
* @returns A promise that resolves to the plain text value, or empty string if not found
|
|
15
|
-
* @throws Error if not running on a server-side scene
|
|
16
|
-
*/
|
|
17
|
-
async get(key) {
|
|
18
|
-
assertIsServer(MODULE_NAME);
|
|
19
|
-
const baseUrl = await getStorageServerUrl();
|
|
20
|
-
const url = `${baseUrl}/env/${encodeURIComponent(key)}`;
|
|
21
|
-
const [error, data] = await wrapSignedFetch({
|
|
22
|
-
url
|
|
23
|
-
});
|
|
24
|
-
if (error) {
|
|
25
|
-
console.error(`Failed to fetch environment variable '${key}': ${error}`);
|
|
26
|
-
return '';
|
|
27
|
-
}
|
|
28
|
-
return data?.value ?? '';
|
|
29
|
-
}
|
|
30
|
-
};
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52LXZhci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9zZXJ2ZXIvZW52LXZhci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFDbkQsT0FBTyxFQUFFLGNBQWMsRUFBRSxlQUFlLEVBQUUsTUFBTSxTQUFTLENBQUE7QUFFekQsTUFBTSxXQUFXLEdBQUcsUUFBUSxDQUFBO0FBRTVCOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsTUFBTSxNQUFNLEdBQUc7SUFDcEI7Ozs7OztPQU1HO0lBQ0gsS0FBSyxDQUFDLEdBQUcsQ0FBQyxHQUFXO1FBQ25CLGNBQWMsQ0FBQyxXQUFXLENBQUMsQ0FBQTtRQUUzQixNQUFNLE9BQU8sR0FBRyxNQUFNLG1CQUFtQixFQUFFLENBQUE7UUFDM0MsTUFBTSxHQUFHLEdBQUcsR0FBRyxPQUFPLFFBQVEsa0JBQWtCLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQTtRQUV2RCxNQUFNLENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxHQUFHLE1BQU0sZUFBZSxDQUFvQjtZQUM3RCxHQUFHO1NBQ0osQ0FBQyxDQUFBO1FBRUYsSUFBSSxLQUFLLEVBQUU7WUFDVCxPQUFPLENBQUMsS0FBSyxDQUFDLHlDQUF5QyxHQUFHLE1BQU0sS0FBSyxFQUFFLENBQUMsQ0FBQTtZQUN4RSxPQUFPLEVBQUUsQ0FBQTtTQUNWO1FBRUQsT0FBTyxJQUFJLEVBQUUsS0FBSyxJQUFJLEVBQUUsQ0FBQTtJQUMxQixDQUFDO0NBQ0YsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGdldFN0b3JhZ2VTZXJ2ZXJVcmwgfSBmcm9tICcuL3N0b3JhZ2UtdXJsJ1xuaW1wb3J0IHsgYXNzZXJ0SXNTZXJ2ZXIsIHdyYXBTaWduZWRGZXRjaCB9IGZyb20gJy4vdXRpbHMnXG5cbmNvbnN0IE1PRFVMRV9OQU1FID0gJ0VudlZhcidcblxuLyoqXG4gKiBFbnZWYXIgcHJvdmlkZXMgbWV0aG9kcyB0byBmZXRjaCBlbnZpcm9ubWVudCB2YXJpYWJsZXMgZnJvbSB0aGVcbiAqIFNlcnZlciBTaWRlIFN0b3JhZ2Ugc2VydmljZS4gVGhpcyBtb2R1bGUgb25seSB3b3JrcyB3aGVuIHJ1bm5pbmdcbiAqIG9uIHNlcnZlci1zaWRlIHNjZW5lcy5cbiAqL1xuZXhwb3J0IGNvbnN0IEVudlZhciA9IHtcbiAgLyoqXG4gICAqIEZldGNoZXMgYSBzcGVjaWZpYyBlbnZpcm9ubWVudCB2YXJpYWJsZSBieSBrZXkgYXMgcGxhaW4gdGV4dC5cbiAgICpcbiAgICogQHBhcmFtIGtleSAtIFRoZSBuYW1lIG9mIHRoZSBlbnZpcm9ubWVudCB2YXJpYWJsZSB0byBmZXRjaFxuICAgKiBAcmV0dXJucyBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgcGxhaW4gdGV4dCB2YWx1ZSwgb3IgZW1wdHkgc3RyaW5nIGlmIG5vdCBmb3VuZFxuICAgKiBAdGhyb3dzIEVycm9yIGlmIG5vdCBydW5uaW5nIG9uIGEgc2VydmVyLXNpZGUgc2NlbmVcbiAgICovXG4gIGFzeW5jIGdldChrZXk6IHN0cmluZyk6IFByb21pc2U8c3RyaW5nPiB7XG4gICAgYXNzZXJ0SXNTZXJ2ZXIoTU9EVUxFX05BTUUpXG5cbiAgICBjb25zdCBiYXNlVXJsID0gYXdhaXQgZ2V0U3RvcmFnZVNlcnZlclVybCgpXG4gICAgY29uc3QgdXJsID0gYCR7YmFzZVVybH0vZW52LyR7ZW5jb2RlVVJJQ29tcG9uZW50KGtleSl9YFxuXG4gICAgY29uc3QgW2Vycm9yLCBkYXRhXSA9IGF3YWl0IHdyYXBTaWduZWRGZXRjaDx7IHZhbHVlOiBzdHJpbmcgfT4oe1xuICAgICAgdXJsXG4gICAgfSlcblxuICAgIGlmIChlcnJvcikge1xuICAgICAgY29uc29sZS5lcnJvcihgRmFpbGVkIHRvIGZldGNoIGVudmlyb25tZW50IHZhcmlhYmxlICcke2tleX0nOiAke2Vycm9yfWApXG4gICAgICByZXR1cm4gJydcbiAgICB9XG5cbiAgICByZXR1cm4gZGF0YT8udmFsdWUgPz8gJydcbiAgfVxufVxuIl19
|
package/server/index.d.ts
DELETED
package/server/index.js
DELETED
|
@@ -1,3 +0,0 @@
|
|
|
1
|
-
export { EnvVar } from './env-var';
|
|
2
|
-
export { Storage } from './storage';
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvc2VydmVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxXQUFXLENBQUE7QUFDbEMsT0FBTyxFQUFFLE9BQU8sRUFBOEUsTUFBTSxXQUFXLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBFbnZWYXIgfSBmcm9tICcuL2Vudi12YXInXG5leHBvcnQgeyBTdG9yYWdlLCBJU3RvcmFnZSwgSVNjZW5lU3RvcmFnZSwgSVBsYXllclN0b3JhZ2UsIEdldFZhbHVlc09wdGlvbnMsIEdldFZhbHVlc1Jlc3VsdCB9IGZyb20gJy4vc3RvcmFnZSdcbiJdfQ==
|