- App.tsx: full navigation (Auth stack + Main tabs with 5 screens) - Auth: LoginScreen, RegisterScreen, ForgotPasswordScreen - HomeScreen: dashboard with IoT metrics, weather widget, alerts, quick actions, sensors - MapScreen: interactive map with layer toggles (6 layers) - MarketplaceScreen: categories (6), products (5), search - ChatScreen: AI chat with quick prompts (4), bot responses - ProfileScreen: user info, stats, menu (9 items), logout - AlertsScreen: alert list with severity, acknowledge - SensorsScreen: sensor list with type filters (6 types), search - ZonesScreen: zone cards with stats - SettingsScreen: language picker (FR/EN/ES/DE), privacy, about - Stores: iotStore (sensors, zones, alerts), notificationStore, uiStore + i18n - Hooks: useSensors, useAlerts, useNotifications, useLocation - Components: Card, Button, LoadingSpinner, ErrorBoundary, Header - Services: iotService, notificationService (with axios API client) - Utils: formatters (temp, AQI, noise, dates), validators (email, password, IBAN) - Theme: colors.ts with full design system (Blue Ocean palette) - Ditto: fixed MongoDB connection, new JWT secrets, official gateway image
167 lines
7.1 KiB
JavaScript
167 lines
7.1 KiB
JavaScript
/**
|
|
* A message frame packer that serializes a messageKey and a payload into a binary format.
|
|
*
|
|
* +------------------+-------------------+----------------------------+--------------------------+
|
|
* | 4 bytes (Uint32) | Variable length | 1 byte (Uint8) | Variable length |
|
|
* | MessageKeyLength | MessageKey (JSON) | PayloadTypeIndicator (enum)| Payload (binary data) |
|
|
* +------------------+-------------------+----------------------------+--------------------------+
|
|
*
|
|
* MessageFrame Format:
|
|
*
|
|
* 1. MessageKeyLength (4 bytes):
|
|
* - A 4-byte unsigned integer indicating the length of the MessageKey JSON string.
|
|
*
|
|
* 2. MessageKey (Variable length):
|
|
* - The JSON string representing the message key, encoded as UTF-8.
|
|
*
|
|
* 3. PayloadTypeIndicator (1 byte):
|
|
* - A single byte enum value representing the type of the payload (e.g., Uint8Array, String, Object, ArrayBuffer, Blob).
|
|
*
|
|
* 4. Payload (Variable length):
|
|
* - The actual payload data, which can vary in type and length depending on the PayloadType.
|
|
*
|
|
*/
|
|
import { blobToArrayBufferAsync } from './blobUtils';
|
|
var PayloadTypeIndicator;
|
|
(function (PayloadTypeIndicator) {
|
|
PayloadTypeIndicator[PayloadTypeIndicator["Uint8Array"] = 1] = "Uint8Array";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["String"] = 2] = "String";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["Number"] = 3] = "Number";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["Null"] = 4] = "Null";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["Undefined"] = 5] = "Undefined";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["Object"] = 6] = "Object";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["ArrayBuffer"] = 7] = "ArrayBuffer";
|
|
PayloadTypeIndicator[PayloadTypeIndicator["Blob"] = 8] = "Blob";
|
|
})(PayloadTypeIndicator || (PayloadTypeIndicator = {}));
|
|
export class MessageFramePacker {
|
|
textEncoder = new TextEncoder();
|
|
textDecoder = new TextDecoder();
|
|
async pack({ messageKey, payload }) {
|
|
const messageKeyString = JSON.stringify(messageKey);
|
|
const messageKeyBytes = this.textEncoder.encode(messageKeyString);
|
|
const messageKeyLength = messageKeyBytes.length;
|
|
const payloadBinary = await this.payloadToUint8Array(payload);
|
|
const totalLength = 4 + messageKeyLength + 1 + payloadBinary.byteLength;
|
|
const buffer = new ArrayBuffer(totalLength);
|
|
const packedArray = new Uint8Array(buffer);
|
|
// [0] messageKeyLength (4 bytes)
|
|
const messageKeyLengthView = new DataView(buffer, 0, 4);
|
|
messageKeyLengthView.setUint32(0, messageKeyLength, false);
|
|
// [1] messageKey (variable length)
|
|
packedArray.set(messageKeyBytes, 4);
|
|
// [2] payloadTypeIndicator (1 byte)
|
|
const payloadTypeView = new DataView(buffer, 4 + messageKeyLength, 1);
|
|
payloadTypeView.setUint8(0, MessageFramePacker.getPayloadTypeIndicator(payload));
|
|
// [3] payload (variable length)
|
|
packedArray.set(payloadBinary, 4 + messageKeyLength + 1);
|
|
return packedArray;
|
|
}
|
|
async unpack(packedData) {
|
|
// [0] messageKeyLength (4 bytes)
|
|
const messageKeyLengthView = new DataView(packedData, 0, 4);
|
|
const messageKeyLength = messageKeyLengthView.getUint32(0, false);
|
|
// [1] messageKey (variable length)
|
|
const messageKeyBytes = packedData.slice(4, 4 + messageKeyLength);
|
|
const messageKeyString = this.textDecoder.decode(messageKeyBytes);
|
|
const messageKey = JSON.parse(messageKeyString);
|
|
// [2] payloadTypeIndicator (1 byte)
|
|
const payloadTypeView = new DataView(packedData, 4 + messageKeyLength, 1);
|
|
const payloadType = payloadTypeView.getUint8(0);
|
|
// [3] payload (variable length)
|
|
const payloadBinary = packedData.slice(4 + messageKeyLength + 1);
|
|
const payload = await this.deserializePayload(payloadBinary, payloadType);
|
|
return { messageKey, payload };
|
|
}
|
|
async payloadToUint8Array(payload) {
|
|
if (payload instanceof Uint8Array) {
|
|
return payload;
|
|
}
|
|
else if (typeof payload === 'string') {
|
|
return this.textEncoder.encode(payload);
|
|
}
|
|
else if (typeof payload === 'number') {
|
|
const buffer = new ArrayBuffer(8);
|
|
const view = new DataView(buffer);
|
|
view.setFloat64(0, payload, false);
|
|
return new Uint8Array(buffer);
|
|
}
|
|
else if (payload === null) {
|
|
return new Uint8Array(0);
|
|
}
|
|
else if (payload === undefined) {
|
|
return new Uint8Array(0);
|
|
}
|
|
else if (payload instanceof ArrayBuffer) {
|
|
return new Uint8Array(payload);
|
|
}
|
|
else if (payload instanceof Blob) {
|
|
return new Uint8Array(await blobToArrayBufferAsync(payload));
|
|
}
|
|
else {
|
|
return this.textEncoder.encode(JSON.stringify(payload));
|
|
}
|
|
}
|
|
async deserializePayload(payloadBinary, payloadTypeIndicator) {
|
|
switch (payloadTypeIndicator) {
|
|
case PayloadTypeIndicator.Uint8Array: {
|
|
return new Uint8Array(payloadBinary);
|
|
}
|
|
case PayloadTypeIndicator.String: {
|
|
return this.textDecoder.decode(payloadBinary);
|
|
}
|
|
case PayloadTypeIndicator.Number: {
|
|
const view = new DataView(payloadBinary);
|
|
return view.getFloat64(0, false);
|
|
}
|
|
case PayloadTypeIndicator.Null: {
|
|
return null;
|
|
}
|
|
case PayloadTypeIndicator.Undefined: {
|
|
return undefined;
|
|
}
|
|
case PayloadTypeIndicator.Object: {
|
|
const jsonString = this.textDecoder.decode(payloadBinary);
|
|
return JSON.parse(jsonString);
|
|
}
|
|
case PayloadTypeIndicator.ArrayBuffer: {
|
|
return payloadBinary;
|
|
}
|
|
case PayloadTypeIndicator.Blob: {
|
|
// @ts-ignore: We only test ArrayBuffer -> Blob on Node.js runtime
|
|
return new Blob([payloadBinary]);
|
|
}
|
|
default:
|
|
throw new Error('Unsupported payload type');
|
|
}
|
|
}
|
|
static getPayloadTypeIndicator(payload) {
|
|
if (payload instanceof Uint8Array) {
|
|
return PayloadTypeIndicator.Uint8Array;
|
|
}
|
|
else if (typeof payload === 'string') {
|
|
return PayloadTypeIndicator.String;
|
|
}
|
|
else if (typeof payload === 'number') {
|
|
return PayloadTypeIndicator.Number;
|
|
}
|
|
else if (payload === null) {
|
|
return PayloadTypeIndicator.Null;
|
|
}
|
|
else if (payload === undefined) {
|
|
return PayloadTypeIndicator.Undefined;
|
|
}
|
|
else if (payload instanceof ArrayBuffer) {
|
|
return PayloadTypeIndicator.ArrayBuffer;
|
|
}
|
|
else if (payload instanceof Blob) {
|
|
return PayloadTypeIndicator.Blob;
|
|
}
|
|
else if (typeof payload === 'object') {
|
|
return PayloadTypeIndicator.Object;
|
|
}
|
|
else {
|
|
throw new Error('Unsupported payload type');
|
|
}
|
|
}
|
|
}
|
|
//# sourceMappingURL=MessageFramePacker.js.map
|