@dongdev/fca-unofficial 0.0.4 → 0.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 (76) hide show
  1. package/.travis.yml +6 -6
  2. package/CHANGELOG.md +1 -1
  3. package/DOCS.md +1738 -1738
  4. package/LICENSE-MIT +21 -21
  5. package/README.md +219 -219
  6. package/index.js +307 -571
  7. package/lib/login.js +0 -0
  8. package/package.json +6 -5
  9. package/src/addExternalModule.js +19 -15
  10. package/src/addUserToGroup.js +113 -77
  11. package/src/changeAdminStatus.js +79 -47
  12. package/src/changeArchivedStatus.js +55 -41
  13. package/src/changeAvatar.js +126 -0
  14. package/src/changeBio.js +66 -54
  15. package/src/changeBlockedStatus.js +40 -29
  16. package/src/changeGroupImage.js +127 -101
  17. package/src/changeNickname.js +50 -36
  18. package/src/changeThreadColor.js +65 -61
  19. package/src/changeThreadEmoji.js +55 -41
  20. package/src/createNewGroup.js +86 -70
  21. package/src/createPoll.js +71 -59
  22. package/src/deleteMessage.js +56 -44
  23. package/src/deleteThread.js +56 -42
  24. package/src/forwardAttachment.js +60 -47
  25. package/src/getCurrentUserID.js +7 -7
  26. package/src/getEmojiUrl.js +29 -27
  27. package/src/getFriendsList.js +83 -73
  28. package/src/getMessage.js +796 -0
  29. package/src/getThreadHistory.js +666 -537
  30. package/src/getThreadInfo.js +232 -171
  31. package/src/getThreadList.js +192 -213
  32. package/src/getThreadPictures.js +79 -59
  33. package/src/getUserID.js +66 -61
  34. package/src/getUserInfo.js +74 -66
  35. package/src/handleFriendRequest.js +61 -46
  36. package/src/handleMessageRequest.js +65 -47
  37. package/src/httpGet.js +52 -44
  38. package/src/httpPost.js +52 -43
  39. package/src/httpPostFormData.js +63 -0
  40. package/src/listenMqtt.js +969 -709
  41. package/src/logout.js +62 -55
  42. package/src/markAsDelivered.js +58 -47
  43. package/src/markAsRead.js +80 -70
  44. package/src/markAsReadAll.js +49 -39
  45. package/src/markAsSeen.js +59 -48
  46. package/src/muteThread.js +52 -45
  47. package/src/postFormData.js +46 -0
  48. package/src/refreshFb_dtsg.js +81 -0
  49. package/src/removeUserFromGroup.js +79 -45
  50. package/src/resolvePhotoUrl.js +45 -36
  51. package/src/searchForThread.js +53 -42
  52. package/src/sendMessage.js +328 -328
  53. package/src/sendMessageMqtt.js +316 -0
  54. package/src/sendTypingIndicator.js +103 -70
  55. package/src/setMessageReaction.js +106 -98
  56. package/src/setPostReaction.js +102 -95
  57. package/src/setTitle.js +86 -70
  58. package/src/threadColors.js +131 -41
  59. package/src/unfriend.js +52 -42
  60. package/src/unsendMessage.js +49 -39
  61. package/src/uploadAttachment.js +95 -0
  62. package/utils.js +1501 -1196
  63. package/.gitattributes +0 -2
  64. package/src/Screenshot.js +0 -83
  65. package/src/changeAvt.js +0 -85
  66. package/src/getThreadHistoryDeprecated.js +0 -71
  67. package/src/getThreadInfoDeprecated.js +0 -56
  68. package/src/getThreadListDeprecated.js +0 -46
  69. package/src/shareContact.js +0 -46
  70. package/test/data/shareAttach.js +0 -146
  71. package/test/data/something.mov +0 -0
  72. package/test/data/test.png +0 -0
  73. package/test/data/test.txt +0 -7
  74. package/test/example-config.json +0 -18
  75. package/test/test-page.js +0 -140
  76. package/test/test.js +0 -385
package/lib/login.js ADDED
File without changes
package/package.json CHANGED
@@ -1,22 +1,23 @@
1
1
  {
2
2
  "name": "@dongdev/fca-unofficial",
3
- "version": "0.0.4",
3
+ "version": "0.0.6",
4
4
  "description": "A Facebook chat API without XMPP, will not be deprecated after April 30th, 2015.",
5
5
  "main": "index.js",
6
6
  "repository": {
7
7
  "type": "git",
8
8
  "url": "https://github.com/DongDev-VN/fca-unofficial.git"
9
9
  },
10
- "author": "Avery, David, Maude, Benjamin, UIRI, MiraiTeam",
10
+ "author": "Avery, David, Maude, Benjamin, UIRI, DongDev",
11
11
  "license": "MIT",
12
12
  "dependencies": {
13
13
  "bluebird": "^2.11.0",
14
14
  "cheerio": "^1.0.0-rc.10",
15
+ "duplexify": "^4.1.3",
15
16
  "https-proxy-agent": "^4.0.0",
16
- "mqtt": "^4.2.8",
17
+ "mqtt": "^4.3.7",
17
18
  "npmlog": "^1.2.0",
18
19
  "request": "^2.53.0",
19
- "websocket-stream": "^5.5.0"
20
+ "ws": "^8.18.1"
20
21
  },
21
22
  "devDependencies": {
22
23
  "eslint": "^7.5.0",
@@ -45,4 +46,4 @@
45
46
  "directories": {
46
47
  "test": "test"
47
48
  }
48
- }
49
+ }
@@ -1,15 +1,19 @@
1
- "use strict";
2
-
3
- const utils = require("../utils");
4
-
5
- module.exports = function (defaultFuncs, api, ctx) {
6
- return function addExternalModule(moduleObj) {
7
- if (utils.getType(moduleObj) == "Object") {
8
- for (let apiName in moduleObj) {
9
- if (utils.getType(moduleObj[apiName]) == "Function") api[apiName] = moduleObj[apiName](defaultFuncs, api, ctx);
10
- else throw new Error(`Item "${apiName}" in moduleObj must be a function, not ${utils.getType(moduleObj[apiName])}!`);
11
- }
12
- }
13
- else throw new Error(`moduleObj must be an object, not ${utils.getType(moduleObj)}!`);
14
- };
15
- };
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+
5
+ module.exports = function (defaultFuncs, api, ctx) {
6
+ return function addExternalModule(moduleObj) {
7
+ if (utils.getType(moduleObj) == "Object") {
8
+ for (const apiName in moduleObj) {
9
+ if (utils.getType(moduleObj[apiName]) == "Function") {
10
+ api[apiName] = moduleObj[apiName](defaultFuncs, api, ctx);
11
+ } else {
12
+ throw new Error(`Item "${apiName}" in moduleObj must be a function, not ${utils.getType(moduleObj[apiName])}!`);
13
+ }
14
+ }
15
+ } else {
16
+ throw new Error(`moduleObj must be an object, not ${utils.getType(moduleObj)}!`);
17
+ }
18
+ };
19
+ };
@@ -1,77 +1,113 @@
1
- "use strict";
2
-
3
- var utils = require("../utils");
4
- var log = require("npmlog");
5
-
6
- module.exports = function (defaultFuncs, api, ctx) {
7
- return function addUserToGroup(userID, threadID, callback) {
8
- var resolveFunc = function () { };
9
- var rejectFunc = function () { };
10
- var returnPromise = new Promise(function (resolve, reject) {
11
- resolveFunc = resolve;
12
- rejectFunc = reject;
13
- });
14
-
15
- if (!callback && (utils.getType(threadID) === "Function" || utils.getType(threadID) === "AsyncFunction")) throw { error: "please pass a threadID as a second argument." };
16
-
17
- if (!callback) {
18
- callback = function (err) {
19
- if (err) return rejectFunc(err);
20
- resolveFunc();
21
- };
22
- }
23
-
24
- if (utils.getType(threadID) !== "Number" && utils.getType(threadID) !== "String") throw { error: "ThreadID should be of type Number or String and not " + utils.getType(threadID) + "." };
25
-
26
- if (utils.getType(userID) !== "Array") userID = [userID];
27
-
28
- var messageAndOTID = utils.generateOfflineThreadingID();
29
- var form = {
30
- client: "mercury",
31
- action_type: "ma-type:log-message",
32
- author: "fbid:" + ctx.userID,
33
- thread_id: "",
34
- timestamp: Date.now(),
35
- timestamp_absolute: "Today",
36
- timestamp_relative: utils.generateTimestampRelative(),
37
- timestamp_time_passed: "0",
38
- is_unread: false,
39
- is_cleared: false,
40
- is_forward: false,
41
- is_filtered_content: false,
42
- is_filtered_content_bh: false,
43
- is_filtered_content_account: false,
44
- is_spoof_warning: false,
45
- source: "source:chat:web",
46
- "source_tags[0]": "source:chat",
47
- log_message_type: "log:subscribe",
48
- status: "0",
49
- offline_threading_id: messageAndOTID,
50
- message_id: messageAndOTID,
51
- threading_id: utils.generateThreadingID(ctx.clientID),
52
- manual_retry_cnt: "0",
53
- thread_fbid: threadID
54
- };
55
-
56
- for (var i = 0; i < userID.length; i++) {
57
- if (utils.getType(userID[i]) !== "Number" && utils.getType(userID[i]) !== "String") throw { error: "Elements of userID should be of type Number or String and not " + utils.getType(userID[i]) + "." };
58
- form["log_message_data[added_participants][" + i + "]"] = "fbid:" + userID[i];
59
- }
60
-
61
- defaultFuncs
62
- .post("https://www.facebook.com/messaging/send/", ctx.jar, form)
63
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
64
- .then(function (resData) {
65
- if (!resData) throw { error: "Add to group failed." };
66
- if (resData.error) throw resData;
67
-
68
- return callback();
69
- })
70
- .catch(function (err) {
71
- log.error("addUserToGroup", err);
72
- return callback(err);
73
- });
74
-
75
- return returnPromise;
76
- };
77
- };
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ const log = require("npmlog");
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ return function addUserToGroup(userID, threadID, callback) {
8
+ let resolveFunc = function () { };
9
+ let rejectFunc = function () { };
10
+ const returnPromise = new Promise(function (resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+
15
+ if (
16
+ !callback &&
17
+ (utils.getType(threadID) === "Function" ||
18
+ utils.getType(threadID) === "AsyncFunction")
19
+ ) {
20
+ throw new utils.CustomError({ error: "please pass a threadID as a second argument." });
21
+ }
22
+
23
+ if (!callback) {
24
+ callback = function (err) {
25
+ if (err) {
26
+ return rejectFunc(err);
27
+ }
28
+ resolveFunc();
29
+ };
30
+ }
31
+
32
+ if (
33
+ utils.getType(threadID) !== "Number" &&
34
+ utils.getType(threadID) !== "String"
35
+ ) {
36
+ throw new utils.CustomError({
37
+ error:
38
+ "ThreadID should be of type Number or String and not " +
39
+ utils.getType(threadID) +
40
+ "."
41
+ });
42
+ }
43
+
44
+ if (utils.getType(userID) !== "Array") {
45
+ userID = [userID];
46
+ }
47
+
48
+ const messageAndOTID = utils.generateOfflineThreadingID();
49
+ const form = {
50
+ client: "mercury",
51
+ action_type: "ma-type:log-message",
52
+ author: "fbid:" + (ctx.i_userID || ctx.userID),
53
+ thread_id: "",
54
+ timestamp: Date.now(),
55
+ timestamp_absolute: "Today",
56
+ timestamp_relative: utils.generateTimestampRelative(),
57
+ timestamp_time_passed: "0",
58
+ is_unread: false,
59
+ is_cleared: false,
60
+ is_forward: false,
61
+ is_filtered_content: false,
62
+ is_filtered_content_bh: false,
63
+ is_filtered_content_account: false,
64
+ is_spoof_warning: false,
65
+ source: "source:chat:web",
66
+ "source_tags[0]": "source:chat",
67
+ log_message_type: "log:subscribe",
68
+ status: "0",
69
+ offline_threading_id: messageAndOTID,
70
+ message_id: messageAndOTID,
71
+ threading_id: utils.generateThreadingID(ctx.clientID),
72
+ manual_retry_cnt: "0",
73
+ thread_fbid: threadID
74
+ };
75
+
76
+ for (let i = 0; i < userID.length; i++) {
77
+ if (
78
+ utils.getType(userID[i]) !== "Number" &&
79
+ utils.getType(userID[i]) !== "String"
80
+ ) {
81
+ throw new utils.CustomError({
82
+ error:
83
+ "Elements of userID should be of type Number or String and not " +
84
+ utils.getType(userID[i]) +
85
+ "."
86
+ });
87
+ }
88
+
89
+ form["log_message_data[added_participants][" + i + "]"] =
90
+ "fbid:" + userID[i];
91
+ }
92
+
93
+ defaultFuncs
94
+ .post("https://www.facebook.com/messaging/send/", ctx.jar, form)
95
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
96
+ .then(function (resData) {
97
+ if (!resData) {
98
+ throw new utils.CustomError({ error: "Add to group failed." });
99
+ }
100
+ if (resData.error) {
101
+ throw new utils.CustomError(resData);
102
+ }
103
+
104
+ return callback();
105
+ })
106
+ .catch(function (err) {
107
+ log.error("addUserToGroup", err);
108
+ return callback(err);
109
+ });
110
+
111
+ return returnPromise;
112
+ };
113
+ };
@@ -1,47 +1,79 @@
1
- "use strict";
2
-
3
- const utils = require("../utils");
4
-
5
- module.exports = function (defaultFuncs, api, ctx) {
6
- return function changeAdminStatus(threadID, adminID, adminStatus) {
7
- if (utils.getType(threadID) !== "String") throw { error: "changeAdminStatus: threadID must be a string" };
8
- if (utils.getType(adminID) !== "String" && utils.getType(adminID) !== "Array") throw { error: "changeAdminStatus: adminID must be a string or an array" };
9
- if (utils.getType(adminStatus) !== "Boolean") throw { error: "changeAdminStatus: adminStatus must be true or false" };
10
-
11
- let wsContent = {
12
- request_id: 1,
13
- type: 3,
14
- payload: {
15
- version_id: '3816854585040595',
16
- tasks: [],
17
- epoch_id: 6763184801413415579,
18
- data_trace_id: null
19
- },
20
- app_id: '772021112871879'
21
- }
22
-
23
- if (utils.getType(adminID) === "Array") {
24
- for (let i = 0; i < adminID.length; i++) {
25
- wsContent.payload.tasks.push({
26
- label: '25',
27
- payload: JSON.stringify({ thread_key: threadID, contact_id: adminID[i], is_admin: adminStatus }),
28
- queue_name: 'admin_status',
29
- task_id: i + 1,
30
- failure_count: null
31
- });
32
- }
33
- }
34
- else {
35
- wsContent.payload.tasks.push({
36
- label: '25',
37
- payload: JSON.stringify({ thread_key: threadID, contact_id: adminID, is_admin: adminStatus }),
38
- queue_name: 'admin_status',
39
- task_id: 1,
40
- failure_count: null
41
- });
42
- }
43
-
44
- wsContent.payload = JSON.stringify(wsContent.payload);
45
- return new Promise((resolve, reject) => ctx.mqttClient && ctx.mqttClient.publish('/ls_req', JSON.stringify(wsContent), {}, (err, _packet) => err ? reject(err) : resolve()));
46
- };
47
- };
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ const log = require("npmlog");
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ return function changeAdminStatus(threadID, adminIDs, adminStatus, callback) {
8
+ if (utils.getType(threadID) !== "String") {
9
+ throw new utils.CustomError({ error: "changeAdminStatus: threadID must be a string" });
10
+ }
11
+
12
+ if (utils.getType(adminIDs) === "String") {
13
+ adminIDs = [adminIDs];
14
+ }
15
+
16
+ if (utils.getType(adminIDs) !== "Array") {
17
+ throw new utils.CustomError({ error: "changeAdminStatus: adminIDs must be an array or string" });
18
+ }
19
+
20
+ if (utils.getType(adminStatus) !== "Boolean") {
21
+ throw new utils.CustomError({ error: "changeAdminStatus: adminStatus must be a string" });
22
+ }
23
+
24
+ let resolveFunc = function () { };
25
+ let rejectFunc = function () { };
26
+ const returnPromise = new Promise(function (resolve, reject) {
27
+ resolveFunc = resolve;
28
+ rejectFunc = reject;
29
+ });
30
+
31
+ if (!callback) {
32
+ callback = function (err) {
33
+ if (err) {
34
+ return rejectFunc(err);
35
+ }
36
+ resolveFunc();
37
+ };
38
+ }
39
+
40
+ if (utils.getType(callback) !== "Function" && utils.getType(callback) !== "AsyncFunction") {
41
+ throw new utils.CustomError({ error: "changeAdminStatus: callback is not a function" });
42
+ }
43
+
44
+ const form = {
45
+ "thread_fbid": threadID
46
+ };
47
+
48
+ let i = 0;
49
+ for (const u of adminIDs) {
50
+ form[`admin_ids[${i++}]`] = u;
51
+ }
52
+ form["add"] = adminStatus;
53
+
54
+ defaultFuncs
55
+ .post("https://www.facebook.com/messaging/save_admins/?dpr=1", ctx.jar, form)
56
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
57
+ .then(function (resData) {
58
+ if (resData.error) {
59
+ switch (resData.error) {
60
+ case 1976004:
61
+ throw new utils.CustomError({ error: "Cannot alter admin status: you are not an admin.", rawResponse: resData });
62
+ case 1357031:
63
+ throw new utils.CustomError({ error: "Cannot alter admin status: this thread is not a group chat.", rawResponse: resData });
64
+ default:
65
+ throw new utils.CustomError({ error: "Cannot alter admin status: unknown error.", rawResponse: resData });
66
+ }
67
+ }
68
+
69
+ callback();
70
+ })
71
+ .catch(function (err) {
72
+ log.error("changeAdminStatus", err);
73
+ return callback(err);
74
+ });
75
+
76
+ return returnPromise;
77
+ };
78
+ };
79
+
@@ -1,41 +1,55 @@
1
- "use strict";
2
-
3
- var utils = require("../utils");
4
- var log = require("npmlog");
5
-
6
- module.exports = function (defaultFuncs, api, ctx) {
7
- return function changeArchivedStatus(threadOrThreads, archive, callback) {
8
- var resolveFunc = function () { };
9
- var rejectFunc = function () { };
10
- var returnPromise = new Promise(function (resolve, reject) {
11
- resolveFunc = resolve;
12
- rejectFunc = reject;
13
- });
14
-
15
- if (!callback) {
16
- callback = function (err) {
17
- if (err) return rejectFunc(err);
18
- resolveFunc();
19
- };
20
- }
21
-
22
- var form = {};
23
-
24
- if (utils.getType(threadOrThreads) === "Array") for (var i = 0; i < threadOrThreads.length; i++) form["ids[" + threadOrThreads[i] + "]"] = archive;
25
- else form["ids[" + threadOrThreads + "]"] = archive;
26
-
27
- defaultFuncs
28
- .post("https://www.facebook.com/ajax/mercury/change_archived_status.php", ctx.jar, form)
29
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
30
- .then(function (resData) {
31
- if (resData.error) throw resData;
32
- return callback();
33
- })
34
- .catch(function (err) {
35
- log.error("changeArchivedStatus", err);
36
- return callback(err);
37
- });
38
-
39
- return returnPromise;
40
- };
41
- };
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ const log = require("npmlog");
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ return function changeArchivedStatus(threadOrThreads, archive, callback) {
8
+ let resolveFunc = function () { };
9
+ let rejectFunc = function () { };
10
+ const returnPromise = new Promise(function (resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+
15
+ if (!callback) {
16
+ callback = function (err) {
17
+ if (err) {
18
+ return rejectFunc(err);
19
+ }
20
+ resolveFunc();
21
+ };
22
+ }
23
+
24
+ const form = {};
25
+
26
+ if (utils.getType(threadOrThreads) === "Array") {
27
+ for (let i = 0; i < threadOrThreads.length; i++) {
28
+ form["ids[" + threadOrThreads[i] + "]"] = archive;
29
+ }
30
+ } else {
31
+ form["ids[" + threadOrThreads + "]"] = archive;
32
+ }
33
+
34
+ defaultFuncs
35
+ .post(
36
+ "https://www.facebook.com/ajax/mercury/change_archived_status.php",
37
+ ctx.jar,
38
+ form
39
+ )
40
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
41
+ .then(function (resData) {
42
+ if (resData.error) {
43
+ throw resData;
44
+ }
45
+
46
+ return callback();
47
+ })
48
+ .catch(function (err) {
49
+ log.error("changeArchivedStatus", err);
50
+ return callback(err);
51
+ });
52
+
53
+ return returnPromise;
54
+ };
55
+ };
@@ -0,0 +1,126 @@
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ const log = require("npmlog");
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ function handleUpload(image, callback) {
8
+ const uploads = [];
9
+
10
+ const form = {
11
+ profile_id: ctx.i_userID || ctx.userID,
12
+ photo_source: 57,
13
+ av: ctx.i_userID || ctx.userID,
14
+ file: image
15
+ };
16
+
17
+ uploads.push(
18
+ defaultFuncs
19
+ .postFormData(
20
+ "https://www.facebook.com/profile/picture/upload/",
21
+ ctx.jar,
22
+ form,
23
+ {}
24
+ )
25
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
26
+ .then(function (resData) {
27
+ if (resData.error) {
28
+ throw resData;
29
+ }
30
+ return resData;
31
+ })
32
+ );
33
+
34
+ // resolve all promises
35
+ Promise
36
+ .all(uploads)
37
+ .then(function (resData) {
38
+ callback(null, resData);
39
+ })
40
+ .catch(function (err) {
41
+ log.error("handleUpload", err);
42
+ return callback(err);
43
+ });
44
+ }
45
+
46
+ return function changeAvatar(image, caption = "", timestamp = null, callback) {
47
+ let resolveFunc = function () { };
48
+ let rejectFunc = function () { };
49
+ const returnPromise = new Promise(function (resolve, reject) {
50
+ resolveFunc = resolve;
51
+ rejectFunc = reject;
52
+ });
53
+
54
+ if (!timestamp && utils.getType(caption) === "Number") {
55
+ timestamp = caption;
56
+ caption = "";
57
+ }
58
+
59
+ if (!timestamp && !callback && (utils.getType(caption) == "Function" || utils.getType(caption) == "AsyncFunction")) {
60
+ callback = caption;
61
+ caption = "";
62
+ timestamp = null;
63
+ }
64
+
65
+ if (!callback) callback = function (err, data) {
66
+ if (err) {
67
+ return rejectFunc(err);
68
+ }
69
+ resolveFunc(data);
70
+ };
71
+
72
+ if (!utils.isReadableStream(image))
73
+ return callback("Image is not a readable stream");
74
+
75
+ handleUpload(image, function (err, payload) {
76
+ if (err) {
77
+ return callback(err);
78
+ }
79
+
80
+ const form = {
81
+ av: ctx.i_userID || ctx.userID,
82
+ fb_api_req_friendly_name: "ProfileCometProfilePictureSetMutation",
83
+ fb_api_caller_class: "RelayModern",
84
+ doc_id: "5066134240065849",
85
+ variables: JSON.stringify({
86
+ input: {
87
+ caption,
88
+ existing_photo_id: payload[0].payload.fbid,
89
+ expiration_time: timestamp,
90
+ profile_id: ctx.i_userID || ctx.userID,
91
+ profile_pic_method: "EXISTING",
92
+ profile_pic_source: "TIMELINE",
93
+ scaled_crop_rect: {
94
+ height: 1,
95
+ width: 1,
96
+ x: 0,
97
+ y: 0
98
+ },
99
+ skip_cropping: true,
100
+ actor_id: ctx.i_userID || ctx.userID,
101
+ client_mutation_id: Math.round(Math.random() * 19).toString()
102
+ },
103
+ isPage: false,
104
+ isProfile: true,
105
+ scale: 3
106
+ })
107
+ };
108
+
109
+ defaultFuncs
110
+ .post("https://www.facebook.com/api/graphql/", ctx.jar, form)
111
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
112
+ .then(function (resData) {
113
+ if (resData.errors) {
114
+ throw resData;
115
+ }
116
+ return callback(null, resData[0].data.profile_picture_set);
117
+ })
118
+ .catch(function (err) {
119
+ log.error("changeAvatar", err);
120
+ return callback(err);
121
+ });
122
+ });
123
+
124
+ return returnPromise;
125
+ };
126
+ };