Remove Twitter API 🔪

(closes #7750)
This commit is contained in:
Bryan Housel 2023-04-27 09:55:51 -04:00
parent 199c38bfc4
commit fc370cb535
5 changed files with 12 additions and 149 deletions

View file

@ -157,7 +157,6 @@ export default function Category(props) {
<th>Wikidata Name/Description<br/>Official Website<br/>Social Links</th>
<th className='logo'>Commons Logo</th>
<th className='logo'>Facebook Logo</th>
<th className='logo'>Twitter Logo</th>
</tr>
</thead>

View file

@ -31,7 +31,7 @@ export default function CategoryInstructions(props) {
// Flags don't have Facebook accounts
let social = '';
if (t !== 'flags') {
social = `You can add the ${itemType}'s Facebook or Twitter usernames, and this project will pick up the logos later.`;
social = `You can add the ${itemType}'s Facebook username, and this project will pick up the logos later.`;
}
return (

View file

@ -184,7 +184,6 @@ relation[${k}=${v}][network:wikidata=${qid}]
</td>
<td className='logo'>{ logo(logos.wikidata) }</td>
<td className='logo'>{ fblogo(identities.facebook, logos.facebook) }</td>
<td className='logo'>{ logo(logos.twitter) }</td>
</tr>
);
}
@ -287,7 +286,7 @@ relation[${k}=${v}][network:wikidata=${qid}]
/* Add an <hr/> line break only if addational information will be displayed. */
if (item.matchNames || item.matchTags || item.note || item.preserveTags || item.fromTemplate)
result += '<hr/>';
/* Are the items being drawn from a template? 'item.fromTemplate' is set to true in nsi.json if templated. */
if (item.fromTemplate) {
let url,text;

View file

@ -134,7 +134,6 @@
"safe-regex": "^2.1.1",
"shelljs": "^0.8.5",
"tap": "^16.3.4",
"twitter": "^1.7.1",
"whatwg-fetch": "^3.6.2",
"wikibase-edit": "^5.0.3",
"wikibase-sdk": "^8.0.1",

View file

@ -10,7 +10,6 @@ import localeCompare from 'locale-compare';
import LocationConflation from '@rapideditor/location-conflation';
import shell from 'shelljs';
import stringify from '@aitodotai/json-stringify-pretty-compact';
import Twitter from 'Twitter';
import wikibase from 'wikibase-sdk';
import wikibaseEdit from 'wikibase-edit';
const withLocale = localeCompare('en-US');
@ -48,30 +47,10 @@ const DRYRUN = false;
// First, try to load the user's secrets.
// This is optional but needed if you want this script to:
// - connect to the Twitter API to fetch logos
// - connect to the Wikibase API to update NSI identifiers.
//
// `secrets.json` looks like this:
// {
// "twitter": [
// {
// "name": "name-suggestion-index-staging",
// "app_id": "16186858",
// "bearer_token": "AAAAAAAAAAAAAAAAAAA…",
// "twitter_consumer_key": "",
// "twitter_consumer_secret": "",
// "twitter_access_token_key": "",
// "twitter_access_token_secret": ""
// }, {
// "name": "name-suggestion-index-dev",
// "app_id": "16186940",
// "bearer_token": "AAAAAAAAAAAAAAAAAAA…",
// "twitter_consumer_key": "",
// "twitter_consumer_secret": "",
// "twitter_access_token_key": "",
// "twitter_access_token_secret": ""
// }
// ],
// "wikibase": {
// "username": "my-wikidata-username",
// "password": "my-wikidata-password"
@ -88,45 +67,15 @@ try {
_secrets = JSON5.parse(fs.readFileSync('./secrets.json', 'utf8'));
} catch (err) { /* ignore */ }
if (_secrets && !_secrets.twitter && !_secrets.wikibase) {
if (_secrets && !_secrets.wikibase) {
console.error(chalk.red('WHOA!'));
console.error(chalk.yellow('The `config/secrets.json` file format has changed a bit.'));
console.error(chalk.yellow('We were expecting to find `twitter` or `wikibase` properties.'));
console.error(chalk.yellow('We were expecting to find a `wikibase` property.'));
console.error(chalk.yellow('Check `scripts/build_wikidata.js` for details...'));
console.error('');
process.exit(1);
}
// To fetch Twitter logos, sign up for API credentials at https://apps.twitter.com/
// and put them into `config/secrets.json`
let _twitterAPIs = [];
let _twitterAPIIndex = 0;
if (_secrets && _secrets.twitter) {
_twitterAPIs = _secrets.twitter.map((s, i) => {
let props;
// if (s.bearer_token) { // use a bearer token if we have it
// props = {
// consumer_key: s.twitter_consumer_key,
// consumer_secret: s.twitter_consumer_secret,
// bearer_token: s.bearer_token
// };
// } else {
props = {
consumer_key: s.twitter_consumer_key,
consumer_secret: s.twitter_consumer_secret,
access_token_key: s.twitter_access_token_key,
access_token_secret: s.twitter_access_token_secret
};
// }
return {
name: s.name || i.toString(),
client: new Twitter(props)
};
});
}
// To update wikidata
// add your username/password into `config/secrets.json`
@ -249,11 +198,10 @@ function doFetch(index) {
//
// `processEntities`
// Here we process the fetched results from the Wikidata API,
// then schedule followup API calls to the Twitter/Facebook APIs,
// then schedule followup API calls to the Facebook API,
// then eventually resolves when all that work is done.
//
function processEntities(result) {
let twitterQueue = [];
let facebookQueue = [];
let wbEditQueue = [];
@ -333,7 +281,6 @@ function processEntities(result) {
const twitterUser = getClaimValue(entity, 'P2002');
if (twitterUser) {
target.identities.twitter = twitterUser;
twitterQueue.push({ qid: qid, username: twitterUser }); // queue logo fetch
}
// P2003 - Instagram ID
@ -496,15 +443,8 @@ function processEntities(result) {
}); // foreach qid
if (_twitterAPIs.length && twitterQueue.length) {
return checkTwitterRateLimit(twitterQueue.length)
.then(() => Promise.all( twitterQueue.map(obj => fetchTwitterUserDetails(obj.qid, obj.username)) ))
.then(() => Promise.all( facebookQueue.map(obj => fetchFacebookLogo(obj.qid, obj.username)) ))
.then(() => processWbEditQueue(wbEditQueue));
} else {
return Promise.all( facebookQueue.map(obj => fetchFacebookLogo(obj.qid, obj.username)) )
.then(() => processWbEditQueue(wbEditQueue));
}
return Promise.all( facebookQueue.map(obj => fetchFacebookLogo(obj.qid, obj.username)) )
.then(() => processWbEditQueue(wbEditQueue));
}
@ -546,9 +486,10 @@ function getClaimValue(entity, prop) {
// `finish`
// Wrap up, write files
// - wikidata.json
// - dissolved.json
// Wrap up, write files:
// - `warnings.json`
// - `wikidata.json`
// - `dissolved.json`
//
function finish() {
const START = '🏗 ' + chalk.yellow('Writing output files');
@ -557,30 +498,11 @@ function finish() {
console.log(START);
console.time(END);
// update `wikidata.json` and `dissolved.json`
let origWikidata;
let dissolved = {};
try {
origWikidata = JSON5.parse(fs.readFileSync('./dist/wikidata.json', 'utf8')).wikidata;
} catch (err) {
origWikidata = {};
}
Object.keys(_wikidata).forEach(qid => {
let target = _wikidata[qid];
// if we haven't been able to access the Twitter API, don't overwrite the Twitter data - #3569
if (!_twitterAPIs.length) {
const origTarget = origWikidata[qid];
['identities', 'logos'].forEach(prop => {
const origTwitter = origTarget && origTarget[prop] && origTarget[prop].twitter;
if (origTwitter) {
target[prop] = target[prop] || {};
target[prop].twitter = origTwitter;
}
});
}
// sort the properties that we are keeping..
['identities', 'logos', 'dissolutions'].forEach(prop => {
if (target[prop] && Object.keys(target[prop]).length) {
@ -610,7 +532,7 @@ function finish() {
console.timeEnd(END);
// output whatever warnings we've gathered
// `console.warn` whatever warnings we've gathered
if (_warnings.length) {
console.log(chalk.yellow.bold(`\nWarnings:`));
_warnings.forEach(warning => console.warn(chalk.yellow(warning.qid.padEnd(12)) + chalk.red(warning.msg)));
@ -618,62 +540,6 @@ function finish() {
}
// check Twitter rate limit status
// https://developer.twitter.com/en/docs/developer-utilities/rate-limit-status/api-reference/get-application-rate_limit_status
// rate limit: 900calls / 15min
function checkTwitterRateLimit(need) {
_twitterAPIIndex = (_twitterAPIIndex + 1) % _twitterAPIs.length; // cycle to next client
const twitterAPI = _twitterAPIs[_twitterAPIIndex];
const which = twitterAPI.name;
return twitterAPI.client
.get('application/rate_limit_status', { resources: 'users' })
.then(result => {
const now = Date.now() / 1000;
const stats = result.resources.users['/users/:id'];
const resetSec = Math.ceil(stats.reset - now) + 30; // +30sec in case server time is different
console.log(chalk.green.bold(`Twitter rate status '${which}': need ${need}, remaining ${stats.remaining}, resets in ${resetSec} seconds...`));
if (need > stats.remaining) {
const delaySec = clamp(resetSec, 10, 60);
console.log(chalk.green.bold(`Twitter rate limit exceeded, pausing for ${delaySec} seconds...`));
return delaySec;
} else {
return 0;
}
})
.then(sec => {
if (sec > 0) {
return delay(sec * 1000)
.then(() => checkTwitterRateLimit(need));
} else {
return Promise.resolve();
}
})
.catch(e => {
console.warn(chalk.green.bold(`Error: Twitter rate limit: ` + JSON.stringify(e)));
});
}
// https://developer.twitter.com/en/docs/accounts-and-users/user-profile-images-and-banners.html
// https://developer.twitter.com/en/docs/accounts-and-users/follow-search-get-users/api-reference/get-users-show
function fetchTwitterUserDetails(qid, username) {
const target = _wikidata[qid];
const twitterAPI = _twitterAPIs[_twitterAPIIndex];
return twitterAPI.client
.get('users/show', { screen_name: username })
.then(user => {
target.logos.twitter = user.profile_image_url_https.replace('_normal', '_bigger');
})
.catch(e => {
const warning = { qid: qid, msg: `Twitter username @${username}: ${JSON.stringify(e)}` };
console.warn(chalk.yellow(warning.qid.padEnd(12)) + chalk.red(warning.msg));
_warnings.push(warning);
});
}
// https://developers.facebook.com/docs/graph-api/reference/user/picture/
function fetchFacebookLogo(qid, username) {
let target = _wikidata[qid];