mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 02:20:54 +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) {
|
while (apiAttempts < API_ATTEMPTS) {
|
||||||
/* Generate a random CSRF token, Twitter just cares that header and cookie match,
|
/* 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 csrfToken = crypto.randomUUID().replace(/-/g, '');
|
||||||
|
|
||||||
const headers: Record<string, string> = {
|
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) */
|
/* Renders card for polls and non-Twitter video embeds (i.e. YouTube) */
|
||||||
export const renderCard = (
|
export const renderCard = (
|
||||||
card: GraphQLTwitterStatus['card']
|
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)) {
|
if (!Array.isArray(card.legacy?.binding_values)) {
|
||||||
return {};
|
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 {};
|
return {};
|
||||||
};
|
};
|
||||||
|
|
|
@ -20,7 +20,8 @@ export const processMedia = (media: TweetMedia): APIPhoto | APIVideo | null => {
|
||||||
width: media.original_info?.width,
|
width: media.original_info?.width,
|
||||||
height: media.original_info?.height,
|
height: media.original_info?.height,
|
||||||
format: bestVariant?.content_type || '',
|
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;
|
return null;
|
||||||
|
|
|
@ -230,6 +230,33 @@ export const buildAPITwitterStatus = async (
|
||||||
if (card.poll) {
|
if (card.poll) {
|
||||||
apiStatus.poll = 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 {
|
} else {
|
||||||
/* Determine if the status contains a YouTube link (either youtube.com or youtu.be) so we can include it */
|
/* 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&]+)/;
|
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;
|
thumbnail_url: string;
|
||||||
format: string;
|
format: string;
|
||||||
duration: number;
|
duration: number;
|
||||||
|
variants: TweetMediaFormat[];
|
||||||
}
|
}
|
||||||
|
|
||||||
interface APIMosaicPhoto extends APIMedia {
|
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'
|
| 'last_updated_datetime_utc'
|
||||||
| 'duration_minutes'
|
| 'duration_minutes'
|
||||||
| 'api'
|
| 'api'
|
||||||
| 'card_url';
|
| 'card_url'
|
||||||
|
| 'unified_card';
|
||||||
value:
|
value:
|
||||||
| {
|
| {
|
||||||
string_value: string; // "Option text"
|
string_value: string; // "Option text"
|
||||||
|
|
Loading…
Add table
Reference in a new issue