mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-20 22:42:53 -05:00
e6856f6ac6
refs https://github.com/TryGhost/Team/issues/1001 We fall back to existing behaviour if no API key is present, or if there is an error communicating with the Twitter API. We're also currently requesting all the data, which will be thinned down once we understand what we need. This also includes a custom renderer for embeds of type "twitter" which will be used to output the custom HTML for emails
80 lines
3 KiB
JavaScript
80 lines
3 KiB
JavaScript
const {extract} = require('oembed-parser');
|
|
|
|
/**
|
|
* @typedef {import('./oembed').ICustomProvider} ICustomProvider
|
|
* @typedef {import('./oembed').IExternalRequest} IExternalRequest
|
|
*/
|
|
|
|
const TWITTER_PATH_REGEX = /\/status\/(\d+)/;
|
|
|
|
/**
|
|
* @implements ICustomProvider
|
|
*/
|
|
class TwitterOEmbedProvider {
|
|
/**
|
|
* @param {object} dependencies
|
|
*/
|
|
constructor(dependencies) {
|
|
this.dependencies = dependencies;
|
|
}
|
|
|
|
/**
|
|
* @param {URL} url
|
|
* @returns {Promise<boolean>}
|
|
*/
|
|
async canSupportRequest(url) {
|
|
return url.host === 'twitter.com' && TWITTER_PATH_REGEX.test(url.pathname);
|
|
}
|
|
|
|
/**
|
|
* @param {URL} url
|
|
* @param {IExternalRequest} externalRequest
|
|
*
|
|
* @returns {Promise<object>}
|
|
*/
|
|
async getOEmbedData(url, externalRequest) {
|
|
const [match, tweetId] = url.pathname.match(TWITTER_PATH_REGEX);
|
|
if (!match) {
|
|
return null;
|
|
}
|
|
|
|
/** @type {object} */
|
|
const oembedData = await extract(url.href);
|
|
|
|
if (this.dependencies.config.bearerToken) {
|
|
const query = {
|
|
expansions: ['attachments.poll_ids', 'attachments.media_keys', 'author_id', 'entities.mentions.username', 'geo.place_id', 'in_reply_to_user_id', 'referenced_tweets.id', 'referenced_tweets.id.author_id'],
|
|
'media.fields': ['duration_ms', 'height', 'media_key', 'preview_image_url', 'type', 'url', 'width', 'public_metrics', 'alt_text'],
|
|
'place.fields': ['contained_within', 'country', 'country_code', 'full_name', 'geo', 'id', 'name', 'place_type'],
|
|
'poll.fields': ['duration_minutes', 'end_datetime', 'id', 'options', 'voting_status'],
|
|
'tweet.fields': ['attachments', 'author_id', 'context_annotations', 'conversation_id', 'created_at', 'entities', 'geo', 'id', 'in_reply_to_user_id', 'lang', 'public_metrics', 'possibly_sensitive', 'referenced_tweets', 'reply_settings', 'source', 'text', 'withheld'],
|
|
'user.fields': ['created_at', 'description', 'entities', 'id', 'location', 'name', 'pinned_tweet_id', 'profile_image_url', 'protected', 'public_metrics', 'url', 'username', 'verified', 'withheld']
|
|
};
|
|
|
|
const queryString = Object.keys(query).map((key) => {
|
|
return `${key}=${query[key].join(',')}`;
|
|
}).join('&');
|
|
|
|
try {
|
|
const result = await externalRequest(`https://api.twitter.com/2/tweets/${tweetId}?${queryString}`, {
|
|
responseType: 'json',
|
|
headers: {
|
|
Authorization: `Bearer ${this.dependencies.config.bearerToken}`
|
|
}
|
|
});
|
|
|
|
const body = JSON.parse(result.body);
|
|
|
|
oembedData.tweet_data = body.data;
|
|
} catch (err) {
|
|
this.dependencies.logging.error(err);
|
|
}
|
|
}
|
|
|
|
oembedData.type = 'twitter';
|
|
|
|
return oembedData;
|
|
}
|
|
}
|
|
|
|
module.exports = TwitterOEmbedProvider;
|