@lazyneoaz/testfca 1.0.0
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/CHANGELOG.md +229 -0
- package/COOKIE_LOGIN.md +208 -0
- package/LICENSE +3 -0
- package/README.md +492 -0
- package/index.js +2 -0
- package/package.json +120 -0
- package/scripts/build-go.mjs +54 -0
- package/scripts/detect-platform.mjs +36 -0
- package/scripts/download-prebuilt.mjs +119 -0
- package/scripts/package.mjs +6 -0
- package/scripts/postinstall.mjs +113 -0
- package/src/apis/addExternalModule.js +24 -0
- package/src/apis/addUserToGroup.js +108 -0
- package/src/apis/changeAdminStatus.js +148 -0
- package/src/apis/changeArchivedStatus.js +61 -0
- package/src/apis/changeAvatar.js +103 -0
- package/src/apis/changeBio.js +69 -0
- package/src/apis/changeBlockedStatus.js +54 -0
- package/src/apis/changeGroupImage.js +136 -0
- package/src/apis/changeThreadColor.js +116 -0
- package/src/apis/changeThreadEmoji.js +53 -0
- package/src/apis/comment.js +207 -0
- package/src/apis/createAITheme.js +129 -0
- package/src/apis/createNewGroup.js +79 -0
- package/src/apis/createPoll.js +73 -0
- package/src/apis/deleteMessage.js +52 -0
- package/src/apis/deleteThread.js +52 -0
- package/src/apis/e2ee.js +170 -0
- package/src/apis/editMessage.js +78 -0
- package/src/apis/emoji.js +124 -0
- package/src/apis/fetchThemeData.js +82 -0
- package/src/apis/follow.js +81 -0
- package/src/apis/forwardMessage.js +52 -0
- package/src/apis/friend.js +243 -0
- package/src/apis/gcmember.js +122 -0
- package/src/apis/gcname.js +123 -0
- package/src/apis/gcrule.js +119 -0
- package/src/apis/getAccess.js +111 -0
- package/src/apis/getBotInfo.js +88 -0
- package/src/apis/getBotInitialData.js +43 -0
- package/src/apis/getFriendsList.js +79 -0
- package/src/apis/getMessage.js +423 -0
- package/src/apis/getTheme.js +95 -0
- package/src/apis/getThemeInfo.js +116 -0
- package/src/apis/getThreadHistory.js +239 -0
- package/src/apis/getThreadInfo.js +267 -0
- package/src/apis/getThreadList.js +232 -0
- package/src/apis/getThreadPictures.js +58 -0
- package/src/apis/getUserID.js +117 -0
- package/src/apis/getUserInfo.js +513 -0
- package/src/apis/getUserInfoV2.js +146 -0
- package/src/apis/handleMessageRequest.js +50 -0
- package/src/apis/httpGet.js +63 -0
- package/src/apis/httpPost.js +89 -0
- package/src/apis/httpPostFormData.js +69 -0
- package/src/apis/listenMqtt.js +1236 -0
- package/src/apis/listenSpeed.js +179 -0
- package/src/apis/logout.js +93 -0
- package/src/apis/markAsDelivered.js +47 -0
- package/src/apis/markAsRead.js +115 -0
- package/src/apis/markAsReadAll.js +40 -0
- package/src/apis/markAsSeen.js +70 -0
- package/src/apis/mqttDeltaValue.js +250 -0
- package/src/apis/muteThread.js +45 -0
- package/src/apis/nickname.js +132 -0
- package/src/apis/notes.js +163 -0
- package/src/apis/pinMessage.js +150 -0
- package/src/apis/produceMetaTheme.js +180 -0
- package/src/apis/realtime.js +182 -0
- package/src/apis/removeUserFromGroup.js +117 -0
- package/src/apis/resolvePhotoUrl.js +58 -0
- package/src/apis/searchForThread.js +154 -0
- package/src/apis/sendMessage.js +346 -0
- package/src/apis/sendMessageMqtt.js +248 -0
- package/src/apis/sendTypingIndicator.js +105 -0
- package/src/apis/setMessageReaction.js +38 -0
- package/src/apis/setMessageReactionMqtt.js +61 -0
- package/src/apis/setThreadTheme.js +260 -0
- package/src/apis/setThreadThemeMqtt.js +94 -0
- package/src/apis/share.js +107 -0
- package/src/apis/shareContact.js +66 -0
- package/src/apis/stickers.js +257 -0
- package/src/apis/story.js +181 -0
- package/src/apis/theme.js +233 -0
- package/src/apis/unfriend.js +47 -0
- package/src/apis/unsendMessage.js +25 -0
- package/src/database/appStateBackup.js +298 -0
- package/src/database/models/index.js +56 -0
- package/src/database/models/thread.js +31 -0
- package/src/database/models/user.js +32 -0
- package/src/database/threadData.js +101 -0
- package/src/database/userData.js +90 -0
- package/src/e2ee/bridge.js +275 -0
- package/src/e2ee/index.js +60 -0
- package/src/engine/client.js +95 -0
- package/src/engine/models/buildAPI.js +152 -0
- package/src/engine/models/loginHelper.js +574 -0
- package/src/engine/models/setOptions.js +88 -0
- package/src/types/index.d.ts +574 -0
- package/src/utils/antiSuspension.js +529 -0
- package/src/utils/auth-helpers.js +149 -0
- package/src/utils/autoReLogin.js +336 -0
- package/src/utils/axios.js +436 -0
- package/src/utils/cache.js +54 -0
- package/src/utils/clients.js +282 -0
- package/src/utils/constants.js +410 -0
- package/src/utils/formatters/data/formatAttachment.js +370 -0
- package/src/utils/formatters/data/formatDelta.js +109 -0
- package/src/utils/formatters/index.js +159 -0
- package/src/utils/formatters/value/formatCookie.js +91 -0
- package/src/utils/formatters/value/formatDate.js +36 -0
- package/src/utils/formatters/value/formatID.js +16 -0
- package/src/utils/formatters.js +1373 -0
- package/src/utils/headers.js +235 -0
- package/src/utils/index.js +153 -0
- package/src/utils/monitoring.js +333 -0
- package/src/utils/rateLimiter.js +319 -0
- package/src/utils/tokenRefresh.js +680 -0
- package/src/utils/user-agents.js +238 -0
- package/src/utils/validation.js +157 -0
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
|
|
5
|
+
module.exports = (defaultFuncs, api, ctx) => {
|
|
6
|
+
return async function unfriend(userID, callback) {
|
|
7
|
+
let resolveFunc = () => {};
|
|
8
|
+
let rejectFunc = () => {};
|
|
9
|
+
const returnPromise = new Promise((resolve, reject) => {
|
|
10
|
+
resolveFunc = resolve;
|
|
11
|
+
rejectFunc = reject;
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
if (!callback) {
|
|
15
|
+
callback = (err, result) => {
|
|
16
|
+
if (err) return rejectFunc(err);
|
|
17
|
+
resolveFunc(result);
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
try {
|
|
22
|
+
const form = {
|
|
23
|
+
uid: userID,
|
|
24
|
+
unref: "bd_friends_tab",
|
|
25
|
+
floc: "friends_tab",
|
|
26
|
+
"nctr[_mod]": "pagelet_timeline_app_collection_" + ctx.userID + ":2356318349:2"
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
const res = await defaultFuncs.post(
|
|
30
|
+
"https://www.facebook.com/ajax/profile/removefriendconfirm.php",
|
|
31
|
+
ctx.jar,
|
|
32
|
+
form
|
|
33
|
+
).then(utils.parseAndCheckLogin(ctx, defaultFuncs));
|
|
34
|
+
|
|
35
|
+
if (res.error) {
|
|
36
|
+
throw res;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
callback(null, true);
|
|
40
|
+
} catch (err) {
|
|
41
|
+
utils.error("unfriend", err);
|
|
42
|
+
callback(err);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
return returnPromise;
|
|
46
|
+
};
|
|
47
|
+
};
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
|
|
5
|
+
|
|
6
|
+
module.exports = function (defaultFuncs, api, ctx) {
|
|
7
|
+
return async (messageID) => {
|
|
8
|
+
// E2EE routing: unsendMessage({ chatJid, messageId })
|
|
9
|
+
if (utils.getType(messageID) === "Object" && messageID.chatJid && messageID.messageId) {
|
|
10
|
+
if (!api.e2ee || !api.e2ee.isAvailable()) throw new Error("E2EE bridge not available");
|
|
11
|
+
const status = api.e2ee.isConnected();
|
|
12
|
+
if (!status.connected || !status.e2eeConnected) throw new Error("E2EE not connected. Call api.e2ee.connect() and api.e2ee.connectE2EE() first.");
|
|
13
|
+
return api.e2ee.unsendE2EEMessage(messageID.chatJid, messageID.messageId);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
const defData = await defaultFuncs.post("https://www.facebook.com/messaging/unsend_message/", ctx.jar, {
|
|
17
|
+
message_id: messageID
|
|
18
|
+
});
|
|
19
|
+
const resData = await utils.parseAndCheckLogin(ctx, defaultFuncs)(defData);
|
|
20
|
+
if (resData.error) {
|
|
21
|
+
throw new Error(resData);
|
|
22
|
+
}
|
|
23
|
+
return resData;
|
|
24
|
+
};
|
|
25
|
+
};
|
|
@@ -0,0 +1,298 @@
|
|
|
1
|
+
const models = require("./models");
|
|
2
|
+
const utils = require("../utils");
|
|
3
|
+
|
|
4
|
+
let uniqueIndexEnsured = false;
|
|
5
|
+
let autoBackupInterval = null;
|
|
6
|
+
|
|
7
|
+
function getBackupModel() {
|
|
8
|
+
if (!models || !models.sequelize || !models.Sequelize) return null;
|
|
9
|
+
const sequelize = models.sequelize;
|
|
10
|
+
const { DataTypes } = models.Sequelize;
|
|
11
|
+
|
|
12
|
+
if (sequelize.models && sequelize.models.AppStateBackup) {
|
|
13
|
+
return sequelize.models.AppStateBackup;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
const dialect =
|
|
17
|
+
typeof sequelize.getDialect === "function"
|
|
18
|
+
? sequelize.getDialect()
|
|
19
|
+
: "sqlite";
|
|
20
|
+
const LongText =
|
|
21
|
+
dialect === "mysql" || dialect === "mariadb"
|
|
22
|
+
? DataTypes.TEXT("long")
|
|
23
|
+
: DataTypes.TEXT;
|
|
24
|
+
|
|
25
|
+
const AppStateBackup = sequelize.define(
|
|
26
|
+
"AppStateBackup",
|
|
27
|
+
{
|
|
28
|
+
id: { type: DataTypes.INTEGER, primaryKey: true, autoIncrement: true },
|
|
29
|
+
userID: { type: DataTypes.STRING, allowNull: false },
|
|
30
|
+
type: { type: DataTypes.STRING, allowNull: false },
|
|
31
|
+
data: { type: LongText },
|
|
32
|
+
// Add expiry tracking for better session management
|
|
33
|
+
expiresAt: { type: DataTypes.DATE, allowNull: true },
|
|
34
|
+
lastActivityAt: { type: DataTypes.DATE, allowNull: true, defaultValue: DataTypes.NOW }
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
tableName: "app_state_backups",
|
|
38
|
+
timestamps: true,
|
|
39
|
+
indexes: [
|
|
40
|
+
{ unique: true, fields: ["userID", "type"] },
|
|
41
|
+
{ fields: ["expiresAt"] },
|
|
42
|
+
{ fields: ["lastActivityAt"] }
|
|
43
|
+
]
|
|
44
|
+
}
|
|
45
|
+
);
|
|
46
|
+
return AppStateBackup;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
async function ensureUniqueIndex(sequelize) {
|
|
50
|
+
if (uniqueIndexEnsured) return;
|
|
51
|
+
try {
|
|
52
|
+
await sequelize
|
|
53
|
+
.getQueryInterface()
|
|
54
|
+
.addIndex("app_state_backups", ["userID", "type"], {
|
|
55
|
+
unique: true,
|
|
56
|
+
name: "app_state_user_type_unique"
|
|
57
|
+
});
|
|
58
|
+
} catch {}
|
|
59
|
+
uniqueIndexEnsured = true;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
async function upsertBackup(Model, userID, type, data, expiresAt = null) {
|
|
63
|
+
const where = { userID: String(userID || ""), type };
|
|
64
|
+
const updateData = {
|
|
65
|
+
data,
|
|
66
|
+
lastActivityAt: new Date(),
|
|
67
|
+
...(expiresAt && { expiresAt })
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
const row = await Model.findOne({ where });
|
|
71
|
+
if (row) {
|
|
72
|
+
await row.update(updateData);
|
|
73
|
+
utils.log(`Overwrote existing ${type} backup for user ${where.userID}`);
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
await Model.create({ ...where, ...updateData });
|
|
77
|
+
utils.log(`Created new ${type} backup for user ${where.userID}`);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
async function backupAppStateSQL(jar, userID) {
|
|
81
|
+
try {
|
|
82
|
+
const Model = getBackupModel();
|
|
83
|
+
if (!Model) return;
|
|
84
|
+
await Model.sync();
|
|
85
|
+
await ensureUniqueIndex(models.sequelize);
|
|
86
|
+
|
|
87
|
+
const appState = utils.getAppState(jar);
|
|
88
|
+
const cookieStr = cookieHeaderFromJar(jar);
|
|
89
|
+
|
|
90
|
+
// Calculate cookie expiry (default to 90 days if not determinable)
|
|
91
|
+
let expiresAt = null;
|
|
92
|
+
try {
|
|
93
|
+
const cookies = jar.getCookiesSync("https://www.facebook.com");
|
|
94
|
+
const cUserCookie = cookies.find(c => c.key === "c_user");
|
|
95
|
+
if (cUserCookie && cUserCookie.expires) {
|
|
96
|
+
expiresAt = new Date(cUserCookie.expires);
|
|
97
|
+
} else {
|
|
98
|
+
// Default 90 days from now
|
|
99
|
+
expiresAt = new Date(Date.now() + 90 * 24 * 60 * 60 * 1000);
|
|
100
|
+
}
|
|
101
|
+
} catch (e) {
|
|
102
|
+
expiresAt = new Date(Date.now() + 90 * 24 * 60 * 60 * 1000);
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
await upsertBackup(Model, userID, "appstate", JSON.stringify(appState), expiresAt);
|
|
106
|
+
await upsertBackup(Model, userID, "cookie", cookieStr, expiresAt);
|
|
107
|
+
|
|
108
|
+
utils.log("AppState backup stored successfully");
|
|
109
|
+
} catch (e) {
|
|
110
|
+
utils.warn(
|
|
111
|
+
`Failed to save AppState backup: ${
|
|
112
|
+
e && e.message ? e.message : String(e)
|
|
113
|
+
}`
|
|
114
|
+
);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
async function getLatestBackup(userID, type) {
|
|
119
|
+
try {
|
|
120
|
+
const Model = getBackupModel();
|
|
121
|
+
if (!Model) return null;
|
|
122
|
+
const row = await Model.findOne({
|
|
123
|
+
where: { userID: String(userID || ""), type }
|
|
124
|
+
});
|
|
125
|
+
return row ? row.data : null;
|
|
126
|
+
} catch {
|
|
127
|
+
return null;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
async function getLatestBackupAny(type) {
|
|
132
|
+
try {
|
|
133
|
+
const Model = getBackupModel();
|
|
134
|
+
if (!Model) return null;
|
|
135
|
+
const row = await Model.findOne({
|
|
136
|
+
where: { type },
|
|
137
|
+
order: [["updatedAt", "DESC"]]
|
|
138
|
+
});
|
|
139
|
+
return row ? row.data : null;
|
|
140
|
+
} catch {
|
|
141
|
+
return null;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function cookieHeaderFromJar(jar) {
|
|
146
|
+
const urls = ["https://www.facebook.com", "https://www.messenger.com"];
|
|
147
|
+
const seen = new Set();
|
|
148
|
+
const parts = [];
|
|
149
|
+
for (const url of urls) {
|
|
150
|
+
let cookieString = "";
|
|
151
|
+
try {
|
|
152
|
+
if (typeof jar.getCookieStringSync === "function") {
|
|
153
|
+
cookieString = jar.getCookieStringSync(url);
|
|
154
|
+
}
|
|
155
|
+
} catch {}
|
|
156
|
+
if (!cookieString) continue;
|
|
157
|
+
for (const kv of cookieString.split(";")) {
|
|
158
|
+
const trimmed = kv.trim();
|
|
159
|
+
const name = trimmed.split("=")[0];
|
|
160
|
+
if (!name || seen.has(name)) continue;
|
|
161
|
+
seen.add(name);
|
|
162
|
+
parts.push(trimmed);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
return parts.join("; ");
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
async function hydrateJarFromDB(jar, userID) {
|
|
169
|
+
try {
|
|
170
|
+
const { normalizeCookieHeaderString, setJarFromPairs } = require("../utils/formatters/value/formatCookie");
|
|
171
|
+
|
|
172
|
+
let cookieHeader = null;
|
|
173
|
+
let appStateJson = null;
|
|
174
|
+
|
|
175
|
+
if (userID) {
|
|
176
|
+
cookieHeader = await getLatestBackup(userID, "cookie");
|
|
177
|
+
appStateJson = await getLatestBackup(userID, "appstate");
|
|
178
|
+
} else {
|
|
179
|
+
cookieHeader = await getLatestBackupAny("cookie");
|
|
180
|
+
appStateJson = await getLatestBackupAny("appstate");
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
if (cookieHeader) {
|
|
184
|
+
const pairs = normalizeCookieHeaderString(cookieHeader);
|
|
185
|
+
if (pairs.length) {
|
|
186
|
+
setJarFromPairs(jar, pairs);
|
|
187
|
+
return true;
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
if (appStateJson) {
|
|
192
|
+
let parsed = null;
|
|
193
|
+
try {
|
|
194
|
+
parsed = JSON.parse(appStateJson);
|
|
195
|
+
} catch {}
|
|
196
|
+
if (Array.isArray(parsed)) {
|
|
197
|
+
const pairs = parsed.map(c => [c.name || c.key, c.value].join("="));
|
|
198
|
+
setJarFromPairs(jar, pairs);
|
|
199
|
+
return true;
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
return false;
|
|
204
|
+
} catch {
|
|
205
|
+
return false;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
/**
|
|
210
|
+
* Start automatic periodic backup of appState
|
|
211
|
+
* @param {Object} jar - Cookie jar
|
|
212
|
+
* @param {string} userID - User ID
|
|
213
|
+
* @param {number} intervalMs - Backup interval in milliseconds (default: 5 minutes)
|
|
214
|
+
*/
|
|
215
|
+
function startAutoBackup(jar, userID, intervalMs = 5 * 60 * 1000) {
|
|
216
|
+
// Clear existing interval if any
|
|
217
|
+
if (autoBackupInterval) {
|
|
218
|
+
clearInterval(autoBackupInterval);
|
|
219
|
+
autoBackupInterval = null;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
autoBackupInterval = setInterval(async () => {
|
|
223
|
+
try {
|
|
224
|
+
await backupAppStateSQL(jar, userID);
|
|
225
|
+
utils.log("AppStateBackup", `Auto-backup completed for user ${userID}`);
|
|
226
|
+
} catch (err) {
|
|
227
|
+
utils.warn("AppStateBackup", "Auto-backup failed:", err.message);
|
|
228
|
+
}
|
|
229
|
+
}, intervalMs);
|
|
230
|
+
|
|
231
|
+
utils.log("AppStateBackup", `Auto-backup started (interval: ${intervalMs}ms)`);
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
/**
|
|
235
|
+
* Stop automatic backup
|
|
236
|
+
*/
|
|
237
|
+
function stopAutoBackup() {
|
|
238
|
+
if (autoBackupInterval) {
|
|
239
|
+
clearInterval(autoBackupInterval);
|
|
240
|
+
autoBackupInterval = null;
|
|
241
|
+
utils.log("AppStateBackup", "Auto-backup stopped");
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
/**
|
|
246
|
+
* Update session activity timestamp
|
|
247
|
+
* @param {string} userID - User ID
|
|
248
|
+
*/
|
|
249
|
+
async function updateSessionActivity(userID) {
|
|
250
|
+
try {
|
|
251
|
+
const Model = getBackupModel();
|
|
252
|
+
if (!Model) return;
|
|
253
|
+
|
|
254
|
+
await Model.update(
|
|
255
|
+
{ lastActivityAt: new Date() },
|
|
256
|
+
{ where: { userID: String(userID || "") } }
|
|
257
|
+
);
|
|
258
|
+
} catch (e) {
|
|
259
|
+
// Silent fail - not critical
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
/**
|
|
264
|
+
* Check if session is still valid based on expiry
|
|
265
|
+
* @param {string} userID - User ID
|
|
266
|
+
* @returns {Promise<boolean>}
|
|
267
|
+
*/
|
|
268
|
+
async function isSessionValid(userID) {
|
|
269
|
+
try {
|
|
270
|
+
const Model = getBackupModel();
|
|
271
|
+
if (!Model) return true;
|
|
272
|
+
|
|
273
|
+
const row = await Model.findOne({
|
|
274
|
+
where: { userID: String(userID || ""), type: "appstate" }
|
|
275
|
+
});
|
|
276
|
+
|
|
277
|
+
if (!row || !row.expiresAt) return true;
|
|
278
|
+
|
|
279
|
+
return new Date(row.expiresAt) > new Date();
|
|
280
|
+
} catch {
|
|
281
|
+
return true;
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
module.exports = {
|
|
286
|
+
getBackupModel,
|
|
287
|
+
ensureUniqueIndex,
|
|
288
|
+
upsertBackup,
|
|
289
|
+
backupAppStateSQL,
|
|
290
|
+
getLatestBackup,
|
|
291
|
+
getLatestBackupAny,
|
|
292
|
+
hydrateJarFromDB,
|
|
293
|
+
cookieHeaderFromJar,
|
|
294
|
+
startAutoBackup,
|
|
295
|
+
stopAutoBackup,
|
|
296
|
+
updateSessionActivity,
|
|
297
|
+
isSessionValid
|
|
298
|
+
};
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
const { Sequelize } = require("sequelize");
|
|
2
|
+
const fs = require("fs");
|
|
3
|
+
const path = require("path");
|
|
4
|
+
|
|
5
|
+
const databasePath = path.join(process.cwd(), "nkxfca_database");
|
|
6
|
+
if (!fs.existsSync(databasePath)) {
|
|
7
|
+
fs.mkdirSync(databasePath, { recursive: true });
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
const sequelize = new Sequelize({
|
|
11
|
+
dialect: "sqlite",
|
|
12
|
+
storage: path.join(databasePath, "nkxfca.sqlite"),
|
|
13
|
+
logging: false,
|
|
14
|
+
pool: {
|
|
15
|
+
max: 5,
|
|
16
|
+
min: 0,
|
|
17
|
+
acquire: 30000,
|
|
18
|
+
idle: 10000
|
|
19
|
+
},
|
|
20
|
+
retry: {
|
|
21
|
+
max: 3
|
|
22
|
+
},
|
|
23
|
+
dialectOptions: {
|
|
24
|
+
timeout: 5000
|
|
25
|
+
},
|
|
26
|
+
isolationLevel: Sequelize.Transaction.ISOLATION_LEVELS.READ_COMMITTED
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
const models = {};
|
|
30
|
+
|
|
31
|
+
fs.readdirSync(__dirname)
|
|
32
|
+
.filter(file => file.endsWith(".js") && file !== "index.js")
|
|
33
|
+
.forEach(file => {
|
|
34
|
+
const model = require(path.join(__dirname, file))(sequelize);
|
|
35
|
+
models[model.name] = model;
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
Object.keys(models).forEach(modelName => {
|
|
39
|
+
if (models[modelName].associate) {
|
|
40
|
+
models[modelName].associate(models);
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
models.sequelize = sequelize;
|
|
45
|
+
models.Sequelize = Sequelize;
|
|
46
|
+
|
|
47
|
+
models.syncAll = async () => {
|
|
48
|
+
try {
|
|
49
|
+
await sequelize.sync({ force: false });
|
|
50
|
+
} catch (error) {
|
|
51
|
+
console.error("Failed to synchronize nkxfca database models:", error);
|
|
52
|
+
throw error;
|
|
53
|
+
}
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
module.exports = models;
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
module.exports = function(sequelize) {
|
|
2
|
+
const { Model, DataTypes } = require("sequelize");
|
|
3
|
+
|
|
4
|
+
class Thread extends Model {}
|
|
5
|
+
|
|
6
|
+
Thread.init(
|
|
7
|
+
{
|
|
8
|
+
num: {
|
|
9
|
+
type: DataTypes.INTEGER,
|
|
10
|
+
allowNull: false,
|
|
11
|
+
autoIncrement: true,
|
|
12
|
+
primaryKey: true
|
|
13
|
+
},
|
|
14
|
+
threadID: {
|
|
15
|
+
type: DataTypes.STRING,
|
|
16
|
+
allowNull: false,
|
|
17
|
+
unique: true
|
|
18
|
+
},
|
|
19
|
+
data: {
|
|
20
|
+
type: DataTypes.JSON,
|
|
21
|
+
allowNull: true
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
sequelize,
|
|
26
|
+
modelName: "FcaNeoProThread",
|
|
27
|
+
timestamps: true
|
|
28
|
+
}
|
|
29
|
+
);
|
|
30
|
+
return Thread;
|
|
31
|
+
};
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
module.exports = function (sequelize) {
|
|
2
|
+
const { Model, DataTypes } = require("sequelize");
|
|
3
|
+
|
|
4
|
+
class User extends Model { }
|
|
5
|
+
|
|
6
|
+
User.init(
|
|
7
|
+
{
|
|
8
|
+
num: {
|
|
9
|
+
type: DataTypes.INTEGER,
|
|
10
|
+
allowNull: false,
|
|
11
|
+
autoIncrement: true,
|
|
12
|
+
primaryKey: true
|
|
13
|
+
},
|
|
14
|
+
userID: {
|
|
15
|
+
type: DataTypes.STRING,
|
|
16
|
+
allowNull: false,
|
|
17
|
+
unique: true
|
|
18
|
+
},
|
|
19
|
+
data: {
|
|
20
|
+
type: DataTypes.JSON,
|
|
21
|
+
allowNull: true
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
sequelize,
|
|
26
|
+
modelName: "FcaNeoProUser",
|
|
27
|
+
timestamps: true
|
|
28
|
+
}
|
|
29
|
+
);
|
|
30
|
+
|
|
31
|
+
return User;
|
|
32
|
+
};
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
const { Thread } = require("./models");
|
|
2
|
+
|
|
3
|
+
const validateThreadID = threadID => {
|
|
4
|
+
if (typeof threadID !== "string" && typeof threadID !== "number") {
|
|
5
|
+
throw new Error("Invalid threadID: must be a string or number.");
|
|
6
|
+
}
|
|
7
|
+
return String(threadID);
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
const validateData = data => {
|
|
11
|
+
if (!data || typeof data !== "object" || Array.isArray(data)) {
|
|
12
|
+
throw new Error("Invalid data: must be a non-empty object.");
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
module.exports = function(api) {
|
|
17
|
+
return {
|
|
18
|
+
async create(threadID, data) {
|
|
19
|
+
try {
|
|
20
|
+
let thread = await Thread.findOne({ where: { threadID } });
|
|
21
|
+
if (thread) {
|
|
22
|
+
return { thread: thread.get(), created: false };
|
|
23
|
+
}
|
|
24
|
+
thread = await Thread.create({ threadID, ...data });
|
|
25
|
+
return { thread: thread.get(), created: true };
|
|
26
|
+
} catch (error) {
|
|
27
|
+
throw new Error(`Failed to create thread: ${error.message}`);
|
|
28
|
+
}
|
|
29
|
+
},
|
|
30
|
+
|
|
31
|
+
async get(threadID) {
|
|
32
|
+
try {
|
|
33
|
+
threadID = validateThreadID(threadID);
|
|
34
|
+
const thread = await Thread.findOne({ where: { threadID } });
|
|
35
|
+
return thread ? thread.get() : null;
|
|
36
|
+
} catch (error) {
|
|
37
|
+
throw new Error(`Failed to get thread: ${error.message}`);
|
|
38
|
+
}
|
|
39
|
+
},
|
|
40
|
+
|
|
41
|
+
async update(threadID, data) {
|
|
42
|
+
try {
|
|
43
|
+
threadID = validateThreadID(threadID);
|
|
44
|
+
validateData(data);
|
|
45
|
+
const thread = await Thread.findOne({ where: { threadID } });
|
|
46
|
+
|
|
47
|
+
if (thread) {
|
|
48
|
+
await thread.update(data);
|
|
49
|
+
return { thread: thread.get(), created: false };
|
|
50
|
+
} else {
|
|
51
|
+
const newThread = await Thread.create({ ...data, threadID });
|
|
52
|
+
return { thread: newThread.get(), created: true };
|
|
53
|
+
}
|
|
54
|
+
} catch (error) {
|
|
55
|
+
throw new Error(`Failed to update thread: ${error.message}`);
|
|
56
|
+
}
|
|
57
|
+
},
|
|
58
|
+
|
|
59
|
+
async del(threadID) {
|
|
60
|
+
try {
|
|
61
|
+
if (!threadID) {
|
|
62
|
+
throw new Error("threadID is required and cannot be undefined");
|
|
63
|
+
}
|
|
64
|
+
threadID = validateThreadID(threadID);
|
|
65
|
+
if (!threadID) {
|
|
66
|
+
throw new Error("Invalid threadID");
|
|
67
|
+
}
|
|
68
|
+
const result = await Thread.destroy({ where: { threadID } });
|
|
69
|
+
if (result === 0) {
|
|
70
|
+
throw new Error("No thread found with the specified threadID");
|
|
71
|
+
}
|
|
72
|
+
return result;
|
|
73
|
+
} catch (error) {
|
|
74
|
+
throw new Error(`Failed to delete thread: ${error.message}`);
|
|
75
|
+
}
|
|
76
|
+
},
|
|
77
|
+
|
|
78
|
+
async delAll() {
|
|
79
|
+
try {
|
|
80
|
+
return await Thread.destroy({ where: {} });
|
|
81
|
+
} catch (error) {
|
|
82
|
+
throw new Error(`Failed to delete all threads: ${error.message}`);
|
|
83
|
+
}
|
|
84
|
+
},
|
|
85
|
+
|
|
86
|
+
async getAll(keys = null) {
|
|
87
|
+
try {
|
|
88
|
+
const attributes =
|
|
89
|
+
typeof keys === "string"
|
|
90
|
+
? [keys]
|
|
91
|
+
: Array.isArray(keys)
|
|
92
|
+
? keys
|
|
93
|
+
: undefined;
|
|
94
|
+
const threads = await Thread.findAll({ attributes });
|
|
95
|
+
return threads.map(thread => thread.get());
|
|
96
|
+
} catch (error) {
|
|
97
|
+
throw new Error(`Failed to get all threads: ${error.message}`);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
};
|
|
101
|
+
};
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
const { User } = require("./models");
|
|
2
|
+
|
|
3
|
+
const validateUserID = userID => {
|
|
4
|
+
if (typeof userID !== "string" && typeof userID !== "number") {
|
|
5
|
+
throw new Error("Invalid userID: must be a string or number.");
|
|
6
|
+
}
|
|
7
|
+
return String(userID);
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
const validateData = data => {
|
|
11
|
+
if (!data || typeof data !== "object" || Array.isArray(data)) {
|
|
12
|
+
throw new Error("Invalid data: must be a non-empty object.");
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
module.exports = function (api) {
|
|
17
|
+
return {
|
|
18
|
+
async create(userID, data) {
|
|
19
|
+
try {
|
|
20
|
+
userID = validateUserID(userID);
|
|
21
|
+
validateData(data);
|
|
22
|
+
let user = await User.findOne({ where: { userID } });
|
|
23
|
+
if (user) return { user: user.get(), created: false };
|
|
24
|
+
const payload = Object.prototype.hasOwnProperty.call(data, "data") ? data : { data };
|
|
25
|
+
user = await User.create({ userID, ...payload });
|
|
26
|
+
return { user: user.get(), created: true };
|
|
27
|
+
} catch (error) {
|
|
28
|
+
throw new Error(`Failed to create user: ${error.message}`);
|
|
29
|
+
}
|
|
30
|
+
},
|
|
31
|
+
|
|
32
|
+
async get(userID) {
|
|
33
|
+
try {
|
|
34
|
+
userID = validateUserID(userID);
|
|
35
|
+
const user = await User.findOne({ where: { userID } });
|
|
36
|
+
return user ? user.get() : null;
|
|
37
|
+
} catch (error) {
|
|
38
|
+
throw new Error(`Failed to get user: ${error.message}`);
|
|
39
|
+
}
|
|
40
|
+
},
|
|
41
|
+
|
|
42
|
+
async update(userID, data) {
|
|
43
|
+
try {
|
|
44
|
+
userID = validateUserID(userID);
|
|
45
|
+
validateData(data);
|
|
46
|
+
const payload = Object.prototype.hasOwnProperty.call(data, "data") ? data : { data };
|
|
47
|
+
const user = await User.findOne({ where: { userID } });
|
|
48
|
+
if (user) {
|
|
49
|
+
await user.update(payload);
|
|
50
|
+
return { user: user.get(), created: false };
|
|
51
|
+
} else {
|
|
52
|
+
const newUser = await User.create({ userID, ...payload });
|
|
53
|
+
return { user: newUser.get(), created: true };
|
|
54
|
+
}
|
|
55
|
+
} catch (error) {
|
|
56
|
+
throw new Error(`Failed to update user: ${error.message}`);
|
|
57
|
+
}
|
|
58
|
+
},
|
|
59
|
+
|
|
60
|
+
async del(userID) {
|
|
61
|
+
try {
|
|
62
|
+
if (!userID) throw new Error("userID is required and cannot be undefined");
|
|
63
|
+
userID = validateUserID(userID);
|
|
64
|
+
const result = await User.destroy({ where: { userID } });
|
|
65
|
+
if (result === 0) throw new Error("No user found with the specified userID");
|
|
66
|
+
return result;
|
|
67
|
+
} catch (error) {
|
|
68
|
+
throw new Error(`Failed to delete user: ${error.message}`);
|
|
69
|
+
}
|
|
70
|
+
},
|
|
71
|
+
|
|
72
|
+
async delAll() {
|
|
73
|
+
try {
|
|
74
|
+
return await User.destroy({ where: {} });
|
|
75
|
+
} catch (error) {
|
|
76
|
+
throw new Error(`Failed to delete all users: ${error.message}`);
|
|
77
|
+
}
|
|
78
|
+
},
|
|
79
|
+
|
|
80
|
+
async getAll(keys = null) {
|
|
81
|
+
try {
|
|
82
|
+
const attributes = typeof keys === "string" ? [keys] : Array.isArray(keys) ? keys : undefined;
|
|
83
|
+
const users = await User.findAll({ attributes });
|
|
84
|
+
return users.map(u => u.get());
|
|
85
|
+
} catch (error) {
|
|
86
|
+
throw new Error(`Failed to get all users: ${error.message}`);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
};
|
|
90
|
+
};
|