@crysnovax/baileys 1.0.3

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 (112) hide show
  1. package/README.md +467 -0
  2. package/WAProto/V +1 -0
  3. package/WAProto/index.js +104236 -0
  4. package/engine-requirements.js +13 -0
  5. package/lib/Defaults/index.js +148 -0
  6. package/lib/Signal/Group/ciphertext-message.js +11 -0
  7. package/lib/Signal/Group/group-session-builder.js +29 -0
  8. package/lib/Signal/Group/group_cipher.js +81 -0
  9. package/lib/Signal/Group/index.js +11 -0
  10. package/lib/Signal/Group/keyhelper.js +17 -0
  11. package/lib/Signal/Group/sender-chain-key.js +25 -0
  12. package/lib/Signal/Group/sender-key-distribution-message.js +62 -0
  13. package/lib/Signal/Group/sender-key-message.js +65 -0
  14. package/lib/Signal/Group/sender-key-name.js +47 -0
  15. package/lib/Signal/Group/sender-key-record.js +40 -0
  16. package/lib/Signal/Group/sender-key-state.js +83 -0
  17. package/lib/Signal/Group/sender-message-key.js +25 -0
  18. package/lib/Signal/libsignal.js +406 -0
  19. package/lib/Signal/lid-mapping.js +276 -0
  20. package/lib/Socket/Client/index.js +2 -0
  21. package/lib/Socket/Client/types.js +10 -0
  22. package/lib/Socket/Client/websocket.js +53 -0
  23. package/lib/Socket/business.js +378 -0
  24. package/lib/Socket/chats.js +1059 -0
  25. package/lib/Socket/communities.js +430 -0
  26. package/lib/Socket/groups.js +328 -0
  27. package/lib/Socket/index.js +11 -0
  28. package/lib/Socket/messages-recv.js +1476 -0
  29. package/lib/Socket/messages-send.js +1268 -0
  30. package/lib/Socket/mex.js +41 -0
  31. package/lib/Socket/newsletter.js +251 -0
  32. package/lib/Socket/socket.js +949 -0
  33. package/lib/Store/index.js +3 -0
  34. package/lib/Store/make-in-memory-store.js +420 -0
  35. package/lib/Store/make-ordered-dictionary.js +78 -0
  36. package/lib/Store/object-repository.js +23 -0
  37. package/lib/Types/Auth.js +1 -0
  38. package/lib/Types/Bussines.js +1 -0
  39. package/lib/Types/Call.js +1 -0
  40. package/lib/Types/Chat.js +7 -0
  41. package/lib/Types/Contact.js +1 -0
  42. package/lib/Types/Events.js +1 -0
  43. package/lib/Types/GroupMetadata.js +1 -0
  44. package/lib/Types/Label.js +24 -0
  45. package/lib/Types/LabelAssociation.js +6 -0
  46. package/lib/Types/Message.js +17 -0
  47. package/lib/Types/Newsletter.js +33 -0
  48. package/lib/Types/Product.js +1 -0
  49. package/lib/Types/RichType.js +22 -0
  50. package/lib/Types/Signal.js +1 -0
  51. package/lib/Types/Socket.js +2 -0
  52. package/lib/Types/State.js +12 -0
  53. package/lib/Types/USync.js +1 -0
  54. package/lib/Types/index.js +25 -0
  55. package/lib/Utils/auth-utils.js +289 -0
  56. package/lib/Utils/bot-planning-replay.js +206 -0
  57. package/lib/Utils/browser-utils.js +28 -0
  58. package/lib/Utils/business.js +230 -0
  59. package/lib/Utils/chat-utils.js +811 -0
  60. package/lib/Utils/companion-reg-client-utils.js +32 -0
  61. package/lib/Utils/crypto.js +117 -0
  62. package/lib/Utils/decode-wa-message.js +282 -0
  63. package/lib/Utils/event-buffer.js +589 -0
  64. package/lib/Utils/generics.js +385 -0
  65. package/lib/Utils/history.js +130 -0
  66. package/lib/Utils/identity-change-handler.js +48 -0
  67. package/lib/Utils/index.js +26 -0
  68. package/lib/Utils/link-preview.js +84 -0
  69. package/lib/Utils/logger.js +2 -0
  70. package/lib/Utils/lt-hash.js +7 -0
  71. package/lib/Utils/make-mutex.js +32 -0
  72. package/lib/Utils/message-retry-manager.js +241 -0
  73. package/lib/Utils/messages-media.js +830 -0
  74. package/lib/Utils/messages.js +1891 -0
  75. package/lib/Utils/meta-compositing.js +208 -0
  76. package/lib/Utils/noise-handler.js +200 -0
  77. package/lib/Utils/offline-node-processor.js +39 -0
  78. package/lib/Utils/pre-key-manager.js +105 -0
  79. package/lib/Utils/process-message.js +527 -0
  80. package/lib/Utils/reporting-utils.js +257 -0
  81. package/lib/Utils/rich-message-utils.js +387 -0
  82. package/lib/Utils/signal.js +158 -0
  83. package/lib/Utils/stanza-ack.js +37 -0
  84. package/lib/Utils/sync-action-utils.js +47 -0
  85. package/lib/Utils/tc-token-utils.js +17 -0
  86. package/lib/Utils/use-multi-file-auth-state.js +120 -0
  87. package/lib/Utils/use-single-file-auth-state.js +96 -0
  88. package/lib/Utils/validate-connection.js +206 -0
  89. package/lib/Utils//360/237/224/226 +217 -0
  90. package/lib/WABinary/constants.js +1372 -0
  91. package/lib/WABinary/decode.js +261 -0
  92. package/lib/WABinary/encode.js +219 -0
  93. package/lib/WABinary/generic-utils.js +227 -0
  94. package/lib/WABinary/index.js +5 -0
  95. package/lib/WABinary/jid-utils.js +95 -0
  96. package/lib/WABinary/types.js +1 -0
  97. package/lib/WAM/BinaryInfo.js +9 -0
  98. package/lib/WAM/constants.js +22852 -0
  99. package/lib/WAM/encode.js +149 -0
  100. package/lib/WAM/index.js +3 -0
  101. package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -0
  102. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -0
  103. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +26 -0
  104. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +37 -0
  105. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -0
  106. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +28 -0
  107. package/lib/WAUSync/Protocols/index.js +4 -0
  108. package/lib/WAUSync/USyncQuery.js +93 -0
  109. package/lib/WAUSync/USyncUser.js +22 -0
  110. package/lib/WAUSync/index.js +3 -0
  111. package/lib/index.js +11 -0
  112. package/package.json +83 -0
@@ -0,0 +1,208 @@
1
+ /**
2
+ * Lia@Changes [WIP]
3
+ * Meta Compositing — send rich messages (code, table, text, etc.) with a
4
+ * Meta AI-style progress indicator that appears BEFORE the final message.
5
+ *
6
+ * The progress placeholder is sent first (with BotProgressIndicatorMetadata),
7
+ * then deleted, then the final rich message is sent fresh — so NO "edited"
8
+ * badge ever appears. Identical to how Meta AI itself works.
9
+ *
10
+ * metaTyping() — shows the "typing..." / planning steps indicator only.
11
+ * sendMetaComposited() — full flow: typing → delete → final message.
12
+ *
13
+ * If you use or copy this code, please credit @crysnovax/bailey.
14
+ */
15
+
16
+ import { proto } from '../../WAProto/index.js';
17
+ import { prepareRichResponseMessage, wrapToBotForwardedMessage, botMetadataSignature, botMetadataCertificate } from './rich-message-utils.js';
18
+ import { BOT_RENDERING_CONFIG_METADATA } from '../Defaults/index.js';
19
+ import { generateWAMessageFromContent } from './messages.js';
20
+ import { unixTimestampSeconds, delay } from './generics.js';
21
+
22
+ // ─── Step status enum mirrors proto.BotProgressIndicatorMetadata.BotPlanningStepMetadata status field ───
23
+ export const PlanningStepStatus = {
24
+ IN_PROGRESS: 0,
25
+ DONE: 1,
26
+ FAILED: 2
27
+ };
28
+
29
+ /**
30
+ * Build a BotProgressIndicatorMetadata object.
31
+ * @param {string} description - Top-level label shown while "thinking"
32
+ * @param {Array} steps - Array of { title, body?, status?, isReasoning? }
33
+ * @param {number} estimatedMs - Optional estimated completion time in ms
34
+ */
35
+ export const buildProgressIndicator = (description, steps = [], estimatedMs) => {
36
+ const stepsMetadata = steps.map(step => {
37
+ const s = {
38
+ statusTitle: step.title,
39
+ status: step.status ?? PlanningStepStatus.IN_PROGRESS
40
+ };
41
+ if (step.body) s.statusBody = step.body;
42
+ if (step.isReasoning) s.isReasoning = true;
43
+ if (step.isEnhancedSearch) s.isEnhancedSearch = true;
44
+ return s;
45
+ });
46
+
47
+ const indicator = { stepsMetadata };
48
+ if (description) indicator.progressDescription = description;
49
+ if (estimatedMs != null) indicator.estimatedCompletionTime = estimatedMs;
50
+ return indicator;
51
+ };
52
+
53
+ /**
54
+ * Build the compositing placeholder message — the one that shows the
55
+ * "Meta AI is thinking..." / planning steps indicator in the chat bubble.
56
+ *
57
+ * This is a botForwardedMessage with richResponseMessage = null content
58
+ * but with progressIndicatorMetadata on the botMetadata.
59
+ *
60
+ * @param {object} options
61
+ * @param {string} options.description - "Thinking…" label
62
+ * @param {Array} options.steps - Planning steps array
63
+ * @param {number} [options.estimatedMs] - Estimated completion ms
64
+ * @param {string} [options.placeholderText] - Text shown in the bubble body while loading
65
+ */
66
+ export const buildCompositingPlaceholder = ({
67
+ description = 'Thinking…',
68
+ steps = [],
69
+ estimatedMs,
70
+ placeholderText = ''
71
+ }) => {
72
+ const progressIndicatorMetadata = buildProgressIndicator(description, steps, estimatedMs);
73
+
74
+ // Minimal richResponseMessage so the bubble renders in Bot mode
75
+ const textEncoder = new TextEncoder();
76
+ const unifiedData = textEncoder.encode(JSON.stringify({
77
+ response_id: crypto.randomUUID(),
78
+ sections: placeholderText ? [{
79
+ view_model: {
80
+ primitive: {
81
+ text: placeholderText,
82
+ inline_entities: [],
83
+ __typename: 'GenAIMarkdownTextUXPrimitive'
84
+ },
85
+ __typename: 'GenAISingleLayoutViewModel'
86
+ }
87
+ }] : []
88
+ }));
89
+
90
+ const richResponseMessage = {
91
+ messageType: proto.AIRichResponseMessageType.AI_RICH_RESPONSE_TYPE_STANDARD,
92
+ unifiedResponse: { data: unifiedData },
93
+ submessages: []
94
+ };
95
+
96
+ return {
97
+ messageContextInfo: {
98
+ botMetadata: {
99
+ pluginMetadata: {},
100
+ progressIndicatorMetadata,
101
+ verificationMetadata: {
102
+ proofs: [{
103
+ certificateChain: [
104
+ botMetadataCertificate(684),
105
+ botMetadataCertificate(892)
106
+ ],
107
+ version: 1,
108
+ useCase: 1,
109
+ signature: botMetadataSignature()
110
+ }]
111
+ },
112
+ botRenderingConfigMetadata: BOT_RENDERING_CONFIG_METADATA
113
+ }
114
+ },
115
+ botForwardedMessage: {
116
+ message: { richResponseMessage }
117
+ }
118
+ };
119
+ };
120
+
121
+ /**
122
+ * metaTyping — sends ONLY the progress/compositing indicator to a JID.
123
+ * Does NOT delete or send a follow-up — caller controls that.
124
+ *
125
+ * Returns the sent message key so you can delete it later.
126
+ *
127
+ * @param {object} sock - Baileys socket
128
+ * @param {string} jid - Destination JID
129
+ * @param {object} options - Same options as buildCompositingPlaceholder
130
+ * @returns {Promise<object>} - The sent WAMessage
131
+ */
132
+ export const metaTyping = async (sock, jid, {
133
+ description = 'Thinking…',
134
+ steps = [],
135
+ estimatedMs,
136
+ placeholderText = ''
137
+ } = {}) => {
138
+ const placeholder = buildCompositingPlaceholder({
139
+ description,
140
+ steps,
141
+ estimatedMs,
142
+ placeholderText
143
+ });
144
+
145
+ return sock.sendMessage(jid, { raw: true, ...placeholder });
146
+ };
147
+
148
+ /**
149
+ * sendMetaComposited — full Meta AI flow:
150
+ * 1. Send progress indicator placeholder
151
+ * 2. Wait `thinkingMs` (default 2000ms)
152
+ * 3. Delete the placeholder (no "edited" badge ever appears)
153
+ * 4. Send the final rich message fresh
154
+ *
155
+ * Supports all existing richResponse content types: text, code, table,
156
+ * expressions (LaTeX), items (reels carousel), or the richResponse array.
157
+ *
158
+ * @param {object} sock - Baileys socket
159
+ * @param {string} jid - Destination JID
160
+ * @param {object} content - Same content object as sendMessage richResponse
161
+ * @param {object} [options]
162
+ * @param {number} [options.thinkingMs=2000] - How long placeholder shows
163
+ * @param {string} [options.description='Thinking…'] - Placeholder label
164
+ * @param {Array} [options.steps=[]] - Planning steps to show
165
+ * @param {string} [options.placeholderText=''] - Body text while loading
166
+ * @param {object} [options.sendOptions={}] - Extra options for final sendMessage
167
+ * @returns {Promise<object>} - The final sent WAMessage
168
+ */
169
+ export const sendMetaComposited = async (sock, jid, content, {
170
+ thinkingMs = 2000,
171
+ description = 'Thinking…',
172
+ steps = [],
173
+ placeholderText = '',
174
+ sendOptions = {}
175
+ } = {}) => {
176
+ // 1. Send the compositing placeholder
177
+ const placeholder = await metaTyping(sock, jid, {
178
+ description,
179
+ steps,
180
+ placeholderText
181
+ });
182
+
183
+ try {
184
+ // 2. Wait — this is the "thinking" window
185
+ await delay(thinkingMs);
186
+
187
+ // 3. Delete the placeholder silently
188
+ if (placeholder?.key) {
189
+ await sock.sendMessage(jid, { delete: placeholder.key });
190
+ }
191
+ } catch (_) {
192
+ // Non-fatal — always attempt final send
193
+ }
194
+
195
+ // 4. Send the final rich message as a brand-new message (no edit = no badge)
196
+ return sock.sendMessage(jid, content, sendOptions);
197
+ };
198
+
199
+ /**
200
+ * Convenience: build a steps array from plain strings with IN_PROGRESS status.
201
+ * Use PlanningStepStatus.DONE to mark a step complete.
202
+ *
203
+ * @example
204
+ * buildSteps(['Searching…', 'Reading sources…', 'Writing response…'])
205
+ */
206
+ export const buildSteps = (titles, status = PlanningStepStatus.IN_PROGRESS) =>
207
+ titles.map(title => ({ title, status }));
208
+
@@ -0,0 +1,200 @@
1
+ import { Boom } from '@hapi/boom';
2
+ import { proto } from '../../WAProto/index.js';
3
+ import { NOISE_MODE, WA_CERT_DETAILS } from '../Defaults/index.js';
4
+ import { decodeBinaryNode } from '../WABinary/index.js';
5
+ import { aesDecryptGCM, aesEncryptGCM, Curve, hkdf, sha256 } from './crypto.js';
6
+ const IV_LENGTH = 12;
7
+ const EMPTY_BUFFER = Buffer.alloc(0);
8
+ const generateIV = (counter) => {
9
+ const iv = new ArrayBuffer(IV_LENGTH);
10
+ new DataView(iv).setUint32(8, counter);
11
+ return new Uint8Array(iv);
12
+ };
13
+ class TransportState {
14
+ constructor(encKey, decKey) {
15
+ this.encKey = encKey;
16
+ this.decKey = decKey;
17
+ this.readCounter = 0;
18
+ this.writeCounter = 0;
19
+ this.iv = new Uint8Array(IV_LENGTH);
20
+ }
21
+ encrypt(plaintext) {
22
+ const c = this.writeCounter++;
23
+ this.iv[8] = (c >>> 24) & 0xff;
24
+ this.iv[9] = (c >>> 16) & 0xff;
25
+ this.iv[10] = (c >>> 8) & 0xff;
26
+ this.iv[11] = c & 0xff;
27
+ return aesEncryptGCM(plaintext, this.encKey, this.iv, EMPTY_BUFFER);
28
+ }
29
+ decrypt(ciphertext) {
30
+ const c = this.readCounter++;
31
+ this.iv[8] = (c >>> 24) & 0xff;
32
+ this.iv[9] = (c >>> 16) & 0xff;
33
+ this.iv[10] = (c >>> 8) & 0xff;
34
+ this.iv[11] = c & 0xff;
35
+ return aesDecryptGCM(ciphertext, this.decKey, this.iv, EMPTY_BUFFER);
36
+ }
37
+ }
38
+ export const makeNoiseHandler = ({ keyPair: { private: privateKey, public: publicKey }, NOISE_HEADER, logger, routingInfo }) => {
39
+ logger = logger.child({ class: 'ns' });
40
+ const data = Buffer.from(NOISE_MODE);
41
+ let hash = data.byteLength === 32 ? data : sha256(data);
42
+ let salt = hash;
43
+ let encKey = hash;
44
+ let decKey = hash;
45
+ let counter = 0;
46
+ let sentIntro = false;
47
+ let inBytes = Buffer.alloc(0);
48
+ let transport = null;
49
+ let isWaitingForTransport = false;
50
+ let pendingOnFrame = null;
51
+ let introHeader;
52
+ if (routingInfo) {
53
+ introHeader = Buffer.alloc(7 + routingInfo.byteLength + NOISE_HEADER.length);
54
+ introHeader.write('ED', 0, 'utf8');
55
+ introHeader.writeUint8(0, 2);
56
+ introHeader.writeUint8(1, 3);
57
+ introHeader.writeUint8(routingInfo.byteLength >> 16, 4);
58
+ introHeader.writeUint16BE(routingInfo.byteLength & 65535, 5);
59
+ introHeader.set(routingInfo, 7);
60
+ introHeader.set(NOISE_HEADER, 7 + routingInfo.byteLength);
61
+ }
62
+ else {
63
+ introHeader = Buffer.from(NOISE_HEADER);
64
+ }
65
+ const authenticate = (data) => {
66
+ if (!transport) {
67
+ hash = sha256(Buffer.concat([hash, data]));
68
+ }
69
+ };
70
+ const encrypt = (plaintext) => {
71
+ if (transport) {
72
+ return transport.encrypt(plaintext);
73
+ }
74
+ const result = aesEncryptGCM(plaintext, encKey, generateIV(counter++), hash);
75
+ authenticate(result);
76
+ return result;
77
+ };
78
+ const decrypt = (ciphertext) => {
79
+ if (transport) {
80
+ return transport.decrypt(ciphertext);
81
+ }
82
+ const result = aesDecryptGCM(ciphertext, decKey, generateIV(counter++), hash);
83
+ authenticate(ciphertext);
84
+ return result;
85
+ };
86
+ const localHKDF = (data) => {
87
+ const key = hkdf(Buffer.from(data), 64, { salt, info: '' });
88
+ return [key.subarray(0, 32), key.subarray(32)];
89
+ };
90
+ const mixIntoKey = (data) => {
91
+ const [write, read] = localHKDF(data);
92
+ salt = write;
93
+ encKey = read;
94
+ decKey = read;
95
+ counter = 0;
96
+ };
97
+ const finishInit = async () => {
98
+ isWaitingForTransport = true;
99
+ const [write, read] = localHKDF(new Uint8Array(0));
100
+ transport = new TransportState(write, read);
101
+ isWaitingForTransport = false;
102
+ logger.trace('Noise handler transitioned to Transport state');
103
+ if (pendingOnFrame) {
104
+ logger.trace({ length: inBytes.length }, 'Flushing buffered frames after transport ready');
105
+ await processData(pendingOnFrame);
106
+ pendingOnFrame = null;
107
+ }
108
+ };
109
+ const processData = async (onFrame) => {
110
+ let size;
111
+ while (true) {
112
+ if (inBytes.length < 3)
113
+ return;
114
+ size = (inBytes[0] << 16) | (inBytes[1] << 8) | inBytes[2];
115
+ if (inBytes.length < size + 3)
116
+ return;
117
+ let frame = inBytes.subarray(3, size + 3);
118
+ inBytes = inBytes.subarray(size + 3);
119
+ if (transport) {
120
+ const result = transport.decrypt(frame);
121
+ frame = await decodeBinaryNode(result);
122
+ }
123
+ if (logger.level === 'trace') {
124
+ logger.trace({ msg: frame?.attrs?.id }, 'recv frame');
125
+ }
126
+ onFrame(frame);
127
+ }
128
+ };
129
+ authenticate(NOISE_HEADER);
130
+ authenticate(publicKey);
131
+ return {
132
+ encrypt,
133
+ decrypt,
134
+ authenticate,
135
+ mixIntoKey,
136
+ finishInit,
137
+ processHandshake: ({ serverHello }, noiseKey) => {
138
+ authenticate(serverHello.ephemeral);
139
+ mixIntoKey(Curve.sharedKey(privateKey, serverHello.ephemeral));
140
+ const decStaticContent = decrypt(serverHello.static);
141
+ mixIntoKey(Curve.sharedKey(privateKey, decStaticContent));
142
+ const certDecoded = decrypt(serverHello.payload);
143
+ const { intermediate: certIntermediate, leaf } = proto.CertChain.decode(certDecoded);
144
+ // leaf
145
+ if (!leaf?.details || !leaf?.signature) {
146
+ throw new Boom('invalid noise leaf certificate', { statusCode: 400 });
147
+ }
148
+ if (!certIntermediate?.details || !certIntermediate?.signature) {
149
+ throw new Boom('invalid noise intermediate certificate', { statusCode: 400 });
150
+ }
151
+ const details = proto.CertChain.NoiseCertificate.Details.decode(certIntermediate.details);
152
+ const { issuerSerial } = details;
153
+ const verify = Curve.verify(details.key, leaf.details, leaf.signature);
154
+ const verifyIntermediate = Curve.verify(WA_CERT_DETAILS.PUBLIC_KEY, certIntermediate.details, certIntermediate.signature);
155
+ if (!verify) {
156
+ throw new Boom('noise certificate signature invalid', { statusCode: 400 });
157
+ }
158
+ if (!verifyIntermediate) {
159
+ throw new Boom('noise intermediate certificate signature invalid', { statusCode: 400 });
160
+ }
161
+ if (issuerSerial !== WA_CERT_DETAILS.SERIAL) {
162
+ throw new Boom('certification match failed', { statusCode: 400 });
163
+ }
164
+ const keyEnc = encrypt(noiseKey.public);
165
+ mixIntoKey(Curve.sharedKey(noiseKey.private, serverHello.ephemeral));
166
+ return keyEnc;
167
+ },
168
+ encodeFrame: (data) => {
169
+ if (transport) {
170
+ data = transport.encrypt(data);
171
+ }
172
+ const dataLen = data.byteLength;
173
+ const introSize = sentIntro ? 0 : introHeader.length;
174
+ const frame = Buffer.allocUnsafe(introSize + 3 + dataLen);
175
+ if (!sentIntro) {
176
+ frame.set(introHeader);
177
+ sentIntro = true;
178
+ }
179
+ frame[introSize] = (dataLen >>> 16) & 0xff;
180
+ frame[introSize + 1] = (dataLen >>> 8) & 0xff;
181
+ frame[introSize + 2] = dataLen & 0xff;
182
+ frame.set(data, introSize + 3);
183
+ return frame;
184
+ },
185
+ decodeFrame: async (newData, onFrame) => {
186
+ if (isWaitingForTransport) {
187
+ inBytes = Buffer.concat([inBytes, newData]);
188
+ pendingOnFrame = onFrame;
189
+ return;
190
+ }
191
+ if (inBytes.length === 0) {
192
+ inBytes = Buffer.from(newData);
193
+ }
194
+ else {
195
+ inBytes = Buffer.concat([inBytes, newData]);
196
+ }
197
+ await processData(onFrame);
198
+ }
199
+ };
200
+ };
@@ -0,0 +1,39 @@
1
+ /**
2
+ * Creates a processor for offline stanza nodes that:
3
+ * - Queues nodes for sequential processing
4
+ * - Yields to the event loop periodically to avoid blocking
5
+ * - Catches handler errors to prevent the processing loop from crashing
6
+ */
7
+ export function makeOfflineNodeProcessor(nodeProcessorMap, deps, batchSize = 10) {
8
+ const nodes = [];
9
+ let isProcessing = false;
10
+ const enqueue = (type, node) => {
11
+ nodes.push({ type, node });
12
+ if (isProcessing) {
13
+ return;
14
+ }
15
+ isProcessing = true;
16
+ const promise = async () => {
17
+ let processedInBatch = 0;
18
+ while (nodes.length && deps.isWsOpen()) {
19
+ const { type, node } = nodes.shift();
20
+ const nodeProcessor = nodeProcessorMap.get(type);
21
+ if (!nodeProcessor) {
22
+ deps.onUnexpectedError(new Error(`unknown offline node type: ${type}`), 'processing offline node');
23
+ continue;
24
+ }
25
+ await nodeProcessor(node).catch(err => deps.onUnexpectedError(err, `processing offline ${type}`));
26
+ processedInBatch++;
27
+ // Yield to event loop after processing a batch
28
+ // This prevents blocking the event loop for too long when there are many offline nodes
29
+ if (processedInBatch >= batchSize) {
30
+ processedInBatch = 0;
31
+ await deps.yieldToEventLoop();
32
+ }
33
+ }
34
+ isProcessing = false;
35
+ };
36
+ promise().catch(error => deps.onUnexpectedError(error, 'processing offline nodes'));
37
+ };
38
+ return { enqueue };
39
+ }
@@ -0,0 +1,105 @@
1
+ import PQueue from 'p-queue';
2
+ /**
3
+ * Manages pre-key operations with proper concurrency control
4
+ */
5
+ export class PreKeyManager {
6
+ constructor(store, logger) {
7
+ this.store = store;
8
+ this.logger = logger;
9
+ this.queues = new Map();
10
+ }
11
+ /**
12
+ * Get or create a queue for a specific key type
13
+ */
14
+ getQueue(keyType) {
15
+ if (!this.queues.has(keyType)) {
16
+ this.queues.set(keyType, new PQueue({ concurrency: 1 }));
17
+ }
18
+ return this.queues.get(keyType);
19
+ }
20
+ /**
21
+ * Process pre-key operations (updates and deletions)
22
+ */
23
+ async processOperations(data, keyType, transactionCache, mutations, isInTransaction) {
24
+ const keyData = data[keyType];
25
+ if (!keyData)
26
+ return;
27
+ return this.getQueue(keyType).add(async () => {
28
+ // Ensure structures exist
29
+ transactionCache[keyType] = transactionCache[keyType] || {};
30
+ mutations[keyType] = mutations[keyType] || {};
31
+ // Separate deletions from updates
32
+ const deletions = [];
33
+ const updates = {};
34
+ for (const keyId in keyData) {
35
+ if (keyData[keyId] === null) {
36
+ deletions.push(keyId);
37
+ }
38
+ else {
39
+ updates[keyId] = keyData[keyId];
40
+ }
41
+ }
42
+ // Process updates (no validation needed)
43
+ if (Object.keys(updates).length > 0) {
44
+ Object.assign(transactionCache[keyType], updates);
45
+ Object.assign(mutations[keyType], updates);
46
+ }
47
+ // Process deletions with validation
48
+ if (deletions.length > 0) {
49
+ await this.processDeletions(keyType, deletions, transactionCache, mutations, isInTransaction);
50
+ }
51
+ });
52
+ }
53
+ /**
54
+ * Process deletions with validation
55
+ */
56
+ async processDeletions(keyType, ids, transactionCache, mutations, isInTransaction) {
57
+ if (isInTransaction) {
58
+ // In transaction, only allow deletion if key exists in cache
59
+ for (const keyId of ids) {
60
+ if (transactionCache[keyType]?.[keyId]) {
61
+ transactionCache[keyType][keyId] = null;
62
+ mutations[keyType][keyId] = null;
63
+ }
64
+ else {
65
+ this.logger.warn(`Skipping deletion of non-existent ${keyType} in transaction: ${keyId}`);
66
+ }
67
+ }
68
+ }
69
+ else {
70
+ // Outside transaction, validate against store
71
+ const existingKeys = await this.store.get(keyType, ids);
72
+ for (const keyId of ids) {
73
+ if (existingKeys[keyId]) {
74
+ transactionCache[keyType][keyId] = null;
75
+ mutations[keyType][keyId] = null;
76
+ }
77
+ else {
78
+ this.logger.warn(`Skipping deletion of non-existent ${keyType}: ${keyId}`);
79
+ }
80
+ }
81
+ }
82
+ }
83
+ /**
84
+ * Validate and process pre-key deletions outside transactions
85
+ */
86
+ async validateDeletions(data, keyType) {
87
+ const keyData = data[keyType];
88
+ if (!keyData)
89
+ return;
90
+ return this.getQueue(keyType).add(async () => {
91
+ // Find all deletion requests
92
+ const deletionIds = Object.keys(keyData).filter(id => keyData[id] === null);
93
+ if (deletionIds.length === 0)
94
+ return;
95
+ // Validate deletions
96
+ const existingKeys = await this.store.get(keyType, deletionIds);
97
+ for (const keyId of deletionIds) {
98
+ if (!existingKeys[keyId]) {
99
+ this.logger.warn(`Skipping deletion of non-existent ${keyType}: ${keyId}`);
100
+ delete data[keyType][keyId];
101
+ }
102
+ }
103
+ });
104
+ }
105
+ }