@baileys-md/baileys 11.2.4 → 12.0.1

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 (126) hide show
  1. package/LICENSE +1 -1
  2. package/WAProto/WAProto.proto +5311 -0
  3. package/WAProto/index.js +70084 -131686
  4. package/lib/Defaults/index.js +118 -117
  5. package/lib/KeyDB/BinarySearch.js +20 -0
  6. package/lib/KeyDB/KeyedDB.js +167 -0
  7. package/lib/KeyDB/index.js +4 -0
  8. package/lib/Signal/Group/ciphertext-message.js +13 -0
  9. package/lib/Signal/Group/group-session-builder.js +32 -0
  10. package/lib/Signal/Group/group_cipher.js +84 -0
  11. package/lib/Signal/Group/index.js +13 -0
  12. package/lib/Signal/Group/keyhelper.js +20 -0
  13. package/lib/Signal/Group/sender-chain-key.js +28 -0
  14. package/lib/Signal/Group/sender-key-distribution-message.js +65 -0
  15. package/lib/Signal/Group/sender-key-message.js +68 -0
  16. package/{WASignalGroup/sender_key_name.js → lib/Signal/Group/sender-key-name.js} +10 -28
  17. package/lib/Signal/Group/sender-key-record.js +43 -0
  18. package/lib/Signal/Group/sender-key-state.js +86 -0
  19. package/lib/Signal/Group/sender-message-key.js +28 -0
  20. package/lib/Signal/libsignal.js +314 -151
  21. package/lib/Signal/lid-mapping.js +155 -0
  22. package/lib/Socket/Client/index.js +4 -19
  23. package/lib/Socket/Client/types.js +13 -0
  24. package/lib/Socket/Client/websocket.js +52 -0
  25. package/lib/Socket/business.js +359 -242
  26. package/lib/Socket/chats.js +847 -844
  27. package/lib/Socket/communities.js +413 -0
  28. package/lib/Socket/groups.js +304 -319
  29. package/lib/Socket/index.js +15 -9
  30. package/lib/Socket/messages-recv.js +1109 -989
  31. package/lib/Socket/messages-send.js +611 -347
  32. package/lib/Socket/mex.js +45 -0
  33. package/lib/Socket/newsletter.js +230 -231
  34. package/lib/Socket/socket.js +795 -616
  35. package/lib/Store/index.js +6 -8
  36. package/lib/Store/make-cache-manager-store.js +73 -81
  37. package/lib/Store/make-in-memory-store.js +286 -427
  38. package/lib/Store/make-ordered-dictionary.js +77 -79
  39. package/lib/Store/object-repository.js +24 -26
  40. package/lib/Types/Auth.js +3 -2
  41. package/lib/Types/Bussines.js +3 -0
  42. package/lib/Types/Call.js +3 -2
  43. package/lib/Types/Chat.js +9 -4
  44. package/lib/Types/Contact.js +3 -2
  45. package/lib/Types/Events.js +3 -2
  46. package/lib/Types/GroupMetadata.js +3 -2
  47. package/lib/Types/Label.js +24 -26
  48. package/lib/Types/LabelAssociation.js +6 -8
  49. package/lib/Types/Message.js +12 -7
  50. package/lib/Types/Newsletter.js +32 -17
  51. package/lib/Types/Product.js +3 -2
  52. package/lib/Types/Signal.js +3 -2
  53. package/lib/Types/Socket.js +4 -2
  54. package/lib/Types/State.js +11 -2
  55. package/lib/Types/USync.js +3 -2
  56. package/lib/Types/index.js +27 -41
  57. package/lib/Utils/auth-utils.js +211 -198
  58. package/lib/Utils/baileys-event-stream.js +42 -61
  59. package/lib/Utils/browser-utils.js +25 -0
  60. package/lib/Utils/business.js +213 -214
  61. package/lib/Utils/chat-utils.js +711 -689
  62. package/lib/Utils/crypto.js +112 -133
  63. package/lib/Utils/decode-wa-message.js +254 -186
  64. package/lib/Utils/event-buffer.js +510 -502
  65. package/lib/Utils/generics.js +318 -408
  66. package/lib/Utils/history.js +83 -90
  67. package/lib/Utils/index.js +21 -33
  68. package/lib/Utils/link-preview.js +71 -116
  69. package/lib/Utils/logger.js +5 -7
  70. package/lib/Utils/lt-hash.js +40 -46
  71. package/lib/Utils/make-mutex.js +34 -41
  72. package/lib/Utils/message-retry-manager.js +113 -0
  73. package/lib/Utils/messages-media.js +575 -671
  74. package/lib/Utils/messages.js +354 -462
  75. package/lib/Utils/noise-handler.js +138 -149
  76. package/lib/Utils/pre-key-manager.js +85 -0
  77. package/lib/Utils/process-message.js +323 -354
  78. package/lib/Utils/signal.js +148 -130
  79. package/lib/Utils/use-multi-file-auth-state.js +109 -91
  80. package/lib/Utils/validate-connection.js +183 -190
  81. package/lib/WABinary/constants.js +1298 -35
  82. package/lib/WABinary/decode.js +231 -256
  83. package/lib/WABinary/encode.js +207 -239
  84. package/lib/WABinary/generic-utils.js +119 -40
  85. package/lib/WABinary/index.js +7 -21
  86. package/lib/WABinary/jid-utils.js +88 -64
  87. package/lib/WABinary/types.js +3 -2
  88. package/lib/WAM/BinaryInfo.js +10 -12
  89. package/lib/WAM/constants.js +22851 -15348
  90. package/lib/WAM/encode.js +135 -136
  91. package/lib/WAM/index.js +5 -19
  92. package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -30
  93. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +49 -53
  94. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +27 -28
  95. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +36 -39
  96. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -50
  97. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +26 -20
  98. package/lib/WAUSync/Protocols/index.js +6 -20
  99. package/lib/WAUSync/USyncQuery.js +86 -85
  100. package/lib/WAUSync/USyncUser.js +23 -25
  101. package/lib/WAUSync/index.js +5 -19
  102. package/lib/index.js +18 -33
  103. package/package.json +52 -57
  104. package/README.md +0 -2
  105. package/WASignalGroup/GroupProtocol.js +0 -1697
  106. package/WASignalGroup/ciphertext_message.js +0 -16
  107. package/WASignalGroup/group_cipher.js +0 -120
  108. package/WASignalGroup/group_session_builder.js +0 -46
  109. package/WASignalGroup/index.js +0 -5
  110. package/WASignalGroup/keyhelper.js +0 -21
  111. package/WASignalGroup/protobufs.js +0 -3
  112. package/WASignalGroup/queue_job.js +0 -69
  113. package/WASignalGroup/sender_chain_key.js +0 -50
  114. package/WASignalGroup/sender_key_distribution_message.js +0 -78
  115. package/WASignalGroup/sender_key_message.js +0 -92
  116. package/WASignalGroup/sender_key_record.js +0 -56
  117. package/WASignalGroup/sender_key_state.js +0 -129
  118. package/WASignalGroup/sender_message_key.js +0 -39
  119. package/lib/Defaults/baileys-version.json +0 -3
  120. package/lib/Defaults/phonenumber-mcc.json +0 -223
  121. package/lib/Socket/Client/abstract-socket-client.js +0 -13
  122. package/lib/Socket/Client/mobile-socket-client.js +0 -65
  123. package/lib/Socket/Client/web-socket-client.js +0 -62
  124. package/lib/Socket/registration.js +0 -166
  125. package/lib/Socket/usync.js +0 -70
  126. package/lib/Store/make-mongo-store.js +0 -567
@@ -1,122 +1,123 @@
1
- "use strict";
2
- var __importDefault = (this && this.__importDefault) || function (mod) {
3
- return (mod && mod.__esModule) ? mod : { "default": mod };
1
+ //=======================================================//
2
+ import { makeLibSignalRepository } from "../Signal/libsignal.js";
3
+ import { Browsers } from "../Utils/browser-utils.js";
4
+ import { proto } from "../../WAProto/index.js";
5
+ import logger from "../Utils/logger.js";
6
+ //=======================================================//
7
+ export const version = [2, 3000, 1033846690]
8
+ export const DICT_VERSION = 3;
9
+ //=======================================================//
10
+ export const NOISE_WA_HEADER = Buffer.from([87, 65, 6, DICT_VERSION]);
11
+ export const CALL_VIDEO_PREFIX = "https://call.whatsapp.com/video/";
12
+ export const CALL_AUDIO_PREFIX = "https://call.whatsapp.com/voice/";
13
+ export const WA_ADV_HOSTED_ACCOUNT_SIG_PREFIX = Buffer.from([6, 5]);
14
+ export const WA_ADV_HOSTED_DEVICE_SIG_PREFIX = Buffer.from([6, 6]);
15
+ export const NOISE_MODE = "Noise_XX_25519_AESGCM_SHA256\0\0\0\0";
16
+ export const WA_ADV_ACCOUNT_SIG_PREFIX = Buffer.from([6, 0]);
17
+ export const WA_ADV_DEVICE_SIG_PREFIX = Buffer.from([6, 1]);
18
+ export const DEFAULT_ORIGIN = "https://web.whatsapp.com";
19
+ export const WA_DEFAULT_EPHEMERAL = 7 * 24 * 60 * 60;
20
+ export const UNAUTHORIZED_CODES = [401, 403, 419];
21
+ export const KEY_BUNDLE_TYPE = Buffer.from([5]);
22
+ export const PHONE_CONNECTION_CB = "CB:Pong";
23
+ export const DEF_CALLBACK_PREFIX = "CB:";
24
+ export const DEF_TAG_PREFIX = "TAG:";
25
+ //=======================================================//
26
+ export const URL_REGEX = /https:\/\/(?![^:@\/\s]+:[^:@\/\s]+@)[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}(:\d+)?(\/[^\s]*)?/g;
27
+ //=======================================================//
28
+ export const WA_CERT_DETAILS = {
29
+ SERIAL: 0
4
30
  };
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.DEFAULT_CACHE_TTLS = exports.INITIAL_PREKEY_COUNT = exports.MIN_PREKEY_COUNT = exports.MEDIA_KEYS = exports.MEDIA_HKDF_KEY_MAPPING = exports.MEDIA_PATH_MAP = exports.DEFAULT_CONNECTION_CONFIG = exports.PROCESSABLE_HISTORY_TYPES = exports.WA_CERT_DETAILS = exports.URL_REGEX = exports.MOBILE_NOISE_HEADER = exports.PROTOCOL_VERSION = exports.NOISE_WA_HEADER = exports.KEY_BUNDLE_TYPE = exports.DICT_VERSION = exports.NOISE_MODE = exports.REGISTRATION_PUBLIC_KEY = exports.MOBILE_USERAGENT = exports.MOBILE_REGISTRATION_ENDPOINT = exports.MOBILE_TOKEN = exports.WA_DEFAULT_EPHEMERAL = exports.PHONE_CONNECTION_CB = exports.DEF_TAG_PREFIX = exports.DEF_CALLBACK_PREFIX = exports.MOBILE_PORT = exports.MOBILE_ENDPOINT = exports.DEFAULT_ORIGIN = exports.PHONENUMBER_MCC = exports.UNAUTHORIZED_CODES = void 0;
7
- const crypto_1 = require("crypto");
8
- const WAProto_1 = require("../../WAProto");
9
- const libsignal_1 = require("../Signal/libsignal");
10
- const Utils_1 = require("../Utils");
11
- const logger_1 = __importDefault(require("../Utils/logger"));
12
- const baileys_version_json_1 = require("./baileys-version.json");
13
- const phonenumber_mcc_json_1 = __importDefault(require("./phonenumber-mcc.json"));
14
- exports.UNAUTHORIZED_CODES = [401, 403, 419];
15
- exports.PHONENUMBER_MCC = phonenumber_mcc_json_1.default;
16
- exports.DEFAULT_ORIGIN = 'https://web.whatsapp.com';
17
- exports.MOBILE_ENDPOINT = 'g.whatsapp.net';
18
- exports.MOBILE_PORT = 443;
19
- exports.DEF_CALLBACK_PREFIX = 'CB:';
20
- exports.DEF_TAG_PREFIX = 'TAG:';
21
- exports.PHONE_CONNECTION_CB = 'CB:Pong';
22
- exports.WA_DEFAULT_EPHEMERAL = 7 * 24 * 60 * 60;
23
- const WA_VERSION = '2.24.6.77';
24
- const WA_VERSION_HASH = (0, crypto_1.createHash)('md5').update(WA_VERSION).digest('hex');
25
- exports.MOBILE_TOKEN = Buffer.from('0a1mLfGUIBVrMKF1RdvLI5lkRBvof6vn0fD2QRSM' + WA_VERSION_HASH);
26
- exports.MOBILE_REGISTRATION_ENDPOINT = 'https://v.whatsapp.net/v2';
27
- exports.MOBILE_USERAGENT = `Telegram Android 11.7.0 Samsung Galaxy-A7-2017`;
28
- exports.REGISTRATION_PUBLIC_KEY = Buffer.from([
29
- 5, 142, 140, 15, 116, 195, 235, 197, 215, 166, 134, 92, 108, 60, 132, 56, 86, 176, 97, 33, 204, 232, 234, 119, 77,
30
- 34, 251, 111, 18, 37, 18, 48, 45,
31
- ]);
32
- exports.NOISE_MODE = 'Noise_XX_25519_AESGCM_SHA256\0\0\0\0';
33
- exports.DICT_VERSION = 2;
34
- exports.KEY_BUNDLE_TYPE = Buffer.from([5]);
35
- exports.NOISE_WA_HEADER = Buffer.from([87, 65, 6, exports.DICT_VERSION]); // last is "DICT_VERSION"
36
- exports.PROTOCOL_VERSION = [5, 2];
37
- exports.MOBILE_NOISE_HEADER = Buffer.concat([Buffer.from('WA'), Buffer.from(exports.PROTOCOL_VERSION)]);
38
- /** from: https://stackoverflow.com/questions/3809401/what-is-a-good-regular-expression-to-match-a-url */
39
- exports.URL_REGEX = /https:\/\/(?![^:@\/\s]+:[^:@\/\s]+@)[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}(:\d+)?(\/[^\s]*)?/g;
40
- exports.WA_CERT_DETAILS = {
41
- SERIAL: 0,
42
- };
43
- exports.PROCESSABLE_HISTORY_TYPES = [
44
- WAProto_1.proto.Message.HistorySyncNotification.HistorySyncType.INITIAL_BOOTSTRAP,
45
- WAProto_1.proto.Message.HistorySyncNotification.HistorySyncType.PUSH_NAME,
46
- WAProto_1.proto.Message.HistorySyncNotification.HistorySyncType.RECENT,
47
- WAProto_1.proto.Message.HistorySyncNotification.HistorySyncType.FULL,
48
- WAProto_1.proto.Message.HistorySyncNotification.HistorySyncType.ON_DEMAND,
31
+ //=======================================================//
32
+ export const PROCESSABLE_HISTORY_TYPES = [
33
+ proto.Message.HistorySyncNotification.HistorySyncType.INITIAL_BOOTSTRAP,
34
+ proto.Message.HistorySyncNotification.HistorySyncType.PUSH_NAME,
35
+ proto.Message.HistorySyncNotification.HistorySyncType.RECENT,
36
+ proto.Message.HistorySyncNotification.HistorySyncType.FULL,
37
+ proto.Message.HistorySyncNotification.HistorySyncType.ON_DEMAND,
38
+ proto.Message.HistorySyncNotification.HistorySyncType.NON_BLOCKING_DATA,
39
+ proto.Message.HistorySyncNotification.HistorySyncType.INITIAL_STATUS_V3
49
40
  ];
50
- exports.DEFAULT_CONNECTION_CONFIG = {
51
- version: baileys_version_json_1.version,
52
- browser: Utils_1.Browsers.ubuntu('Chrome'),
53
- waWebSocketUrl: 'https://web.whatsapp.com/ws/chat',
54
- connectTimeoutMs: 20000,
55
- keepAliveIntervalMs: 1000,
56
- logger: logger_1.default.child({ class: 'baileys' }),
57
- printQRInTerminal: false,
58
- emitOwnEvents: true,
59
- defaultQueryTimeoutMs: 60000,
60
- customUploadHosts: [],
61
- retryRequestDelayMs: 250,
62
- maxMsgRetryCount: 5,
63
- fireInitQueries: true,
64
- auth: undefined,
65
- markOnlineOnConnect: true,
66
- syncFullHistory: false,
67
- patchMessageBeforeSending: msg => msg,
68
- shouldSyncHistoryMessage: () => true,
69
- shouldIgnoreJid: () => false,
70
- linkPreviewImageThumbnailWidth: 192,
71
- transactionOpts: { maxCommitRetries: 10, delayBetweenTriesMs: 3000 },
72
- generateHighQualityLinkPreview: true,
73
- options: {},
74
- appStateMacVerification: {
75
- patch: false,
76
- snapshot: false,
77
- },
78
- countryCode: 'US',
79
- getMessage: async () => undefined,
80
- cachedGroupMetadata: async () => undefined,
81
- makeSignalRepository: libsignal_1.makeLibSignalRepository
41
+ //=======================================================//
42
+ export const DEFAULT_CONNECTION_CONFIG = {
43
+ "version": version,
44
+ "browser": Browsers.iOS("Safari"),
45
+ "waWebSocketUrl": "wss://web.whatsapp.com/ws/chat",
46
+ "connectTimeoutMs": 20000,
47
+ "keepAliveIntervalMs": 10000,
48
+ "logger": logger.child({ "class": "baileys" }),
49
+ "emitOwnEvents": true,
50
+ "defaultQueryTimeoutMs": 60000,
51
+ "customUploadHosts": [],
52
+ "retryRequestDelayMs": 250,
53
+ "maxMsgRetryCount": 5,
54
+ "fireInitQueries": true,
55
+ "auth": undefined,
56
+ "markOnlineOnConnect": true,
57
+ "syncFullHistory": true,
58
+ "patchMessageBeforeSending": msg => msg,
59
+ "shouldSyncHistoryMessage": () => true,
60
+ "shouldIgnoreJid": () => false,
61
+ "linkPreviewImageThumbnailWidth": 192,
62
+ "transactionOpts": { "maxCommitRetries": 10, "delayBetweenTriesMs": 3000 },
63
+ "generateHighQualityLinkPreview": false,
64
+ "enableAutoSessionRecreation": true,
65
+ "enableRecentMessageCache": true,
66
+ "options": {},
67
+ "appStateMacVerification": {
68
+ "patch": false,
69
+ "snapshot": false
70
+ },
71
+ "countryCode": "US",
72
+ "getMessage": async () => undefined,
73
+ "cachedGroupMetadata": async () => undefined,
74
+ "makeSignalRepository": makeLibSignalRepository
75
+ };
76
+ //=======================================================//
77
+ export const MEDIA_PATH_MAP = {
78
+ "image": "/mms/image",
79
+ "video": "/mms/video",
80
+ "document": "/mms/document",
81
+ "audio": "/mms/audio",
82
+ "sticker": "/mms/image",
83
+ "thumbnail-link": "/mms/image",
84
+ "product-catalog-image": "/product/image",
85
+ "md-app-state": "",
86
+ "md-msg-hist": "/mms/md-app-state",
87
+ "biz-cover-photo": "/pps/biz-cover-photo"
82
88
  };
83
- exports.MEDIA_PATH_MAP = {
84
- image: '/mms/image',
85
- video: '/mms/video',
86
- document: '/mms/document',
87
- audio: '/mms/audio',
88
- sticker: '/mms/image',
89
- 'thumbnail-link': '/mms/image',
90
- 'product-catalog-image': '/product/image',
91
- 'md-app-state': '',
92
- 'md-msg-hist': '/mms/md-app-state',
89
+ //=======================================================//
90
+ export const MEDIA_HKDF_KEY_MAPPING = {
91
+ "audio": "Audio",
92
+ "document": "Document",
93
+ "gif": "Video",
94
+ "image": "Image",
95
+ "ppic": "",
96
+ "product": "Image",
97
+ "ptt": "Audio",
98
+ "sticker": "Image",
99
+ "video": "Video",
100
+ "thumbnail-document": "Document Thumbnail",
101
+ "thumbnail-image": "Image Thumbnail",
102
+ "thumbnail-video": "Video Thumbnail",
103
+ "thumbnail-link": "Link Thumbnail",
104
+ "md-msg-hist": "History",
105
+ "md-app-state": "App State",
106
+ "product-catalog-image": "",
107
+ "payment-bg-image": "Payment Background",
108
+ "ptv": "Video",
109
+ "biz-cover-photo": "Image"
93
110
  };
94
- exports.MEDIA_HKDF_KEY_MAPPING = {
95
- 'audio': 'Audio',
96
- 'document': 'Document',
97
- 'gif': 'Video',
98
- 'image': 'Image',
99
- 'ppic': '',
100
- 'product': 'Image',
101
- 'ptt': 'Audio',
102
- 'sticker': 'Image',
103
- 'video': 'Video',
104
- 'thumbnail-document': 'Document Thumbnail',
105
- 'thumbnail-image': 'Image Thumbnail',
106
- 'thumbnail-video': 'Video Thumbnail',
107
- 'thumbnail-link': 'Link Thumbnail',
108
- 'md-msg-hist': 'History',
109
- 'md-app-state': 'App State',
110
- 'product-catalog-image': '',
111
- 'payment-bg-image': 'Payment Background',
112
- 'ptv': 'Video'
111
+ //=======================================================//
112
+ export const MEDIA_KEYS = Object.keys(MEDIA_PATH_MAP);
113
+ export const MIN_PREKEY_COUNT = 5;
114
+ export const INITIAL_PREKEY_COUNT = 812;
115
+ export const UPLOAD_TIMEOUT = 30000;
116
+ export const MIN_UPLOAD_INTERVAL = 5000;
117
+ export const DEFAULT_CACHE_TTLS = {
118
+ SIGNAL_STORE: 5 * 60,
119
+ MSG_RETRY: 60 * 60,
120
+ CALL_OFFER: 5 * 60,
121
+ USER_DEVICES: 5 * 60
113
122
  };
114
- exports.MEDIA_KEYS = Object.keys(exports.MEDIA_PATH_MAP);
115
- exports.MIN_PREKEY_COUNT = 2;
116
- exports.INITIAL_PREKEY_COUNT = 2;
117
- exports.DEFAULT_CACHE_TTLS = {
118
- SIGNAL_STORE: 5 * 60, // 5 minutes
119
- MSG_RETRY: 60 * 60, // 1 hour
120
- CALL_OFFER: 5 * 60, // 5 minutes
121
- USER_DEVICES: 5 * 60, // 5 minutes
122
- };
123
+ //=======================================================//
@@ -0,0 +1,20 @@
1
+ //===================================//
2
+ export default function binarySearch(array, predicate) {
3
+ let low = 0;
4
+ let high = array.length;
5
+ if (array.length === 0) return low;
6
+ if (predicate(array[low]) < 0) return low - 1;
7
+ else if (predicate(array[low]) === 0) return low;
8
+ const maxPred = predicate(array[high - 1]);
9
+ if (maxPred > 0) return high;
10
+ else if (maxPred === 0) return high - 1;
11
+ while (low !== high) {
12
+ const mid = low + Math.floor((high - low) / 2);
13
+ const pred = predicate(array[mid]);
14
+ if (pred < 0) high = mid;
15
+ else if (pred > 0) low = mid + 1;
16
+ else return mid;
17
+ }
18
+ return low;
19
+ }
20
+ //===================================//
@@ -0,0 +1,167 @@
1
+ //===================================//
2
+ import binarySearch from "./BinarySearch.js";
3
+ //===================================//
4
+ export default class KeyedDB {
5
+ constructor(key, id) {
6
+ this.key = key;
7
+ this.idGetter = id || (v => this.key.key(v).toString());
8
+ this.dict = {};
9
+ this.array = [];
10
+ }
11
+ get length() {
12
+ return this.array.length;
13
+ }
14
+ get first() {
15
+ return this.array[0];
16
+ }
17
+ get last() {
18
+ return this.array[this.array.length - 1];
19
+ }
20
+ toJSON() {
21
+ return this.array;
22
+ }
23
+ insert(...values) {
24
+ values.forEach(v => this._insertSingle(v));
25
+ }
26
+ upsert(...values) {
27
+ const updates = [];
28
+ values.forEach(v => {
29
+ if (!v) return;
30
+ const deleted = this.deleteById(this.idGetter(v), false);
31
+ this._insertSingle(v);
32
+ deleted && updates.push(v);
33
+ });
34
+ return updates;
35
+ }
36
+ insertIfAbsent(...values) {
37
+ const insertions = [];
38
+ values.forEach(v => {
39
+ if (!v) return;
40
+ const presentValue = this.get(this.idGetter(v));
41
+ if (presentValue) return;
42
+ const presentKey = this.firstIndex(v);
43
+ if (this.array[presentKey] && this.key.key(this.array[presentKey]) === this.key.key(v)) return;
44
+ this.insert(v);
45
+ insertions.push(v);
46
+ });
47
+ return insertions;
48
+ }
49
+ deleteById(id, assertPresent = true) {
50
+ const value = this.get(id);
51
+ if (!value) {
52
+ if (assertPresent) throw new Error(`Value not found`);
53
+ return;
54
+ }
55
+ return this.delete(value);
56
+ }
57
+ delete(value) {
58
+ const index = this.firstIndex(value);
59
+ if (index < 0 || index >= this.array.length || this.key.key(value) !== this.key.key(this.array[index])) {
60
+ return null;
61
+ }
62
+ delete this.dict[this.idGetter(value)];
63
+ return this.array.splice(index, 1)[0];
64
+ }
65
+ slice(start, end) {
66
+ const db = new KeyedDB(this.key, this.idGetter);
67
+ db.array = this.array.slice(start, end);
68
+ db.array.forEach(item => db.dict[this.idGetter(item)] = item);
69
+ return db;
70
+ }
71
+ clear() {
72
+ this.array = [];
73
+ this.dict = {};
74
+ }
75
+ get(id) {
76
+ return this.dict[id];
77
+ }
78
+ all() {
79
+ return this.array;
80
+ }
81
+ update(id, update) {
82
+ const value = this.get(id);
83
+ if (value) {
84
+ const idx = this.firstIndex(value);
85
+ if (idx >= 0 && idx < this.array.length && this.idGetter(this.array[idx]) === id) {
86
+ const oldKey = this.key.key(value);
87
+ update(value);
88
+ const newKey = this.key.key(value);
89
+ if (newKey !== oldKey) {
90
+ delete this.dict[id];
91
+ this.array.splice(idx, 1);
92
+ this._insertSingle(value);
93
+ return 2;
94
+ }
95
+ return 1;
96
+ }
97
+ }
98
+ }
99
+ updateKey(value, update) {
100
+ return this.update(this.idGetter(value), update);
101
+ }
102
+ filter(predicate) {
103
+ const db = new KeyedDB(this.key, this.idGetter);
104
+ db.array = this.array.filter((value, index) => {
105
+ if (predicate(value, index)) {
106
+ db.dict[this.idGetter(value)] = value;
107
+ return true;
108
+ }
109
+ });
110
+ return db;
111
+ }
112
+ paginatedByValue(value, limit, predicate, mode = "after") {
113
+ return this.paginated(value && this.key.key(value), limit, predicate, mode);
114
+ }
115
+ paginated(cursor, limit, predicate, mode = "after") {
116
+ let index = mode === "after" ? 0 : this.array.length;
117
+ if (cursor !== null && typeof cursor !== "undefined") {
118
+ index = binarySearch(this.array, v => this.key.compare(cursor, this.key.key(v)));
119
+ if (index < 0) index = 0;
120
+ if (this.key.key(this.array[index]) === cursor) index += (mode === "after" ? 1 : 0);
121
+ }
122
+ return this.filtered(index, limit, mode, predicate);
123
+ }
124
+ _insertSingle(value) {
125
+ if (!value) throw new Error("falsey value");
126
+ const valueID = this.idGetter(value);
127
+ if (this.get(valueID)) throw new Error("duplicate ID being inserted: " + valueID);
128
+
129
+ if (this.array.length > 0) {
130
+ const index = this.firstIndex(value);
131
+ if (index >= this.array.length) this.array.push(value);
132
+ else if (index < 0) this.array.unshift(value);
133
+ else if (this.key.key(value) !== this.key.key(this.array[index])) this.array.splice(index, 0, value);
134
+ else throw new Error(`duplicate key: ${this.key.key(value)}, inserting: ${valueID}, present: ${this.idGetter(this.array[index])}`);
135
+ } else {
136
+ this.array.push(value);
137
+ }
138
+ this.dict[valueID] = value;
139
+ }
140
+ filtered(start, count, mode, predicate) {
141
+ let arr;
142
+ if (mode === "after") {
143
+ if (predicate) {
144
+ arr = [];
145
+ for (let item of this.array.slice(start)) {
146
+ predicate(item, start + arr.length) && arr.push(item);
147
+ if (arr.length >= count) break;
148
+ }
149
+ } else arr = this.array.slice(start, start + count);
150
+ } else if (mode === "before") {
151
+ if (predicate) {
152
+ arr = [];
153
+ for (let i = start - 1; i >= 0; i--) {
154
+ let item = this.array[i];
155
+ predicate(item, start + arr.length) && arr.unshift(item);
156
+ if (arr.length >= count) break;
157
+ }
158
+ } else arr = this.array.slice(Math.max(start - count, 0), start);
159
+ }
160
+ return arr;
161
+ }
162
+ firstIndex(value) {
163
+ const valueKey = this.key.key(value);
164
+ return binarySearch(this.array, v => this.key.compare(valueKey, this.key.key(v)));
165
+ }
166
+ }
167
+ //===================================//
@@ -0,0 +1,4 @@
1
+ //===================================//
2
+ export * from "./BinarySearch.js"
3
+ export * from "./KeyedDB.js"
4
+ //===================================//
@@ -0,0 +1,13 @@
1
+ //=======================================================//
2
+ export class CiphertextMessage {
3
+ constructor() {
4
+ this.UNSUPPORTED_VERSION = 1;
5
+ this.CURRENT_VERSION = 3;
6
+ this.WHISPER_TYPE = 2;
7
+ this.PREKEY_TYPE = 3;
8
+ this.SENDERKEY_TYPE = 4;
9
+ this.SENDERKEY_DISTRIBUTION_TYPE = 5;
10
+ this.ENCRYPTED_MESSAGE_OVERHEAD = 53;
11
+ }
12
+ }
13
+ //=======================================================//
@@ -0,0 +1,32 @@
1
+ //=======================================================//
2
+ import { SenderKeyDistributionMessage } from "./sender-key-distribution-message.js";
3
+ import { SenderKeyRecord } from "./sender-key-record.js";
4
+ import { SenderKeyName } from "./sender-key-name.js";
5
+ import * as keyhelper from "./keyhelper.js";
6
+ //=======================================================//
7
+ export class GroupSessionBuilder {
8
+ constructor(senderKeyStore) {
9
+ this.senderKeyStore = senderKeyStore;
10
+ }
11
+ async process(senderKeyName, senderKeyDistributionMessage) {
12
+ const senderKeyRecord = await this.senderKeyStore.loadSenderKey(senderKeyName);
13
+ senderKeyRecord.addSenderKeyState(senderKeyDistributionMessage.getId(), senderKeyDistributionMessage.getIteration(), senderKeyDistributionMessage.getChainKey(), senderKeyDistributionMessage.getSignatureKey());
14
+ await this.senderKeyStore.storeSenderKey(senderKeyName, senderKeyRecord);
15
+ }
16
+ async create(senderKeyName) {
17
+ const senderKeyRecord = await this.senderKeyStore.loadSenderKey(senderKeyName);
18
+ if (senderKeyRecord.isEmpty()) {
19
+ const keyId = keyhelper.generateSenderKeyId();
20
+ const senderKey = keyhelper.generateSenderKey();
21
+ const signingKey = keyhelper.generateSenderSigningKey();
22
+ senderKeyRecord.setSenderKeyState(keyId, 0, senderKey, signingKey);
23
+ await this.senderKeyStore.storeSenderKey(senderKeyName, senderKeyRecord);
24
+ }
25
+ const state = senderKeyRecord.getSenderKeyState();
26
+ if (!state) {
27
+ throw new Error("No session state available");
28
+ }
29
+ return new SenderKeyDistributionMessage(state.getKeyId(), state.getSenderChainKey().getIteration(), state.getSenderChainKey().getSeed(), state.getSigningKeyPublic());
30
+ }
31
+ }
32
+ //# sourceMappingURL=group-session-builder.js.map
@@ -0,0 +1,84 @@
1
+ //=======================================================//
2
+ import { decrypt, encrypt } from "@dyyxyzz/libsignal-dyysilence/src/crypto.js";
3
+ import { SenderKeyMessage } from "./sender-key-message.js";
4
+ import { SenderKeyRecord } from "./sender-key-record.js";
5
+ import { SenderKeyState } from "./sender-key-state.js";
6
+ import { SenderKeyName } from "./sender-key-name.js";
7
+ //=======================================================//
8
+ export class GroupCipher {
9
+ constructor(senderKeyStore, senderKeyName) {
10
+ this.senderKeyStore = senderKeyStore;
11
+ this.senderKeyName = senderKeyName;
12
+ }
13
+ async encrypt(paddedPlaintext) {
14
+ const record = await this.senderKeyStore.loadSenderKey(this.senderKeyName);
15
+ if (!record) {
16
+ throw new Error("No SenderKeyRecord found for encryption");
17
+ }
18
+ const senderKeyState = record.getSenderKeyState();
19
+ if (!senderKeyState) {
20
+ throw new Error("No session to encrypt message");
21
+ }
22
+ const iteration = senderKeyState.getSenderChainKey().getIteration();
23
+ const senderKey = this.getSenderKey(senderKeyState, iteration === 0 ? 0 : iteration + 1);
24
+ const ciphertext = await this.getCipherText(senderKey.getIv(), senderKey.getCipherKey(), paddedPlaintext);
25
+ const senderKeyMessage = new SenderKeyMessage(senderKeyState.getKeyId(), senderKey.getIteration(), ciphertext, senderKeyState.getSigningKeyPrivate());
26
+ await this.senderKeyStore.storeSenderKey(this.senderKeyName, record);
27
+ return senderKeyMessage.serialize();
28
+ }
29
+ async decrypt(senderKeyMessageBytes) {
30
+ const record = await this.senderKeyStore.loadSenderKey(this.senderKeyName);
31
+ if (!record) {
32
+ throw new Error("No SenderKeyRecord found for decryption");
33
+ }
34
+ const senderKeyMessage = new SenderKeyMessage(null, null, null, null, senderKeyMessageBytes);
35
+ const senderKeyState = record.getSenderKeyState(senderKeyMessage.getKeyId());
36
+ if (!senderKeyState) {
37
+ throw new Error("No session found to decrypt message");
38
+ }
39
+ senderKeyMessage.verifySignature(senderKeyState.getSigningKeyPublic());
40
+ const senderKey = this.getSenderKey(senderKeyState, senderKeyMessage.getIteration());
41
+ const plaintext = await this.getPlainText(senderKey.getIv(), senderKey.getCipherKey(), senderKeyMessage.getCipherText());
42
+ await this.senderKeyStore.storeSenderKey(this.senderKeyName, record);
43
+ return plaintext;
44
+ }
45
+ getSenderKey(senderKeyState, iteration) {
46
+ let senderChainKey = senderKeyState.getSenderChainKey();
47
+ if (senderChainKey.getIteration() > iteration) {
48
+ if (senderKeyState.hasSenderMessageKey(iteration)) {
49
+ const messageKey = senderKeyState.removeSenderMessageKey(iteration);
50
+ if (!messageKey) {
51
+ throw new Error("No sender message key found for iteration");
52
+ }
53
+ return messageKey;
54
+ }
55
+ throw new Error(`Received message with old counter: ${senderChainKey.getIteration()}, ${iteration}`);
56
+ }
57
+ if (iteration - senderChainKey.getIteration() > 2000) {
58
+ throw new Error("Over 2000 messages into the future!");
59
+ }
60
+ while (senderChainKey.getIteration() < iteration) {
61
+ senderKeyState.addSenderMessageKey(senderChainKey.getSenderMessageKey());
62
+ senderChainKey = senderChainKey.getNext();
63
+ }
64
+ senderKeyState.setSenderChainKey(senderChainKey.getNext());
65
+ return senderChainKey.getSenderMessageKey();
66
+ }
67
+ async getPlainText(iv, key, ciphertext) {
68
+ try {
69
+ return decrypt(key, ciphertext, iv);
70
+ }
71
+ catch (e) {
72
+ throw new Error("InvalidMessageException");
73
+ }
74
+ }
75
+ async getCipherText(iv, key, plaintext) {
76
+ try {
77
+ return encrypt(key, plaintext, iv);
78
+ }
79
+ catch (e) {
80
+ throw new Error("InvalidMessageException");
81
+ }
82
+ }
83
+ }
84
+ //=======================================================//
@@ -0,0 +1,13 @@
1
+ //=======================================================//
2
+ export { SenderKeyDistributionMessage } from "./sender-key-distribution-message.js";
3
+ export { GroupSessionBuilder } from "./group-session-builder.js";
4
+ export { CiphertextMessage } from "./ciphertext-message.js";
5
+ export { SenderKeyMessage } from "./sender-key-message.js";
6
+ export { SenderMessageKey } from "./sender-message-key.js";
7
+ export { SenderKeyRecord } from "./sender-key-record.js";
8
+ export { SenderKeyState } from "./sender-key-state.js";
9
+ export { SenderChainKey } from "./sender-chain-key.js";
10
+ export { SenderKeyName } from "./sender-key-name.js";
11
+ export { GroupCipher } from "./group_cipher.js";
12
+ export * as keyhelper from "./keyhelper.js";
13
+ //=======================================================//
@@ -0,0 +1,20 @@
1
+ //=======================================================//
2
+ import { generateKeyPair } from "@dyyxyzz/libsignal-dyysilence/src/curve.js";
3
+ import * as nodeCrypto from "crypto";
4
+ //=======================================================//
5
+ export function generateSenderKey() {
6
+ return nodeCrypto.randomBytes(32);
7
+ }
8
+ export function generateSenderKeyId() {
9
+ return nodeCrypto.randomInt(2147483647);
10
+ }
11
+ export function generateSenderSigningKey(key) {
12
+ if (!key) {
13
+ key = generateKeyPair();
14
+ }
15
+ return {
16
+ public: Buffer.from(key.pubKey),
17
+ private: Buffer.from(key.privKey)
18
+ };
19
+ }
20
+ //=======================================================//
@@ -0,0 +1,28 @@
1
+ //=======================================================//
2
+ import { calculateMAC } from "@dyyxyzz/libsignal-dyysilence/src/crypto.js";
3
+ import { SenderMessageKey } from "./sender-message-key.js";
4
+ //=======================================================//
5
+ export class SenderChainKey {
6
+ constructor(iteration, chainKey) {
7
+ this.MESSAGE_KEY_SEED = Buffer.from([0x01]);
8
+ this.CHAIN_KEY_SEED = Buffer.from([0x02]);
9
+ this.iteration = iteration;
10
+ this.chainKey = Buffer.from(chainKey);
11
+ }
12
+ getIteration() {
13
+ return this.iteration;
14
+ }
15
+ getSenderMessageKey() {
16
+ return new SenderMessageKey(this.iteration, this.getDerivative(this.MESSAGE_KEY_SEED, this.chainKey));
17
+ }
18
+ getNext() {
19
+ return new SenderChainKey(this.iteration + 1, this.getDerivative(this.CHAIN_KEY_SEED, this.chainKey));
20
+ }
21
+ getSeed() {
22
+ return this.chainKey;
23
+ }
24
+ getDerivative(seed, key) {
25
+ return calculateMAC(key, seed);
26
+ }
27
+ }
28
+ //=======================================================//