-
Notifications
You must be signed in to change notification settings - Fork 4
/
twitter-task.js
288 lines (259 loc) · 8.32 KB
/
twitter-task.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
const Twitter = require('./adapters/twitter/twitter.js');
const Data = require('./model/data');
const dotenv = require('dotenv');
const { default: axios } = require('axios');
const {KoiiStorageClient} = require('@_koii/storage-task-sdk');
const { namespaceWrapper } = require('./namespaceWrapper.js');
const { CID } = require('multiformats/cid');
async function isValidCID(cid) {
try {
CID.parse(cid);
return true;
} catch (error) {
return false;
}
}
dotenv.config();
/**
* TwitterTask is a class that handles the Twitter searcher and validator
*
* @description TwitterTask is a class that handles the Twitter searcher and validator
* In this task, the searcher asynchronously populates a database, which is later
* read by the validator. The validator then uses the database to prepare a submission CID
* for the current round, and submits that for rewards.
*
* Four main functions control this process:
* @search searchs Twitter and populates the database
* @validate verifies the submissions of other nodes
* @getRoundCID returns the submission for a given round
* @stop stops the searcher
*
* @param {function} getRound - a function that returns the current round
* @param {number} round - the current round
* @param {string} searchTerm - the search term to use for the searcher
* @param {string} adapter - the adapter to use for the searcher
* @param {string} db - the database to use for the searcher
*
* @returns {TwitterTask} - a TwitterTask object
*
*/
class TwitterTask {
constructor(getRound, round) {
this.round = round;
this.lastRoundCheck = Date.now();
this.isRunning = false;
this.searchTerm = [];
this.adapter = null;
this.db = new Data('db', []);
this.db.initializeData();
this.initialize();
this.setAdapter = async () => {
const username = process.env.TWITTER_USERNAME;
const password = process.env.TWITTER_PASSWORD;
const phone = process.env.TWITTER_PHONE;
if (!username || !password) {
throw new Error(
'Environment variables TWITTER_USERNAME and/or TWITTER_PASSWORD are not set',
);
}
let credentials = {
username: username,
password: password,
phone: phone,
};
this.adapter = new Twitter(credentials, this.db, 3);
await this.adapter.negotiateSession();
};
this.start();
}
async initialize() {
console.log('initializing twitter task');
this.searchTerm = await this.fetchSearchTerms();
//Store this round searchTerm
console.log('creating search term', this.searchTerm, this.round);
this.db.createSearchTerm(this.searchTerm, this.round);
}
/**
* fetchSearchTerms
* @description return the search terms to use for the searcher
* @returns {array} - an array of search terms
*/
async fetchSearchTerms() {
let keyword;
try {
const submitterAccountKeyPair = (
await namespaceWrapper.getSubmitterAccount()
).publicKey;
const key = submitterAccountKeyPair.toBase58();
console.log('submitter key', key);
const response = await axios.get('http://localhost:3000/keywords', {
params: {
key: key,
},
});
// console.log('keywords from middle server', response.data);
keyword = response.data;
} catch (error) {
console.log(
'No Keywords from middle server, loading local keywords.json',
);
const wordsList = require('./top1000words.json');
const randomIndex = Math.floor(Math.random() * wordsList.length);
keyword = wordsList[randomIndex]; // Load local JSON data
}
return encodeURIComponent(keyword);
}
/**
* strat
* @description starts the searcher
*
* @returns {void}
*
*/
async start() {
await this.setAdapter();
this.isRunning = true;
let query = {
limit: 100, // unused
searchTerm: this.searchTerm,
query: `https://x.com/search?q=${this.searchTerm}&src=typed_query&f=live`,
// https://x.com/any/status/<tweets_id>
depth: 3,
round: this.round,
recursive: true,
round: this.round,
};
this.adapter.search(query); // let it ride
}
/**
* stop
* @description stops the searcher
*
* @returns {void}
*/
async stop() {
this.isRunning = false;
this.adapter.stop();
}
/**
* getRoundCID
* @param {*} roundID
* @returns
*/
async getRoundCID(roundID) {
console.log('starting submission prep for ');
let result = await this.adapter.getSubmissionCID(roundID);
console.log('returning round CID', result, 'for round', roundID);
return result;
}
/**
* getJSONofCID
* @description gets the JSON of a CID
* @param {*} cid
* @returns
*/
async getJSONofCID(cid) {
return await getJSONFromCID(cid, 'dataList.json');
}
/**
* validate
* @description validates a round of results from another node against the Twitter API
* @param {*} proofCid
* @returns
*/
async validate(proofCid, round) {
// in order to validate, we need to take the proofCid
// and go get the results from IPFS
try {
let data = await getJSONFromCID(proofCid, 'dataList.json'); // check this
// console.log(`validate got results for CID: ${ proofCid } for round ${ roundID }`, data, typeof(data), data[0]);
// Check for duplicate item IDs
let idSet = new Set();
let duplicatedIDNumber = 0;
for (let item of data) {
if (idSet.has(item.id)) {
console.log('Duplicate Item ID found: ', item.id);
duplicatedIDNumber += 1;
}
idSet.add(item.id);
}
if (duplicatedIDNumber > 10){
console.log(`Detected Potential Risk ; Duplicated ID is ${duplicatedIDNumber}`);
}else{
console.log(`Duplicated ID Check Passed ; Duplicated ID numebr is ${duplicatedIDNumber}`);
}
let proofThreshold = 8; // an arbitrary number of records to check
let passedNumber = 0;
if (data && data !== null && data.length > 0) {
for (let i = 0; i < proofThreshold; i++) {
console.log(`Checking the ${i} th tweet.`)
let randomIndex = Math.floor(Math.random() * data.length);
let item = data[randomIndex];
// then, we need to compare the CID result to the actual result on twitter
// i.e.
// console.log('item was', item);
if (item.id) {
// }
await new Promise(resolve => setTimeout(resolve, 30000));
const result = await this.adapter.verify(item.data.tweets_id, item.data, round);
console.log('Result from verify', result);
if(result){passedNumber += 1;}
} else {
console.log('Invalid Item ID: ', item.id);
continue;
}
}
if (passedNumber >= 5){
console.log(passedNumber,"is passedNumber")
return true;
}else{
console.log(passedNumber,"is passedNumber")
return false;
}
} else {
console.log('no data from proof CID');
}
// if none of the random checks fail, return true
return true;
} catch (e) {
console.log('error in validate', e);
return true;
}
}
}
module.exports = TwitterTask;
/**
* getJSONFromCID
* @description gets the JSON from a CID
* @param {*} cid
* @returns promise<JSON>
*/
const sleep = (ms) => new Promise(resolve => setTimeout(resolve, ms));
const getJSONFromCID = async (
cid,
fileName,
retries = 3
) => {
const validateCID = await isValidCID(cid)
if (!validateCID) {
console.log(`Invalid CID: ${cid}`);
return null;
}
const client = new KoiiStorageClient(undefined, undefined, false);
for (let attempt = 1; attempt <= retries; attempt++) {
try {
const blob = await client.getFile(cid, fileName);
const text = await blob.text(); // Convert Blob to text
const data = JSON.parse(text); // Parse text to JSON
return data;
} catch (error) {
console.log(`Attempt ${attempt}: Error fetching file from Koii IPFS: ${error.message}`);
if (attempt === retries) {
throw new Error(`Failed to fetch file after ${retries} attempts`);
}
// Optionally, you can add a delay between retries
await new Promise(resolve => setTimeout(resolve, 3000)); // 3-second delay
}
}
return null;
};