mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-04 18:10:56 +01:00
Run prettier
This commit is contained in:
parent
855216551a
commit
d912125a78
7 changed files with 127 additions and 88 deletions
|
@ -81,7 +81,7 @@ if (process.env.SENTRY_DSN) {
|
||||||
// https://sentry.io/orgredirect/organizations/:orgslug/settings/auth-tokens/
|
// https://sentry.io/orgredirect/organizations/:orgslug/settings/auth-tokens/
|
||||||
authToken: process.env.SENTRY_AUTH_TOKEN
|
authToken: process.env.SENTRY_AUTH_TOKEN
|
||||||
})
|
})
|
||||||
)
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
await esbuild.build({
|
await esbuild.build({
|
||||||
|
|
|
@ -36,7 +36,11 @@ export const handleStatus = async (
|
||||||
|
|
||||||
let fetchWithThreads = false;
|
let fetchWithThreads = false;
|
||||||
|
|
||||||
if (c.req.header('user-agent')?.includes('Telegram') && !flags?.direct && flags.instantViewUnrollThreads) {
|
if (
|
||||||
|
c.req.header('user-agent')?.includes('Telegram') &&
|
||||||
|
!flags?.direct &&
|
||||||
|
flags.instantViewUnrollThreads
|
||||||
|
) {
|
||||||
fetchWithThreads = true;
|
fetchWithThreads = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -90,17 +90,16 @@ export const fetchTweetDetail = async (
|
||||||
useElongator,
|
useElongator,
|
||||||
(_conversation: unknown) => {
|
(_conversation: unknown) => {
|
||||||
const conversation = _conversation as TweetDetailResult;
|
const conversation = _conversation as TweetDetailResult;
|
||||||
const response = processResponse(conversation?.data?.threaded_conversation_with_injections_v2?.instructions);
|
const response = processResponse(
|
||||||
const tweet = findStatusInBucket(
|
conversation?.data?.threaded_conversation_with_injections_v2?.instructions
|
||||||
status,
|
|
||||||
response
|
|
||||||
);
|
);
|
||||||
|
const tweet = findStatusInBucket(status, response);
|
||||||
if (tweet && isGraphQLTwitterStatus(tweet)) {
|
if (tweet && isGraphQLTwitterStatus(tweet)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
console.log('invalid graphql tweet', tweet);
|
console.log('invalid graphql tweet', tweet);
|
||||||
console.log('finding status', status);
|
console.log('finding status', status);
|
||||||
console.log('from response', JSON.stringify(response))
|
console.log('from response', JSON.stringify(response));
|
||||||
|
|
||||||
return Array.isArray(conversation?.errors);
|
return Array.isArray(conversation?.errors);
|
||||||
},
|
},
|
||||||
|
@ -268,13 +267,15 @@ const findNextStatus = (id: string, bucket: GraphQLProcessBucket): number => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const findPreviousStatus = (id: string, bucket: GraphQLProcessBucket): number => {
|
const findPreviousStatus = (id: string, bucket: GraphQLProcessBucket): number => {
|
||||||
const status = bucket.allStatuses.find(status => (status.rest_id ?? status.legacy?.id_str) === id);
|
const status = bucket.allStatuses.find(
|
||||||
|
status => (status.rest_id ?? status.legacy?.id_str) === id
|
||||||
|
);
|
||||||
if (!status) {
|
if (!status) {
|
||||||
console.log('uhhh, we could not even find that tweet, dunno how that happened');
|
console.log('uhhh, we could not even find that tweet, dunno how that happened');
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
if ((status.rest_id ?? status.legacy?.id_str) === status.legacy?.in_reply_to_status_id_str) {
|
if ((status.rest_id ?? status.legacy?.id_str) === status.legacy?.in_reply_to_status_id_str) {
|
||||||
console.log('Tweet does not have a parent')
|
console.log('Tweet does not have a parent');
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return bucket.allStatuses.findIndex(
|
return bucket.allStatuses.findIndex(
|
||||||
|
@ -317,7 +318,7 @@ export const constructTwitterThread = async (
|
||||||
let response: TweetDetailResult | TweetResultsByRestIdResult | null = null;
|
let response: TweetDetailResult | TweetResultsByRestIdResult | null = null;
|
||||||
let status: APITwitterStatus;
|
let status: APITwitterStatus;
|
||||||
|
|
||||||
console.log('env', c.env)
|
console.log('env', c.env);
|
||||||
/* We can use TweetDetail on elongator accounts to increase per-account rate limit.
|
/* We can use TweetDetail on elongator accounts to increase per-account rate limit.
|
||||||
We also use TweetDetail to process threads (WIP)
|
We also use TweetDetail to process threads (WIP)
|
||||||
|
|
||||||
|
@ -543,11 +544,19 @@ export const constructTwitterThread = async (
|
||||||
author: author,
|
author: author,
|
||||||
code: 200
|
code: 200
|
||||||
};
|
};
|
||||||
|
|
||||||
await Promise.all(threadStatuses.map(async status => {
|
await Promise.all(
|
||||||
const builtStatus = await buildAPITwitterStatus(c, status, undefined, author, false) as APITwitterStatus;
|
threadStatuses.map(async status => {
|
||||||
socialThread.thread?.push(builtStatus);
|
const builtStatus = (await buildAPITwitterStatus(
|
||||||
}));
|
c,
|
||||||
|
status,
|
||||||
|
undefined,
|
||||||
|
author,
|
||||||
|
false
|
||||||
|
)) as APITwitterStatus;
|
||||||
|
socialThread.thread?.push(builtStatus);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
// Sort socialThread.thread by id converted to bigint
|
// Sort socialThread.thread by id converted to bigint
|
||||||
socialThread.thread?.sort((a, b) => {
|
socialThread.thread?.sort((a, b) => {
|
||||||
|
|
|
@ -59,24 +59,24 @@ export const buildAPITwitterStatus = async (
|
||||||
linkFixer(status.legacy.entities?.urls, status.legacy.full_text || '')
|
linkFixer(status.legacy.entities?.urls, status.legacy.full_text || '')
|
||||||
);
|
);
|
||||||
// if (threadAuthor && threadAuthor.id !== apiUser.id) {
|
// if (threadAuthor && threadAuthor.id !== apiUser.id) {
|
||||||
apiStatus.author = {
|
apiStatus.author = {
|
||||||
id: apiUser.id,
|
id: apiUser.id,
|
||||||
name: apiUser.name,
|
name: apiUser.name,
|
||||||
screen_name: apiUser.screen_name,
|
screen_name: apiUser.screen_name,
|
||||||
avatar_url: apiUser.avatar_url?.replace?.('_normal', '_200x200') ?? null,
|
avatar_url: apiUser.avatar_url?.replace?.('_normal', '_200x200') ?? null,
|
||||||
banner_url: apiUser.banner_url,
|
banner_url: apiUser.banner_url,
|
||||||
description: apiUser.description,
|
description: apiUser.description,
|
||||||
location: apiUser.location,
|
location: apiUser.location,
|
||||||
url: apiUser.url,
|
url: apiUser.url,
|
||||||
followers: apiUser.followers,
|
followers: apiUser.followers,
|
||||||
following: apiUser.following,
|
following: apiUser.following,
|
||||||
joined: apiUser.joined,
|
joined: apiUser.joined,
|
||||||
statuses: apiUser.statuses,
|
statuses: apiUser.statuses,
|
||||||
likes: apiUser.likes,
|
likes: apiUser.likes,
|
||||||
protected: apiUser.protected,
|
protected: apiUser.protected,
|
||||||
birthday: apiUser.birthday,
|
birthday: apiUser.birthday,
|
||||||
website: apiUser.website
|
website: apiUser.website
|
||||||
};
|
};
|
||||||
// }
|
// }
|
||||||
apiStatus.replies = status.legacy.reply_count;
|
apiStatus.replies = status.legacy.reply_count;
|
||||||
if (legacyAPI) {
|
if (legacyAPI) {
|
||||||
|
@ -95,7 +95,7 @@ export const buildAPITwitterStatus = async (
|
||||||
} else {
|
} else {
|
||||||
apiStatus.reposts = status.legacy.retweet_count;
|
apiStatus.reposts = status.legacy.retweet_count;
|
||||||
// if ((threadAuthor && threadAuthor.id !== apiUser.id)) {
|
// if ((threadAuthor && threadAuthor.id !== apiUser.id)) {
|
||||||
apiStatus.author.global_screen_name = apiUser.global_screen_name;
|
apiStatus.author.global_screen_name = apiUser.global_screen_name;
|
||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
apiStatus.likes = status.legacy.favorite_count;
|
apiStatus.likes = status.legacy.favorite_count;
|
||||||
|
|
|
@ -56,7 +56,10 @@ export const statusRequest = async (c: Context) => {
|
||||||
} else if (Constants.INSTANT_VIEW_DOMAINS.includes(url.hostname)) {
|
} else if (Constants.INSTANT_VIEW_DOMAINS.includes(url.hostname)) {
|
||||||
console.log('Forced instant view request');
|
console.log('Forced instant view request');
|
||||||
flags.forceInstantView = true;
|
flags.forceInstantView = true;
|
||||||
} else if (experimentCheck(Experiment.IV_FORCE_THREAD_UNROLL, userAgent.includes('Telegram')) || Constants.INSTANT_VIEW_THREADS_DOMAINS.includes(url.hostname)) {
|
} else if (
|
||||||
|
experimentCheck(Experiment.IV_FORCE_THREAD_UNROLL, userAgent.includes('Telegram')) ||
|
||||||
|
Constants.INSTANT_VIEW_THREADS_DOMAINS.includes(url.hostname)
|
||||||
|
) {
|
||||||
console.log('Forced unroll instant view');
|
console.log('Forced unroll instant view');
|
||||||
flags.forceInstantView = true;
|
flags.forceInstantView = true;
|
||||||
flags.instantViewUnrollThreads = true;
|
flags.instantViewUnrollThreads = true;
|
||||||
|
|
|
@ -123,11 +123,22 @@ const truncateSocialCount = (count: number): string => {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const generateInlineAuthorHeader = (status: APIStatus, author: APIUser, authorActionType: AuthorActionType | null): string => {
|
const generateInlineAuthorHeader = (
|
||||||
return `<h4><i><a href="${status.url}">{AuthorAction}</a> from <b>${author.name}</b> (<a href="${author.url}">@${author.screen_name}</a>):</i></h4>`.format({
|
status: APIStatus,
|
||||||
AuthorAction: authorActionType === AuthorActionType.Reply ? 'Reply' : authorActionType === AuthorActionType.Original ? 'Original' : 'Follow-up'
|
author: APIUser,
|
||||||
});
|
authorActionType: AuthorActionType | null
|
||||||
}
|
): string => {
|
||||||
|
return `<h4><i><a href="${status.url}">{AuthorAction}</a> from <b>${author.name}</b> (<a href="${author.url}">@${author.screen_name}</a>):</i></h4>`.format(
|
||||||
|
{
|
||||||
|
AuthorAction:
|
||||||
|
authorActionType === AuthorActionType.Reply
|
||||||
|
? 'Reply'
|
||||||
|
: authorActionType === AuthorActionType.Original
|
||||||
|
? 'Original'
|
||||||
|
: 'Follow-up'
|
||||||
|
}
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
const wrapForeignLinks = (url: string) => {
|
const wrapForeignLinks = (url: string) => {
|
||||||
let unwrap = false;
|
let unwrap = false;
|
||||||
|
@ -160,9 +171,7 @@ const generateStatusFooter = (status: APIStatus, isQuote = false, author: APIUse
|
||||||
{aboutSection}
|
{aboutSection}
|
||||||
`.format({
|
`.format({
|
||||||
socialText: getSocialTextIV(status as APITwitterStatus) || '',
|
socialText: getSocialTextIV(status as APITwitterStatus) || '',
|
||||||
viewOriginal: !isQuote
|
viewOriginal: !isQuote ? `<a href="${status.url}">View full thread</a>` : notApplicableComment,
|
||||||
? `<a href="${status.url}">View full thread</a>`
|
|
||||||
: notApplicableComment,
|
|
||||||
aboutSection: isQuote
|
aboutSection: isQuote
|
||||||
? ''
|
? ''
|
||||||
: `<h2>About author</h2>
|
: `<h2>About author</h2>
|
||||||
|
@ -194,7 +203,7 @@ const generateStatusFooter = (status: APIStatus, isQuote = false, author: APIUse
|
||||||
const generatePoll = (poll: APIPoll, language: string): string => {
|
const generatePoll = (poll: APIPoll, language: string): string => {
|
||||||
// const formatNumber = Intl.NumberFormat(language ?? 'en')
|
// const formatNumber = Intl.NumberFormat(language ?? 'en')
|
||||||
let str = '';
|
let str = '';
|
||||||
|
|
||||||
const barLength = 20;
|
const barLength = 20;
|
||||||
|
|
||||||
poll.choices.forEach(choice => {
|
poll.choices.forEach(choice => {
|
||||||
|
@ -206,11 +215,11 @@ const generatePoll = (poll: APIPoll, language: string): string => {
|
||||||
str += `<br>${formatNumber(poll.total_votes)} votes · ${poll.time_left_en}`;
|
str += `<br>${formatNumber(poll.total_votes)} votes · ${poll.time_left_en}`;
|
||||||
|
|
||||||
return str;
|
return str;
|
||||||
}
|
};
|
||||||
|
|
||||||
const generateCommunityNote = (status: APITwitterStatus): string => {
|
const generateCommunityNote = (status: APITwitterStatus): string => {
|
||||||
if (status.community_note) {
|
if (status.community_note) {
|
||||||
console.log('community_note', status.community_note)
|
console.log('community_note', status.community_note);
|
||||||
const note = status.community_note;
|
const note = status.community_note;
|
||||||
const entities = note.entities;
|
const entities = note.entities;
|
||||||
entities.sort((a, b) => a.fromIndex - b.fromIndex); // sort entities by fromIndex
|
entities.sort((a, b) => a.fromIndex - b.fromIndex); // sort entities by fromIndex
|
||||||
|
@ -220,7 +229,7 @@ const generateCommunityNote = (status: APITwitterStatus): string => {
|
||||||
|
|
||||||
entities.forEach(entity => {
|
entities.forEach(entity => {
|
||||||
if (entity?.ref?.type !== 'TimelineUrl') {
|
if (entity?.ref?.type !== 'TimelineUrl') {
|
||||||
return
|
return;
|
||||||
}
|
}
|
||||||
const fromIndex = entity.fromIndex;
|
const fromIndex = entity.fromIndex;
|
||||||
const toIndex = entity.toIndex;
|
const toIndex = entity.toIndex;
|
||||||
|
@ -248,9 +257,14 @@ const generateCommunityNote = (status: APITwitterStatus): string => {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
return '';
|
return '';
|
||||||
}
|
};
|
||||||
|
|
||||||
const generateStatus = (status: APIStatus, author: APIUser, isQuote = false, authorActionType: AuthorActionType | null): string => {
|
const generateStatus = (
|
||||||
|
status: APIStatus,
|
||||||
|
author: APIUser,
|
||||||
|
isQuote = false,
|
||||||
|
authorActionType: AuthorActionType | null
|
||||||
|
): string => {
|
||||||
let text = paragraphify(sanitizeText(status.text), isQuote);
|
let text = paragraphify(sanitizeText(status.text), isQuote);
|
||||||
text = htmlifyLinks(text);
|
text = htmlifyLinks(text);
|
||||||
text = htmlifyHashtags(text);
|
text = htmlifyHashtags(text);
|
||||||
|
@ -323,34 +337,43 @@ export const renderInstantView = (properties: RenderProperties): ResponseInstruc
|
||||||
<sub><a href="${status.url}">View full thread</a></sub>
|
<sub><a href="${status.url}">View full thread</a></sub>
|
||||||
<h1>${status.author.name} (@${status.author.screen_name})</h1>
|
<h1>${status.author.name} (@${status.author.screen_name})</h1>
|
||||||
|
|
||||||
${thread?.thread?.map((status) => {
|
${thread?.thread
|
||||||
console.log('previousThreadPieceAuthor', previousThreadPieceAuthor)
|
?.map(status => {
|
||||||
if (originalAuthor === null) {
|
console.log('previousThreadPieceAuthor', previousThreadPieceAuthor);
|
||||||
originalAuthor = status.author?.id;
|
if (originalAuthor === null) {
|
||||||
}
|
originalAuthor = status.author?.id;
|
||||||
const differentAuthor = thread?.author?.id !== status.author?.id || (previousThreadPieceAuthor !== null && previousThreadPieceAuthor !== status.author?.id);
|
|
||||||
const isOriginal = thread?.author?.id !== status.author?.id && previousThreadPieceAuthor === null;
|
|
||||||
const isFollowup = thread?.author?.id === status.author?.id && previousThreadPieceAuthor !== null && previousThreadPieceAuthor !== thread?.author?.id && originalAuthor === status.author?.id;
|
|
||||||
console.log('differentAuthor', differentAuthor)
|
|
||||||
console.log('isOriginal', isOriginal)
|
|
||||||
console.log('isFollowup', isFollowup)
|
|
||||||
|
|
||||||
let authorAction = null;
|
|
||||||
|
|
||||||
if (differentAuthor) {
|
|
||||||
if (isFollowup) {
|
|
||||||
authorAction = AuthorActionType.FollowUp;
|
|
||||||
} else if (isOriginal) {
|
|
||||||
authorAction = AuthorActionType.Original;
|
|
||||||
} else if (previousThreadPieceAuthor !== status.author?.id) {
|
|
||||||
authorAction = AuthorActionType.Reply;
|
|
||||||
}
|
}
|
||||||
}
|
const differentAuthor =
|
||||||
|
thread?.author?.id !== status.author?.id ||
|
||||||
|
(previousThreadPieceAuthor !== null && previousThreadPieceAuthor !== status.author?.id);
|
||||||
|
const isOriginal =
|
||||||
|
thread?.author?.id !== status.author?.id && previousThreadPieceAuthor === null;
|
||||||
|
const isFollowup =
|
||||||
|
thread?.author?.id === status.author?.id &&
|
||||||
|
previousThreadPieceAuthor !== null &&
|
||||||
|
previousThreadPieceAuthor !== thread?.author?.id &&
|
||||||
|
originalAuthor === status.author?.id;
|
||||||
|
console.log('differentAuthor', differentAuthor);
|
||||||
|
console.log('isOriginal', isOriginal);
|
||||||
|
console.log('isFollowup', isFollowup);
|
||||||
|
|
||||||
previousThreadPieceAuthor = status.author?.id;
|
let authorAction = null;
|
||||||
|
|
||||||
return generateStatus(status, status.author ?? thread?.author, false, authorAction)
|
if (differentAuthor) {
|
||||||
}).join('')}
|
if (isFollowup) {
|
||||||
|
authorAction = AuthorActionType.FollowUp;
|
||||||
|
} else if (isOriginal) {
|
||||||
|
authorAction = AuthorActionType.Original;
|
||||||
|
} else if (previousThreadPieceAuthor !== status.author?.id) {
|
||||||
|
authorAction = AuthorActionType.Reply;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
previousThreadPieceAuthor = status.author?.id;
|
||||||
|
|
||||||
|
return generateStatus(status, status.author ?? thread?.author, false, authorAction);
|
||||||
|
})
|
||||||
|
.join('')}
|
||||||
${generateStatusFooter(status, false, thread?.author ?? status.author)}
|
${generateStatusFooter(status, false, thread?.author ?? status.author)}
|
||||||
<br>${`<a href="${status.url}">View full thread</a>`}
|
<br>${`<a href="${status.url}">View full thread</a>`}
|
||||||
</article>`;
|
</article>`;
|
||||||
|
|
28
src/types/vendor/twitter.d.ts
vendored
28
src/types/vendor/twitter.d.ts
vendored
|
@ -352,14 +352,14 @@ type GraphQLTwitterStatusLegacy = {
|
||||||
};
|
};
|
||||||
|
|
||||||
type BirdwatchEntity = {
|
type BirdwatchEntity = {
|
||||||
fromIndex: number, // 119
|
fromIndex: number; // 119
|
||||||
toIndex: number, // 154
|
toIndex: number; // 154
|
||||||
ref: {
|
ref: {
|
||||||
type: "TimelineUrl",
|
type: 'TimelineUrl';
|
||||||
url: string, // https://t.co/jxvVatCVCz
|
url: string; // https://t.co/jxvVatCVCz
|
||||||
urlType: "ExternalUrl"
|
urlType: 'ExternalUrl';
|
||||||
}
|
};
|
||||||
}
|
};
|
||||||
|
|
||||||
type GraphQLTwitterStatus = {
|
type GraphQLTwitterStatus = {
|
||||||
// Workaround
|
// Workaround
|
||||||
|
@ -369,15 +369,15 @@ type GraphQLTwitterStatus = {
|
||||||
rest_id: string; // "1674824189176590336",
|
rest_id: string; // "1674824189176590336",
|
||||||
has_birdwatch_notes: boolean;
|
has_birdwatch_notes: boolean;
|
||||||
birdwatch_pivot: {
|
birdwatch_pivot: {
|
||||||
destinationUrl: string, // https://twitter.com/i/birdwatch/n/1784594925926973714
|
destinationUrl: string; // https://twitter.com/i/birdwatch/n/1784594925926973714
|
||||||
note: {
|
note: {
|
||||||
rest_id: string // 1784594925926973714
|
rest_id: string; // 1784594925926973714
|
||||||
},
|
};
|
||||||
subtitle: {
|
subtitle: {
|
||||||
text: string, // "This screenshot is from Sonic 1\n\ninfo.sonicretro.org/Sonic_the_Hedg…"
|
text: string; // "This screenshot is from Sonic 1\n\ninfo.sonicretro.org/Sonic_the_Hedg…"
|
||||||
entities: BirdwatchEntity[]
|
entities: BirdwatchEntity[];
|
||||||
},
|
};
|
||||||
}
|
};
|
||||||
core: {
|
core: {
|
||||||
user_results: {
|
user_results: {
|
||||||
result: GraphQLUser;
|
result: GraphQLUser;
|
||||||
|
|
Loading…
Add table
Reference in a new issue