@nexustechpro/baileys 2.0.1 → 2.0.5
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/LICENSE +1 -1
- package/README.md +924 -1299
- package/lib/Defaults/baileys-version.json +6 -2
- package/lib/Defaults/index.js +172 -172
- package/lib/Signal/libsignal.js +380 -292
- package/lib/Signal/lid-mapping.js +264 -171
- package/lib/Socket/Client/index.js +2 -2
- package/lib/Socket/Client/types.js +10 -10
- package/lib/Socket/Client/websocket.js +45 -310
- package/lib/Socket/business.js +375 -375
- package/lib/Socket/chats.js +909 -963
- package/lib/Socket/communities.js +430 -430
- package/lib/Socket/groups.js +342 -342
- package/lib/Socket/index.js +22 -22
- package/lib/Socket/messages-recv.js +777 -743
- package/lib/Socket/messages-send.js +667 -393
- package/lib/Socket/mex.js +50 -50
- package/lib/Socket/newsletter.js +148 -148
- package/lib/Socket/nexus-handler.js +75 -261
- package/lib/Socket/socket.js +709 -1201
- package/lib/Store/index.js +5 -5
- package/lib/Store/make-cache-manager-store.js +81 -81
- package/lib/Store/make-in-memory-store.js +416 -416
- package/lib/Store/make-ordered-dictionary.js +81 -81
- package/lib/Store/object-repository.js +30 -30
- package/lib/Types/Auth.js +1 -1
- package/lib/Types/Bussines.js +1 -1
- package/lib/Types/Call.js +1 -1
- package/lib/Types/Chat.js +7 -7
- package/lib/Types/Contact.js +1 -1
- package/lib/Types/Events.js +1 -1
- package/lib/Types/GroupMetadata.js +1 -1
- package/lib/Types/Label.js +24 -24
- package/lib/Types/LabelAssociation.js +6 -6
- package/lib/Types/Message.js +10 -10
- package/lib/Types/Newsletter.js +28 -28
- package/lib/Types/Product.js +1 -1
- package/lib/Types/Signal.js +1 -1
- package/lib/Types/Socket.js +2 -2
- package/lib/Types/State.js +12 -12
- package/lib/Types/USync.js +1 -1
- package/lib/Types/index.js +25 -25
- package/lib/Utils/auth-utils.js +264 -256
- package/lib/Utils/baileys-event-stream.js +55 -55
- package/lib/Utils/browser-utils.js +27 -27
- package/lib/Utils/business.js +228 -230
- package/lib/Utils/chat-utils.js +694 -764
- package/lib/Utils/crypto.js +109 -135
- package/lib/Utils/decode-wa-message.js +310 -314
- package/lib/Utils/event-buffer.js +547 -547
- package/lib/Utils/generics.js +297 -297
- package/lib/Utils/history.js +91 -83
- package/lib/Utils/index.js +21 -20
- package/lib/Utils/key-store.js +17 -0
- package/lib/Utils/link-preview.js +97 -88
- package/lib/Utils/logger.js +2 -2
- package/lib/Utils/lt-hash.js +47 -47
- package/lib/Utils/make-mutex.js +39 -39
- package/lib/Utils/message-retry-manager.js +148 -148
- package/lib/Utils/messages-media.js +534 -532
- package/lib/Utils/messages.js +705 -705
- package/lib/Utils/noise-handler.js +255 -255
- package/lib/Utils/pre-key-manager.js +105 -105
- package/lib/Utils/process-message.js +412 -412
- package/lib/Utils/signal.js +160 -158
- package/lib/Utils/use-multi-file-auth-state.js +120 -120
- package/lib/Utils/validate-connection.js +194 -194
- package/lib/WABinary/constants.js +1300 -1300
- package/lib/WABinary/decode.js +237 -237
- package/lib/WABinary/encode.js +232 -232
- package/lib/WABinary/generic-utils.js +252 -211
- package/lib/WABinary/index.js +5 -5
- package/lib/WABinary/jid-utils.js +279 -95
- package/lib/WABinary/types.js +1 -1
- package/lib/WAM/BinaryInfo.js +9 -9
- package/lib/WAM/constants.js +22852 -22852
- package/lib/WAM/encode.js +149 -149
- package/lib/WAM/index.js +3 -3
- package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -28
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -53
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +26 -26
- package/lib/WAUSync/Protocols/USyncStatusProtocol.js +37 -37
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -50
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +28 -28
- package/lib/WAUSync/Protocols/index.js +4 -4
- package/lib/WAUSync/USyncQuery.js +93 -93
- package/lib/WAUSync/USyncUser.js +22 -22
- package/lib/WAUSync/index.js +3 -3
- package/lib/index.js +66 -66
- package/package.json +171 -144
- package/lib/Signal/Group/ciphertext-message.js +0 -12
- package/lib/Signal/Group/group-session-builder.js +0 -30
- package/lib/Signal/Group/group_cipher.js +0 -100
- package/lib/Signal/Group/index.js +0 -12
- package/lib/Signal/Group/keyhelper.js +0 -18
- package/lib/Signal/Group/sender-chain-key.js +0 -26
- package/lib/Signal/Group/sender-key-distribution-message.js +0 -63
- package/lib/Signal/Group/sender-key-message.js +0 -66
- package/lib/Signal/Group/sender-key-name.js +0 -48
- package/lib/Signal/Group/sender-key-record.js +0 -41
- package/lib/Signal/Group/sender-key-state.js +0 -84
- 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
|
};
|
package/lib/Socket/newsletter.js
CHANGED
|
@@ -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
|
}
|