mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-04 18:10:56 +01:00
Pull media from Twitter for Advertisers cards
This commit is contained in:
parent
e2f56af1ee
commit
b112fd86f3
6 changed files with 70 additions and 4 deletions
|
@ -63,7 +63,7 @@ export const twitterFetch = async (
|
|||
|
||||
while (apiAttempts < API_ATTEMPTS) {
|
||||
/* Generate a random CSRF token, Twitter just cares that header and cookie match,
|
||||
REST can use shorter csrf tokens (32 bytes) but graphql prefers 160 bytes */
|
||||
REST can use shorter csrf tokens (32 bytes) but graphql uses a 160 byte one. */
|
||||
const csrfToken = crypto.randomUUID().replace(/-/g, '');
|
||||
|
||||
const headers: Record<string, string> = {
|
||||
|
|
|
@ -3,7 +3,11 @@ import { calculateTimeLeftString } from './pollTime';
|
|||
/* Renders card for polls and non-Twitter video embeds (i.e. YouTube) */
|
||||
export const renderCard = (
|
||||
card: GraphQLTwitterStatus['card']
|
||||
): { poll?: APIPoll; external_media?: APIExternalMedia } => {
|
||||
): {
|
||||
poll?: APIPoll;
|
||||
external_media?: APIExternalMedia;
|
||||
media?: { videos: TweetMedia[]; photos: TweetMedia[] };
|
||||
} => {
|
||||
if (!Array.isArray(card.legacy?.binding_values)) {
|
||||
return {};
|
||||
}
|
||||
|
@ -58,5 +62,37 @@ export const renderCard = (
|
|||
};
|
||||
}
|
||||
|
||||
if (binding_values.unified_card?.string_value) {
|
||||
try {
|
||||
const card = JSON.parse(binding_values.unified_card.string_value);
|
||||
const mediaEntities = card?.media_entities as Record<string, TweetMedia>;
|
||||
|
||||
if (mediaEntities) {
|
||||
const media = {
|
||||
videos: [] as TweetMedia[],
|
||||
photos: [] as TweetMedia[]
|
||||
};
|
||||
Object.keys(mediaEntities).forEach(key => {
|
||||
const mediaItem = mediaEntities[key];
|
||||
switch (mediaItem.type) {
|
||||
case 'photo':
|
||||
media.photos.push(mediaItem);
|
||||
break;
|
||||
case 'animated_gif':
|
||||
case 'video':
|
||||
media.videos.push(mediaItem);
|
||||
break;
|
||||
}
|
||||
});
|
||||
|
||||
console.log('media', media);
|
||||
|
||||
return { media: media };
|
||||
}
|
||||
} catch (e) {
|
||||
console.error('Failed to parse unified card JSON', e);
|
||||
}
|
||||
}
|
||||
|
||||
return {};
|
||||
};
|
||||
|
|
|
@ -20,7 +20,8 @@ export const processMedia = (media: TweetMedia): APIPhoto | APIVideo | null => {
|
|||
width: media.original_info?.width,
|
||||
height: media.original_info?.height,
|
||||
format: bestVariant?.content_type || '',
|
||||
type: media.type === 'animated_gif' ? 'gif' : 'video'
|
||||
type: media.type === 'animated_gif' ? 'gif' : 'video',
|
||||
variants: media.video_info?.variants ?? []
|
||||
};
|
||||
}
|
||||
return null;
|
||||
|
|
|
@ -230,6 +230,33 @@ export const buildAPITwitterStatus = async (
|
|||
if (card.poll) {
|
||||
apiStatus.poll = card.poll;
|
||||
}
|
||||
/* TODO: Right now, we push them after native photos and videos but should we prepend them instead? */
|
||||
if (card.media) {
|
||||
if (card.media.videos) {
|
||||
card.media.videos.forEach(video => {
|
||||
const mediaObject = processMedia(video) as APIVideo;
|
||||
if (mediaObject) {
|
||||
apiStatus.media.all = apiStatus.media?.all ?? [];
|
||||
apiStatus.media?.all?.push(mediaObject);
|
||||
apiStatus.media.videos = apiStatus.media?.videos ?? [];
|
||||
apiStatus.media.videos?.push(mediaObject);
|
||||
}
|
||||
});
|
||||
}
|
||||
if (card.media.photos) {
|
||||
card.media.photos.forEach(photo => {
|
||||
const mediaObject = processMedia(photo) as APIPhoto;
|
||||
if (mediaObject) {
|
||||
apiStatus.media.all = apiStatus.media?.all ?? [];
|
||||
apiStatus.media?.all?.push(mediaObject);
|
||||
apiStatus.media.photos = apiStatus.media?.photos ?? [];
|
||||
apiStatus.media.photos?.push(mediaObject);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
} else {
|
||||
/* Determine if the status contains a YouTube link (either youtube.com or youtu.be) so we can include it */
|
||||
const youtubeIdRegex = /(https?:\/\/)?(www\.)?(youtube\.com\/watch\?v=|youtu\.be\/)([^\s&]+)/;
|
||||
|
|
1
src/types/types.d.ts
vendored
1
src/types/types.d.ts
vendored
|
@ -101,6 +101,7 @@ interface APIVideo extends APIMedia {
|
|||
thumbnail_url: string;
|
||||
format: string;
|
||||
duration: number;
|
||||
variants: TweetMediaFormat[];
|
||||
}
|
||||
|
||||
interface APIMosaicPhoto extends APIMedia {
|
||||
|
|
3
src/types/vendor/twitter.d.ts
vendored
3
src/types/vendor/twitter.d.ts
vendored
|
@ -420,7 +420,8 @@ type GraphQLTwitterStatus = {
|
|||
| 'last_updated_datetime_utc'
|
||||
| 'duration_minutes'
|
||||
| 'api'
|
||||
| 'card_url';
|
||||
| 'card_url'
|
||||
| 'unified_card';
|
||||
value:
|
||||
| {
|
||||
string_value: string; // "Option text"
|
||||
|
|
Loading…
Add table
Reference in a new issue