diff --git a/src/api.ts b/src/api.ts index 65d0b0c..93ab764 100644 --- a/src/api.ts +++ b/src/api.ts @@ -23,6 +23,7 @@ const processMedia = (media: TweetMedia): APIPhoto | APIVideo | null => { return { url: bestVariant?.url || '', thumbnail_url: media.media_url_https, + duration: (media.video_info?.duration_millis || 0) / 1000, width: media.original_info.width, height: media.original_info.height, format: bestVariant?.content_type || '', @@ -93,6 +94,16 @@ const populateTweetProperties = async ( apiTweet.twitter_card = 'player'; apiTweet.media = apiTweet.media || {}; apiTweet.media.video = mediaObject as APIVideo; + apiTweet.media.videos = apiTweet.media.videos || []; + apiTweet.media.videos.push(mediaObject); + + apiTweet.media.video = { + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + // @ts-expect-error + WARNING: + 'video is deprecated and will be removed. Please use videos[0] instead.', + ...mediaObject + }; } } }); diff --git a/src/linkFixer.ts b/src/linkFixer.ts index 80b7925..17739b2 100644 --- a/src/linkFixer.ts +++ b/src/linkFixer.ts @@ -5,7 +5,7 @@ export const linkFixer = (tweet: TweetPartial, text: string): string => { text = text.replace(url.url, url.expanded_url); }); - text = text.replace(/ ?https:\/\/t\.co\/\w{10}/, ''); + text = text.replace(/ ?https:\/\/t\.co\/\w{10}/g, ''); } return text; diff --git a/src/status.ts b/src/status.ts index 34cc550..29fbeab 100644 --- a/src/status.ts +++ b/src/status.ts @@ -49,10 +49,11 @@ export const handleStatus = async ( if (flags?.direct && tweet.media) { let redirectUrl: string | null = null; - if (tweet.media.video) { - redirectUrl = tweet.media.video.url; + if (tweet.media.videos) { + const { videos } = tweet.media; + redirectUrl = (videos[(mediaNumber || 1) - 1] || videos[0]).url; } else if (tweet.media.photos) { - const photos = tweet.media.photos; + const { photos } = tweet.media; redirectUrl = (photos[(mediaNumber || 1) - 1] || photos[0]).url; } if (redirectUrl) { @@ -68,7 +69,7 @@ export const handleStatus = async ( let authorText = getAuthorText(tweet) || Strings.DEFAULT_AUTHOR_TEXT; const engagementText = authorText.replace(/ {4}/g, ' '); - const siteName = Constants.BRANDING_NAME; + let siteName = Constants.BRANDING_NAME; let newText = tweet.text; const headers = [ @@ -102,14 +103,15 @@ export const handleStatus = async ( } /* Video renderer */ - if (tweet.media?.video) { + if (tweet.media?.videos) { authorText = newText || ''; if (tweet?.translation) { authorText = tweet.translation?.text || ''; } - const { video } = tweet.media; + const { videos } = tweet.media; + const video = videos[(mediaNumber || 1) - 1]; /* Multiplying by 0.5 is an ugly hack to fix Discord disliking videos that are too large lol */ @@ -123,6 +125,24 @@ export const handleStatus = async ( sizeMultiplier = 2; } + const videoCounter = Strings.VIDEO_COUNT.format({ + number: String(videos.indexOf(video) + 1), + total: String(videos.length) + }); + + authorText = + authorText === Strings.DEFAULT_AUTHOR_TEXT + ? videoCounter + : `${authorText}${authorText ? ' ― ' : ''}${videoCounter}`; + + let siteName = `${Constants.BRANDING_NAME} - ${videoCounter}`; + + if (engagementText) { + siteName = `${Constants.BRANDING_NAME} - ${engagementText} - ${videoCounter}`; + } + + headers.push(``); + headers.push( ``, ``, @@ -160,15 +180,13 @@ export const handleStatus = async ( authorText = authorText === Strings.DEFAULT_AUTHOR_TEXT ? photoCounter - : `${authorText} ― ${photoCounter}`; + : `${authorText}${authorText ? ' ― ' : ''}${photoCounter}`; - let siteName = `${Constants.BRANDING_NAME} - ${photoCounter}`; + siteName = `${Constants.BRANDING_NAME} - ${photoCounter}`; if (engagementText) { siteName = `${Constants.BRANDING_NAME} - ${engagementText} - ${photoCounter}`; } - - headers.push(``); } headers.push( diff --git a/src/strings.ts b/src/strings.ts index bb3b0dd..55cd7b7 100644 --- a/src/strings.ts +++ b/src/strings.ts @@ -72,6 +72,7 @@ This is caused by Twitter API downtime or a new bug. Try again in a little while TRANSLATE_TEXT: `═ ↘️ Translated from {language} ═════`, TRANSLATE_TEXT_INTL: `═ ↘️ {source} ➡️ {destination} ═════`, PHOTO_COUNT: `Photo {number} of {total}`, + VIDEO_COUNT: `Video {number} of {total}`, SINGULAR_DAY_LEFT: 'day left', PLURAL_DAYS_LEFT: 'days left', diff --git a/src/types.d.ts b/src/types.d.ts index d0a3d32..00b39e6 100644 --- a/src/types.d.ts +++ b/src/types.d.ts @@ -101,6 +101,7 @@ interface APIVideo { width: number; height: number; format: string; + duration: number; } interface APITweet { @@ -125,6 +126,7 @@ interface APITweet { external?: APIExternalMedia; photos?: APIPhoto[]; video?: APIVideo; + videos?: APIVideo[]; mosaic?: APIMosaicPhoto; }; diff --git a/test/index.test.ts b/test/index.test.ts index 0cc54c5..8923b8e 100644 --- a/test/index.test.ts +++ b/test/index.test.ts @@ -114,13 +114,14 @@ test('API fetch video Tweet', async () => { expect(tweet.lang).toEqual('en'); expect(tweet.replying_to).toBeNull(); expect(tweet.media?.video).toBeTruthy(); - const video = tweet.media?.video as APIVideo; + const video = tweet.media?.videos?.[0] as APIVideo; expect(video.url).toEqual( 'https://video.twimg.com/amplify_video/854415175776059393/vid/720x720/dNEi0crU-jA4mTtr.mp4' ); expect(video.thumbnail_url).toEqual('https://pbs.twimg.com/media/C9t-btLVoAEqZI1.jpg'); expect(video.width).toEqual(1596); expect(video.height).toEqual(1600); + expect(video.duration).toEqual(65.667); expect(video.format).toEqual('video/mp4'); expect(video.type).toEqual('video'); }); @@ -179,3 +180,64 @@ test('API fetch multi-photo Tweet', async () => { 'https://mosaic.fxtwitter.com/webp/1554870933449482240/FZQCeMmXwAAOJTt/FZQCl-lWIAMtoW9/FZQCsQPX0AIbY6H/FZQCxmLXEAMST4q' ); }); + +test('API fetch multi-video Tweet', async () => { + const result = await cacheWrapper( + new Request('https://api.fxtwitter.com/dangeredwolf/status/1557914172763127808', { + method: 'GET', + headers: botHeaders + }) + ); + expect(result.status).toEqual(200); + const response = (await result.json()) as APIResponse; + expect(response).toBeTruthy(); + expect(response.code).toEqual(200); + expect(response.message).toEqual('OK'); + + const tweet = response.tweet as APITweet; + expect(tweet).toBeTruthy(); + expect(tweet.url).toEqual( + 'https://twitter.com/dangeredwolf/status/1557914172763127808' + ); + expect(tweet.id).toEqual('1557914172763127808'); + expect(tweet.text).toEqual(''); + expect(tweet.author.screen_name?.toLowerCase()).toEqual('dangeredwolf'); + expect(tweet.author.name).toBeTruthy(); + expect(tweet.author.avatar_url).toBeTruthy(); + expect(tweet.author.banner_url).toBeTruthy(); + expect(tweet.author.avatar_color).toBeTruthy(); + expect(tweet.twitter_card).toEqual('player'); + expect(tweet.created_at).toEqual('Fri Aug 12 02:17:38 +0000 2022'); + expect(tweet.created_timestamp).toEqual(1660270658); + expect(tweet.replying_to).toBeNull(); + expect(tweet.media?.videos).toBeTruthy(); + const videos = tweet.media?.videos as APIVideo[]; + expect(videos[0].url).toEqual('https://video.twimg.com/ext_tw_video/1539029945124528130/pu/vid/1662x1080/ZQP4eoQhnGnKcLEb.mp4?tag=14'); + expect(videos[0].thumbnail_url).toEqual("https://pbs.twimg.com/ext_tw_video_thumb/1539029945124528130/pu/img/6Z1MXMliums60j03.jpg"); + expect(videos[0].width).toEqual(3548); + expect(videos[0].height).toEqual(2304); + expect(videos[0].duration).toEqual(37.75); + expect(videos[0].format).toEqual('video/mp4'); + expect(videos[0].type).toEqual('video'); + expect(videos[1].url).toEqual('https://video.twimg.com/ext_tw_video/1543316856697769984/pu/vid/1920x1080/3fo7b4EnWv2WO8Z1.mp4?tag=14'); + expect(videos[1].thumbnail_url).toEqual("https://pbs.twimg.com/ext_tw_video_thumb/1543316856697769984/pu/img/eCl67JRWO8r4r8A4.jpg"); + expect(videos[1].width).toEqual(1920); + expect(videos[1].height).toEqual(1080); + expect(videos[1].duration).toEqual(71.855); + expect(videos[1].format).toEqual('video/mp4'); + expect(videos[1].type).toEqual('video'); + expect(videos[2].url).toEqual('https://video.twimg.com/ext_tw_video/1543797953105625088/pu/vid/1920x1080/GHSLxzBrwiDLhLYD.mp4?tag=14'); + expect(videos[2].thumbnail_url).toEqual("https://pbs.twimg.com/ext_tw_video_thumb/1543797953105625088/pu/img/2eX2QQkd7b2S1YDl.jpg"); + expect(videos[2].width).toEqual(1920); + expect(videos[2].height).toEqual(1080); + expect(videos[2].duration).toEqual(22.018); + expect(videos[2].format).toEqual('video/mp4'); + expect(videos[2].type).toEqual('video'); + expect(videos[3].url).toEqual('https://video.twimg.com/ext_tw_video/1548602342488129536/pu/vid/720x1280/I_D3svYfjBl7_xGS.mp4?tag=14'); + expect(videos[3].thumbnail_url).toEqual("https://pbs.twimg.com/ext_tw_video_thumb/1548602342488129536/pu/img/V_1u5Nv5BwKBynwv.jpg"); + expect(videos[3].width).toEqual(720); + expect(videos[3].height).toEqual(1280); + expect(videos[3].duration).toEqual(25.133); + expect(videos[3].format).toEqual('video/mp4'); + expect(videos[3].type).toEqual('video'); +}); \ No newline at end of file