sagor-fca 0.0.9 → 0.0.11

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 (71) hide show
  1. package/func/checkUpdate.js +219 -10
  2. package/func/logAdapter.js +19 -16
  3. package/func/logger.js +29 -47
  4. package/index.js +5 -14
  5. package/module/config.js +22 -20
  6. package/module/login.js +100 -78
  7. package/module/loginHelper.js +3 -3
  8. package/module/options.js +33 -26
  9. package/package.json +10 -5
  10. package/src/api/action/addExternalModule.js +16 -10
  11. package/src/api/action/changeBio.js +45 -18
  12. package/src/api/action/enableAutoSaveAppState.js +53 -52
  13. package/src/api/action/getCurrentUserID.js +0 -4
  14. package/src/api/action/refreshFb_dtsg.js +37 -51
  15. package/src/api/http/httpGet.js +33 -14
  16. package/src/api/http/httpPost.js +37 -16
  17. package/src/api/http/postFormData.js +39 -14
  18. package/src/api/messaging/sendMessage.js +1 -1
  19. package/src/api/socket/core/connectMqtt.js +174 -181
  20. package/src/api/socket/core/emitAuth.js +74 -50
  21. package/src/api/socket/core/getSeqID.js +267 -115
  22. package/src/api/socket/core/getTaskResponseData.js +7 -9
  23. package/src/api/socket/detail/constants.js +4 -5
  24. package/src/api/socket/listenMqtt.js +1 -1
  25. package/src/api/socket/middleware/index.js +181 -45
  26. package/src/api/threads/getThreadHistory.js +3 -3
  27. package/src/api/threads/getThreadInfo.js +1 -1
  28. package/src/api/threads/getThreadPictures.js +0 -4
  29. package/src/api/users/getUserID.js +46 -29
  30. package/src/api/users/getUserInfo.js +2 -2
  31. package/src/core/sendReqMqtt.js +56 -29
  32. package/src/database/models/index.js +1 -1
  33. package/src/database/models/thread.js +0 -4
  34. package/src/database/models/user.js +0 -4
  35. package/src/remote/remoteClient.js +88 -36
  36. package/src/utils/broadcast.js +34 -44
  37. package/src/utils/client.js +7 -7
  38. package/src/utils/constants.js +5 -35
  39. package/src/utils/cookies.js +49 -54
  40. package/src/utils/headers.js +96 -75
  41. package/src/utils/request/client.js +10 -16
  42. package/src/utils/request/config.js +11 -13
  43. package/src/utils/request/defaults.js +26 -20
  44. package/src/utils/request/helpers.js +27 -12
  45. package/src/utils/request/index.js +13 -8
  46. package/src/utils/request/methods.js +91 -20
  47. package/src/utils/request/proxy.js +8 -10
  48. package/src/utils/request/retry.js +41 -51
  49. package/src/utils/request/sanitize.js +25 -17
  50. package/README.md +0 -223
  51. package/func/utils.js +0 -5
  52. package/src/api/feed/commentOnPost.js +0 -120
  53. package/src/api/feed/getPostInfo.js +0 -98
  54. package/src/api/groups/getGroupJoinRequests.js +0 -126
  55. package/src/api/groups/handleGroupJoinRequest.js +0 -134
  56. package/src/api/live/getLiveStreamInfo.js +0 -35
  57. package/src/api/live/reactLiveStream.js +0 -31
  58. package/src/api/marketplace/getMarketplaceItemDetail.js +0 -35
  59. package/src/api/marketplace/getMarketplaceItems.js +0 -30
  60. package/src/api/messaging/notes.js +0 -98
  61. package/src/api/messaging/sendVoiceMessage.js +0 -173
  62. package/src/api/privacy/getPrivacySettings.js +0 -32
  63. package/src/api/privacy/setPrivacySettings.js +0 -31
  64. package/src/api/professional/getProfessionalInfo.js +0 -35
  65. package/src/api/professional/toggleProfessionalMode.js +0 -31
  66. package/src/api/reels/getReels.js +0 -30
  67. package/src/api/reels/reactReel.js +0 -31
  68. package/src/utils/format/s +0 -0
  69. package/src/utils/loginParser/s +0 -0
  70. package/src/utils/sessionKeeper.js +0 -275
  71. package/src/utils/userAgent.js +0 -28
@@ -1,13 +1,222 @@
1
- // ============================================================
2
- // SAGOR-FCA v2.0 Check Update (Suspended)
3
- // © 2025 Sagor. All Rights Reserved.
4
- // autoUpdate Disabled Does not slow down takeoff Render
5
- // ============================================================
6
- "use strict";
7
-
8
- async function checkAndUpdateVersion() {
9
- // ✅ Permanently disabled — does not check for updates
10
- return Promise.resolve();
1
+ const logger = require("./logger");
2
+ const fs = require("fs");
3
+ const path = require("path");
4
+ const { exec } = require("child_process");
5
+ const https = require("https");
6
+ const pkgName = "@dongdev/fca-unofficial";
7
+
8
+ let axios = null;
9
+ try {
10
+ axios = require("axios");
11
+ } catch (e) {
12
+ }
13
+
14
+ const TEMP_DIR = path.join(process.cwd(), "temp");
15
+ const LOCK_FILE = path.join(TEMP_DIR, ".fca-update-lock.json");
16
+ const RESTART_COOLDOWN_MS = 10 * 60 * 1000;
17
+
18
+ function execPromise(cmd) {
19
+ return new Promise((resolve, reject) => {
20
+ exec(cmd, { cwd: process.cwd() }, (error, stdout, stderr) => {
21
+ if (error) return reject({ error, stderr });
22
+ resolve({ stdout, stderr });
23
+ });
24
+ });
25
+ }
26
+
27
+ function ensureTemp() {
28
+ try { fs.mkdirSync(TEMP_DIR, { recursive: true }); } catch { }
29
+ }
30
+
31
+ function readLock() {
32
+ try { return JSON.parse(fs.readFileSync(LOCK_FILE, "utf8")); } catch { return null; }
33
+ }
34
+
35
+ function writeLock(data) {
36
+ ensureTemp();
37
+ try { fs.writeFileSync(LOCK_FILE, JSON.stringify(data)); } catch { }
38
+ }
39
+
40
+ function clearLock() {
41
+ try { fs.unlinkSync(LOCK_FILE); } catch { }
42
+ }
43
+
44
+ function getInstalledVersion() {
45
+ try {
46
+ const p = require.resolve(`${pkgName}/package.json`, { paths: [process.cwd(), __dirname] });
47
+ return JSON.parse(fs.readFileSync(p, "utf8")).version;
48
+ } catch {
49
+ return null;
50
+ }
51
+ }
52
+
53
+ async function getInstalledVersionByNpm() {
54
+ try {
55
+ const { stdout } = await execPromise(`npm ls ${pkgName} --json --depth=0`);
56
+ const json = JSON.parse(stdout || "{}");
57
+ const v = (json && json.dependencies && json.dependencies[pkgName] && json.dependencies[pkgName].version) || null;
58
+ return v;
59
+ } catch {
60
+ return null;
61
+ }
62
+ }
63
+
64
+ async function getLatestVersionFromNpmRegistry() {
65
+ const url = `https://registry.npmjs.org/${pkgName}/latest`;
66
+
67
+ if (axios) {
68
+ try {
69
+ const response = await axios.get(url, {
70
+ timeout: 10000,
71
+ headers: {
72
+ 'User-Agent': 'fca-unofficial-updater',
73
+ 'Accept': 'application/json'
74
+ }
75
+ });
76
+ if (response && response.data && response.data.version) {
77
+ return response.data.version;
78
+ }
79
+ throw new Error("Invalid response from npm registry");
80
+ } catch (error) {
81
+ if (error.code === 'ECONNABORTED' || error.code === 'ETIMEDOUT') {
82
+ throw new Error("Request timeout");
83
+ }
84
+ throw error;
85
+ }
86
+ }
87
+ return new Promise((resolve, reject) => {
88
+ const timeout = setTimeout(() => {
89
+ reject(new Error("Request timeout"));
90
+ }, 10000);
91
+
92
+ https.get(url, {
93
+ headers: {
94
+ 'User-Agent': 'fca-unofficial-updater',
95
+ 'Accept': 'application/json'
96
+ }
97
+ }, (res) => {
98
+ let data = "";
99
+ res.on("data", (chunk) => { data += chunk; });
100
+ res.on("end", () => {
101
+ clearTimeout(timeout);
102
+ try {
103
+ const json = JSON.parse(data);
104
+ if (json && json.version) {
105
+ resolve(json.version);
106
+ } else {
107
+ reject(new Error("Invalid response from npm registry"));
108
+ }
109
+ } catch (e) {
110
+ reject(e);
111
+ }
112
+ });
113
+ }).on("error", (err) => {
114
+ clearTimeout(timeout);
115
+ reject(err);
116
+ });
117
+ });
118
+ }
119
+
120
+ async function getLatestVersion() {
121
+ const nodeVersion = process.version;
122
+ const nodeMajor = parseInt(nodeVersion.split('.')[0].substring(1), 10);
123
+
124
+ const shouldUseNpmCommand = nodeMajor >= 12 && nodeMajor < 20;
125
+
126
+ if (shouldUseNpmCommand) {
127
+ try {
128
+ const { stdout } = await execPromise(`npm view ${pkgName} version`);
129
+ const version = stdout.trim();
130
+ if (version && version.length > 0) {
131
+ return version;
132
+ }
133
+ } catch (npmError) {
134
+ const errorMsg = npmError.error && npmError.error.message ? npmError.error.message : String(npmError);
135
+ if (errorMsg.includes("npm") && errorMsg.includes("not to run")) {
136
+ logger("npm version incompatible, using registry API instead", "warn");
137
+ } else {
138
+ logger("npm view failed, using registry API instead", "warn");
139
+ }
140
+ }
141
+ } else {
142
+ logger("Using npm registry API (bypassing npm command)", "info");
143
+ }
144
+ try {
145
+ const version = await getLatestVersionFromNpmRegistry();
146
+ return version;
147
+ } catch (httpError) {
148
+ const errorMsg = httpError && httpError.message ? httpError.message : String(httpError);
149
+ logger(`Failed to get latest version: ${errorMsg}`, "error");
150
+ throw new Error("Cannot check for updates: npm command failed and registry API unavailable");
151
+ }
152
+ }
153
+
154
+ async function _checkAndUpdateVersionImpl() {
155
+ const lock = readLock();
156
+ if (lock && Date.now() - (lock.ts || 0) < RESTART_COOLDOWN_MS) {
157
+ logger("Skip auto-update due to recent attempt", "info");
158
+ return;
159
+ }
160
+
161
+ logger("Checking version...", "info");
162
+ let latest;
163
+ try {
164
+ latest = await getLatestVersion();
165
+ } catch (error) {
166
+ logger(`Cannot check for updates: ${error.message || error}. Skipping version check.`, "warn");
167
+ return;
168
+ }
169
+
170
+ let installed = getInstalledVersion();
171
+ if (!installed) installed = await getInstalledVersionByNpm();
172
+
173
+ if (installed && installed === latest) {
174
+ clearLock();
175
+ logger(`You're already on the latest version - ${latest}`, "info");
176
+ return;
177
+ }
178
+
179
+ if (lock && lock.latest === latest && Date.now() - (lock.ts || 0) < RESTART_COOLDOWN_MS) {
180
+ logger("Update already attempted recently, skipping restart loop", "info");
181
+ return;
182
+ }
183
+
184
+ logger(`New version available (${latest}). Current version (${installed || "not installed"}). Updating...`, "info");
185
+
186
+ try {
187
+ const { stderr } = await execPromise(`npm i ${pkgName}@latest`);
188
+ if (stderr) logger(stderr, "error");
189
+ } catch (e) {
190
+ logger(`Error running npm install: ${e.error || e}. Trying to install from GitHub...`, "error");
191
+ try {
192
+ const { stderr } = await execPromise("npm i https://github.com/Donix-VN/fca-unofficial");
193
+ if (stderr) logger(stderr, "error");
194
+ } catch (gitErr) {
195
+ writeLock({ ts: Date.now(), latest, status: "failed" });
196
+ logger(`Error installing from GitHub: ${gitErr.error || gitErr}`, "error");
197
+ throw (gitErr.error || gitErr);
198
+ }
199
+ }
200
+
201
+ let after = getInstalledVersion();
202
+ if (!after) after = await getInstalledVersionByNpm();
203
+
204
+ if (after && after === latest) {
205
+ writeLock({ ts: Date.now(), latest, status: "updated" });
206
+ logger(`Updated fca to the latest version: ${latest}, Restart to apply`, "info");
207
+ process.exit(1);
208
+ } else {
209
+ writeLock({ ts: Date.now(), latest, status: "mismatch" });
210
+ logger(`Installed but version mismatch (have: ${after || "unknown"}, want: ${latest}). Skip restart to avoid loop`, "error");
211
+ }
212
+ }
213
+
214
+ function checkAndUpdateVersion(callback) {
215
+ if (typeof callback === "function") {
216
+ _checkAndUpdateVersionImpl().then(() => callback(null)).catch(err => callback(err));
217
+ return;
218
+ }
219
+ return _checkAndUpdateVersionImpl();
11
220
  }
12
221
 
13
222
  module.exports = { checkAndUpdateVersion };
@@ -1,30 +1,33 @@
1
- // ============================================================
2
- // SAGOR-FCA v2.0 — Log Adapter
3
- // © 2025 Sagor. All Rights Reserved.
4
- // ============================================================
5
- "use strict";
6
-
7
1
  const logger = require("./logger");
8
2
 
9
3
  function formatArgs(args) {
10
4
  const [prefix, msg] = args;
5
+
6
+ // Single argument: log as-is
11
7
  if (msg === undefined) {
12
- if (prefix instanceof Error) return prefix.stack || prefix.message || String(prefix);
13
- return String(prefix ?? "");
8
+ if (prefix instanceof Error) {
9
+ return prefix.stack || prefix.message || String(prefix);
10
+ }
11
+ return String(prefix);
14
12
  }
15
- const tag = prefix == null ? "" : String(prefix);
16
- const text = msg instanceof Error
17
- ? (msg.message || String(msg))
18
- : (msg == null ? "" : String(msg));
13
+
14
+ // Two arguments: mimic npmlog("tag", message)
15
+ const tag = prefix == null ? "" : String(prefix);
16
+ if (msg instanceof Error) {
17
+ const base = msg.message || String(msg);
18
+ return tag ? `${tag}: ${base}` : base;
19
+ }
20
+ const text = msg == null ? "" : String(msg);
19
21
  return tag ? `${tag}: ${text}` : text;
20
22
  }
21
23
 
22
24
  const log = {
23
- info: (...args) => logger(formatArgs(args), "info"),
24
- warn: (...args) => logger(formatArgs(args), "warn"),
25
- error: (...args) => logger(formatArgs(args), "error"),
25
+ info: (...args) => logger(formatArgs(args), "info"),
26
+ warn: (...args) => logger(formatArgs(args), "warn"),
27
+ error: (...args) => logger(formatArgs(args), "error"),
26
28
  verbose: (...args) => logger(formatArgs(args), "info"),
27
- silly: (...args) => logger(formatArgs(args), "info"),
29
+ silly: (...args) => logger(formatArgs(args), "info")
28
30
  };
29
31
 
30
32
  module.exports = log;
33
+
package/func/logger.js CHANGED
@@ -1,66 +1,48 @@
1
- // ============================================================
2
- // SAGOR-FCA v2.0 — Logger
3
- // © 2025 Sagor. All Rights Reserved.
4
- // All rights reserved to Sagor
5
- // ============================================================
6
- "use strict";
7
-
8
- const chalk = require("chalk");
1
+ const chalk = require("chalk");
9
2
  const gradient = require("gradient-string");
10
3
 
11
- const THEMES = [
12
- "blue","dream2","dream","fiery","rainbow","pastel",
13
- "cristal","red","aqua","pink","retro","sunlight",
14
- "teen","summer","flower","ghost","hacker"
4
+ const themes = [
5
+ "blue", "dream2", "dream", "fiery", "rainbow", "pastel", "cristal", "red", "aqua", "pink", "retro", "sunlight", "teen", "summer", "flower", "ghost", "hacker"
15
6
  ];
16
7
 
17
8
  function buildGradient(name) {
18
9
  const t = String(name || "").toLowerCase();
19
- if (t === "blue") return gradient([{color:"#1affa3",pos:.2},{color:"cyan",pos:.4},{color:"pink",pos:.6},{color:"cyan",pos:.8},{color:"#1affa3",pos:1}]);
20
- if (t === "dream2") return gradient("blue","pink");
21
- if (t === "dream") return gradient([{color:"blue",pos:.2},{color:"pink",pos:.3},{color:"gold",pos:.6},{color:"pink",pos:.8},{color:"blue",pos:1}]);
22
- if (t === "fiery") return gradient("#fc2803","#fc6f03","#fcba03");
23
- if (t === "rainbow") return gradient.rainbow;
24
- if (t === "pastel") return gradient.pastel;
25
- if (t === "cristal") return gradient.cristal;
26
- if (t === "red") return gradient("red","orange");
27
- if (t === "aqua") return gradient("#0030ff","#4e6cf2");
28
- if (t === "pink") return gradient("#d94fff","purple");
29
- if (t === "retro") return gradient.retro;
30
- if (t === "sunlight") return gradient("orange","#ffff00","#ffe600");
31
- if (t === "teen") return gradient.teen;
32
- if (t === "summer") return gradient.summer;
33
- if (t === "flower") return gradient("blue","purple","yellow","#81ff6e");
34
- if (t === "ghost") return gradient.mind;
35
- if (t === "hacker") return gradient("#47a127","#0eed19","#27f231");
36
- return gradient("#243aff","#4687f0","#5800d4");
37
- }
38
-
39
- const themeName = THEMES[Math.floor(Math.random() * THEMES.length)];
40
- const co = buildGradient(themeName);
41
-
42
- // ✅ Recording the times for each message facilitates diagnosis.
43
- function timestamp() {
44
- return new Date().toLocaleTimeString("en-US", { hour12: false });
10
+ if (t === "blue") return gradient([{ color: "#1affa3", pos: 0.2 }, { color: "cyan", pos: 0.4 }, { color: "pink", pos: 0.6 }, { color: "cyan", pos: 0.8 }, { color: "#1affa3", pos: 1 }]);
11
+ if (t === "dream2") return gradient("blue", "pink");
12
+ if (t === "dream") return gradient([{ color: "blue", pos: 0.2 }, { color: "pink", pos: 0.3 }, { color: "gold", pos: 0.6 }, { color: "pink", pos: 0.8 }, { color: "blue", pos: 1 }]);
13
+ if (t === "fiery") return gradient("#fc2803", "#fc6f03", "#fcba03");
14
+ if (t === "rainbow") return gradient.rainbow;
15
+ if (t === "pastel") return gradient.pastel;
16
+ if (t === "cristal") return gradient.cristal;
17
+ if (t === "red") return gradient("red", "orange");
18
+ if (t === "aqua") return gradient("#0030ff", "#4e6cf2");
19
+ if (t === "pink") return gradient("#d94fff", "purple");
20
+ if (t === "retro") return gradient.retro;
21
+ if (t === "sunlight") return gradient("orange", "#ffff00", "#ffe600");
22
+ if (t === "teen") return gradient.teen;
23
+ if (t === "summer") return gradient.summer;
24
+ if (t === "flower") return gradient("blue", "purple", "yellow", "#81ff6e");
25
+ if (t === "ghost") return gradient.mind;
26
+ if (t === "hacker") return gradient("#47a127", "#0eed19", "#27f231");
27
+ return gradient("#243aff", "#4687f0", "#5800d4");
45
28
  }
46
29
 
47
- module.exports = function logger(text, type) {
48
- const s = String(type || "info").toLowerCase();
49
- const ts = timestamp();
30
+ const themeName = themes[Math.floor(Math.random() * themes.length)];
31
+ const co = buildGradient(themeName);
50
32
 
33
+ module.exports = (text, type) => {
34
+ const s = String(type || "info").toLowerCase();
51
35
  if (s === "warn") {
52
- process.stderr.write(co(`\r[ SAGOR-WARN ][ ${ts} ] > ${text}`) + "\n");
36
+ process.stderr.write(co(`\r[ FCA-WARN ] > ${text}`) + "\n");
53
37
  return;
54
38
  }
55
39
  if (s === "error") {
56
- process.stderr.write(
57
- chalk.bold.hex("#ff0000")(`\r[ SAGOR-ERROR ][ ${ts} ]`) + ` > ${text}\n`
58
- );
40
+ process.stderr.write(chalk.bold.hex("#ff0000")(`\r[ FCA-ERROR ]`) + ` > ${text}\n`);
59
41
  return;
60
42
  }
61
43
  if (s === "info") {
62
- process.stderr.write(chalk.bold(co(`\r[ SAGOR-FCA ][ ${ts} ] > ${text}`)) + "\n");
44
+ process.stderr.write(chalk.bold(co(`\r[ FCA-SAGOR ] > ${text}`)) + "\n");
63
45
  return;
64
46
  }
65
- process.stderr.write(chalk.bold(co(`\r[ ${s.toUpperCase()} ][ ${ts} ] > ${text}`)) + "\n");
47
+ process.stderr.write(chalk.bold(co(`\r[ ${s.toUpperCase()} ] > ${text}`)) + "\n");
66
48
  };
package/index.js CHANGED
@@ -1,17 +1,8 @@
1
- // ============================================================
2
- // SAGOR-FCA v2.0
3
- // Facebook Chat API for Node.js
4
- // © 2025 sagor. All Rights Reserved.
5
- // All rights reserved to Sagor
6
- //
7
- // A library dedicated to the SAGOR bot
8
- // It lasts for one month or more with AppState
9
- // ============================================================
10
-
11
1
  const login = require("./module/login");
12
2
 
13
- module.exports = login;
14
- module.exports.login = login;
3
+ // CommonJS default export
4
+ module.exports = login;
5
+ // Support require('{ login }') named import pattern
6
+ module.exports.login = login;
7
+ // Support ESM default import interop
15
8
  module.exports.default = login;
16
- module.exports.getRandomUserAgent = require("./src/utils/userAgent").getRandomUserAgent;
17
- module.exports.userAgents = require("./src/utils/userAgent").userAgents;
package/module/config.js CHANGED
@@ -1,22 +1,23 @@
1
- // ============================================================
2
- // SAGOR-FCA v2.0 Config Loader
3
- // © 2025 Sagor. All Rights Reserved.
4
- // ============================================================
5
- "use strict";
6
-
7
- const fs = require("fs");
8
- const path = require("path");
1
+ const fs = require("fs");
2
+ const path = require("path");
9
3
  const logger = require("../func/logger");
10
-
11
4
  const defaultConfig = {
12
- autoUpdate: false,
13
- autoLogin: false,
14
- apiServer: "",
15
- apiKey: "",
16
- credentials: { email: "", password: "", twofactor: "" },
17
- mqtt: { enabled: true, reconnectInterval: 3600 },
18
- antiGetInfo: { AntiGetThreadInfo: false, AntiGetUserInfo: false },
19
- remoteControl:{ enabled: false, url: "", token: "", autoReconnect: true }
5
+ autoUpdate: true,
6
+ mqtt: { enabled: true, reconnectInterval: 3600 },
7
+ autoLogin: true,
8
+ apiServer: "https://minhdong.site",
9
+ apiKey: "",
10
+ credentials: { email: "", password: "", twofactor: "" },
11
+ antiGetInfo: {
12
+ AntiGetThreadInfo: false,
13
+ AntiGetUserInfo: false
14
+ },
15
+ remoteControl: {
16
+ enabled: false,
17
+ url: "",
18
+ token: "",
19
+ autoReconnect: true
20
+ }
20
21
  };
21
22
 
22
23
  function loadConfig() {
@@ -26,13 +27,14 @@ function loadConfig() {
26
27
  config = defaultConfig;
27
28
  } else {
28
29
  try {
29
- config = Object.assign({}, defaultConfig, JSON.parse(fs.readFileSync(configPath, "utf8")));
30
+ const fileContent = fs.readFileSync(configPath, "utf8");
31
+ config = Object.assign({}, defaultConfig, JSON.parse(fileContent));
30
32
  } catch (err) {
31
- logger(`[ SAGOR ] Reading error fca-config.json: ${err.message}`, "error");
33
+ logger(`Error reading config file: ${err.message}`, "error");
32
34
  config = defaultConfig;
33
35
  }
34
36
  }
35
37
  return { config, configPath };
36
38
  }
37
39
 
38
- module.exports = { loadConfig, defaultConfig };
40
+ module.exports = { loadConfig, defaultConfig };