2021-01-25 23:40:30 -05:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2020-07-18 23:35:19 -04:00
|
|
|
|
using System.Linq;
|
2022-05-13 11:29:28 -04:00
|
|
|
|
using System.IO;
|
2022-05-05 20:15:07 -04:00
|
|
|
|
using System.Net.Http;
|
|
|
|
|
using System.Text.Json;
|
2022-11-25 15:05:05 -05:00
|
|
|
|
using System.Text.Json.Nodes;
|
2022-05-05 20:15:07 -04:00
|
|
|
|
using System.Threading.Tasks;
|
2020-03-22 01:29:51 -04:00
|
|
|
|
using BirdsiteLive.Common.Settings;
|
2021-01-12 23:53:23 -05:00
|
|
|
|
using BirdsiteLive.Statistics.Domain;
|
2020-03-22 01:29:51 -04:00
|
|
|
|
using BirdsiteLive.Twitter.Models;
|
2021-01-30 00:22:29 -05:00
|
|
|
|
using BirdsiteLive.Twitter.Tools;
|
2021-01-25 23:40:30 -05:00
|
|
|
|
using Microsoft.Extensions.Logging;
|
2022-05-10 17:32:07 -04:00
|
|
|
|
using System.Text.RegularExpressions;
|
2020-03-21 18:58:23 -04:00
|
|
|
|
|
|
|
|
|
namespace BirdsiteLive.Twitter
|
|
|
|
|
{
|
2021-01-18 02:07:09 -05:00
|
|
|
|
public interface ITwitterTweetsService
|
2020-03-21 18:58:23 -04:00
|
|
|
|
{
|
2020-07-22 20:19:40 -04:00
|
|
|
|
ExtractedTweet GetTweet(long statusId);
|
|
|
|
|
ExtractedTweet[] GetTimeline(string username, int nberTweets, long fromTweetId = -1);
|
2020-03-21 18:58:23 -04:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-18 02:07:09 -05:00
|
|
|
|
public class TwitterTweetsService : ITwitterTweetsService
|
2020-03-21 18:58:23 -04:00
|
|
|
|
{
|
2021-01-30 00:22:29 -05:00
|
|
|
|
private readonly ITwitterAuthenticationInitializer _twitterAuthenticationInitializer;
|
2021-01-12 23:53:23 -05:00
|
|
|
|
private readonly ITwitterStatisticsHandler _statisticsHandler;
|
2021-01-18 02:07:09 -05:00
|
|
|
|
private readonly ITwitterUserService _twitterUserService;
|
2021-01-25 23:40:30 -05:00
|
|
|
|
private readonly ILogger<TwitterTweetsService> _logger;
|
2022-05-05 20:15:07 -04:00
|
|
|
|
private HttpClient _httpClient = new HttpClient();
|
2020-03-21 18:58:23 -04:00
|
|
|
|
|
|
|
|
|
#region Ctor
|
2022-05-08 19:19:09 -04:00
|
|
|
|
public TwitterTweetsService(ITwitterAuthenticationInitializer twitterAuthenticationInitializer, ITwitterStatisticsHandler statisticsHandler, ITwitterUserService twitterUserService, ILogger<TwitterTweetsService> logger)
|
2020-03-21 18:58:23 -04:00
|
|
|
|
{
|
2021-01-30 00:22:29 -05:00
|
|
|
|
_twitterAuthenticationInitializer = twitterAuthenticationInitializer;
|
2021-01-12 23:53:23 -05:00
|
|
|
|
_statisticsHandler = statisticsHandler;
|
2021-01-18 02:07:09 -05:00
|
|
|
|
_twitterUserService = twitterUserService;
|
2021-01-25 23:40:30 -05:00
|
|
|
|
_logger = logger;
|
2020-03-21 18:58:23 -04:00
|
|
|
|
}
|
|
|
|
|
#endregion
|
2021-01-25 23:40:30 -05:00
|
|
|
|
|
2022-05-05 20:15:07 -04:00
|
|
|
|
|
2020-07-22 20:19:40 -04:00
|
|
|
|
public ExtractedTweet GetTweet(long statusId)
|
2022-05-05 20:15:07 -04:00
|
|
|
|
{
|
|
|
|
|
return GetTweetAsync(statusId).Result;
|
|
|
|
|
}
|
|
|
|
|
public async Task<ExtractedTweet> GetTweetAsync(long statusId)
|
2020-07-01 22:45:43 -04:00
|
|
|
|
{
|
2022-11-26 16:42:15 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var client = await _twitterAuthenticationInitializer.MakeHttpClient();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string reqURL = "https://twitter.com/i/api/graphql/BoHLKeBvibdYDiJON1oqTg/TweetDetail?variables=%7B%22focalTweetId%22%3A%22"
|
|
|
|
|
+ statusId + "%22%2C%22referrer%22%3A%22profile%22%2C%22rux_context%22%3A%22HHwWgICypZb4saYsAAAA%22%2C%22with_rux_injections%22%3Atrue%2C%22includePromotedContent%22%3Atrue%2C%22withCommunity%22%3Atrue%2C%22withQuickPromoteEligibilityTweetFields%22%3Atrue%2C%22withBirdwatchNotes%22%3Afalse%2C%22withSuperFollowsUserFields%22%3Atrue%2C%22withDownvotePerspective%22%3Afalse%2C%22withReactionsMetadata%22%3Afalse%2C%22withReactionsPerspective%22%3Afalse%2C%22withSuperFollowsTweetFields%22%3Atrue%2C%22withVoice%22%3Atrue%2C%22withV2Timeline%22%3Atrue%7D&features=%7B%22responsive_web_twitter_blue_verified_badge_is_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%2C%22unified_cards_ad_metadata_container_dynamic_card_content_query_enabled%22%3Atrue%2C%22tweetypie_unmention_optimization_enabled%22%3Atrue%2C%22responsive_web_uc_gql_enabled%22%3Atrue%2C%22vibe_api_enabled%22%3Atrue%2C%22responsive_web_edit_tweet_api_enabled%22%3Atrue%2C%22graphql_is_translatable_rweb_tweet_is_translatable_enabled%22%3Afalse%2C%22standardized_nudges_misinfo%22%3Atrue%2C%22tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled%22%3Afalse%2C%22interactive_text_enabled%22%3Atrue%2C%22responsive_web_text_conversations_enabled%22%3Afalse%2C%22responsive_web_enhance_cards_enabled%22%3Atrue%7D";
|
|
|
|
|
|
2021-01-25 23:40:30 -05:00
|
|
|
|
try
|
|
|
|
|
{
|
2022-05-05 20:15:07 -04:00
|
|
|
|
JsonDocument tweet;
|
2022-05-13 18:51:23 -04:00
|
|
|
|
using (var request = new HttpRequestMessage(new HttpMethod("GET"), reqURL))
|
2022-11-25 13:48:49 -05:00
|
|
|
|
{
|
|
|
|
|
var httpResponse = await client.SendAsync(request);
|
2022-05-05 20:15:07 -04:00
|
|
|
|
httpResponse.EnsureSuccessStatusCode();
|
|
|
|
|
var c = await httpResponse.Content.ReadAsStringAsync();
|
|
|
|
|
tweet = JsonDocument.Parse(c);
|
|
|
|
|
}
|
2021-01-30 00:22:29 -05:00
|
|
|
|
|
2022-05-13 11:29:28 -04:00
|
|
|
|
|
2022-11-26 16:42:15 -05:00
|
|
|
|
var timeline = tweet.RootElement.GetProperty("data").GetProperty("threaded_conversation_with_injections_v2")
|
2022-11-26 17:15:30 -05:00
|
|
|
|
.GetProperty("instructions").GetProperty("entries").EnumerateArray();
|
2022-05-13 11:29:28 -04:00
|
|
|
|
|
2022-11-26 17:15:30 -05:00
|
|
|
|
return Extract( timeline.First() );
|
2021-01-25 23:40:30 -05:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(e, "Error retrieving tweet {TweetId}", statusId);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2020-07-22 20:19:40 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public ExtractedTweet[] GetTimeline(string username, int nberTweets, long fromTweetId = -1)
|
2020-07-18 23:35:19 -04:00
|
|
|
|
{
|
2022-05-05 20:15:07 -04:00
|
|
|
|
return GetTimelineAsync(username, nberTweets, fromTweetId).Result;
|
|
|
|
|
}
|
|
|
|
|
public async Task<ExtractedTweet[]> GetTimelineAsync(string username, int nberTweets, long fromTweetId = -1)
|
|
|
|
|
{
|
2022-05-17 17:51:21 -04:00
|
|
|
|
if (nberTweets < 5)
|
|
|
|
|
nberTweets = 5;
|
|
|
|
|
|
|
|
|
|
if (nberTweets > 100)
|
|
|
|
|
nberTweets = 100;
|
2021-01-30 00:22:29 -05:00
|
|
|
|
|
2022-11-25 13:48:49 -05:00
|
|
|
|
var client = await _twitterAuthenticationInitializer.MakeHttpClient();
|
2021-01-29 23:10:02 -05:00
|
|
|
|
|
2021-02-01 20:07:53 -05:00
|
|
|
|
var user = _twitterUserService.GetUser(username);
|
|
|
|
|
if (user == null || user.Protected) return new ExtractedTweet[0];
|
|
|
|
|
|
2022-11-25 13:48:49 -05:00
|
|
|
|
|
2022-11-26 14:08:13 -05:00
|
|
|
|
var reqURL = "https://twitter.com/i/api/graphql/s0hG9oAmWEYVBqOLJP-TBQ/UserTweetsAndReplies?variables=%7B%22userId%22%3A%22"
|
2022-05-07 18:35:35 -04:00
|
|
|
|
+ user.Id +
|
2022-11-25 13:48:49 -05:00
|
|
|
|
"%22%2C%22count%22%3A40%2C%22includePromotedContent%22%3Atrue%2C%22withQuickPromoteEligibilityTweetFields%22%3Atrue%2C%22withSuperFollowsUserFields%22%3Atrue%2C%22withDownvotePerspective%22%3Afalse%2C%22withReactionsMetadata%22%3Afalse%2C%22withReactionsPerspective%22%3Afalse%2C%22withSuperFollowsTweetFields%22%3Atrue%2C%22withVoice%22%3Atrue%2C%22withV2Timeline%22%3Atrue%7D&features=%7B%22responsive_web_twitter_blue_verified_badge_is_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%2C%22unified_cards_ad_metadata_container_dynamic_card_content_query_enabled%22%3Atrue%2C%22tweetypie_unmention_optimization_enabled%22%3Atrue%2C%22responsive_web_uc_gql_enabled%22%3Atrue%2C%22vibe_api_enabled%22%3Atrue%2C%22responsive_web_edit_tweet_api_enabled%22%3Atrue%2C%22graphql_is_translatable_rweb_tweet_is_translatable_enabled%22%3Afalse%2C%22standardized_nudges_misinfo%22%3Atrue%2C%22tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled%22%3Afalse%2C%22interactive_text_enabled%22%3Atrue%2C%22responsive_web_text_conversations_enabled%22%3Afalse%2C%22responsive_web_enhance_cards_enabled%22%3Atrue%7D";
|
|
|
|
|
JsonDocument results;
|
|
|
|
|
List<ExtractedTweet> extractedTweets = new List<ExtractedTweet>();
|
2022-05-05 20:15:07 -04:00
|
|
|
|
try
|
2021-02-01 20:07:53 -05:00
|
|
|
|
{
|
2022-05-07 18:35:35 -04:00
|
|
|
|
using (var request = new HttpRequestMessage(new HttpMethod("GET"), reqURL))
|
2022-11-25 13:48:49 -05:00
|
|
|
|
{
|
2022-11-25 14:42:26 -05:00
|
|
|
|
|
2022-11-25 13:48:49 -05:00
|
|
|
|
var httpResponse = await client.SendAsync(request);
|
2022-05-05 20:15:07 -04:00
|
|
|
|
httpResponse.EnsureSuccessStatusCode();
|
|
|
|
|
var c = await httpResponse.Content.ReadAsStringAsync();
|
2022-11-25 13:48:49 -05:00
|
|
|
|
results = JsonDocument.Parse(c);
|
2022-05-05 20:15:07 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_statisticsHandler.CalledTweetApi();
|
2022-11-25 13:48:49 -05:00
|
|
|
|
if (results == null) return null; //TODO: test this
|
2020-07-18 23:35:19 -04:00
|
|
|
|
}
|
2022-05-05 20:15:07 -04:00
|
|
|
|
catch (Exception e)
|
2020-07-18 23:35:19 -04:00
|
|
|
|
{
|
2022-05-05 20:15:07 -04:00
|
|
|
|
_logger.LogError(e, "Error retrieving timeline ", username);
|
|
|
|
|
return null;
|
2020-07-18 23:35:19 -04:00
|
|
|
|
}
|
|
|
|
|
|
2022-11-25 13:48:49 -05:00
|
|
|
|
var timeline = results.RootElement.GetProperty("data").GetProperty("user").GetProperty("result")
|
|
|
|
|
.GetProperty("timeline_v2").GetProperty("timeline").GetProperty("instructions").EnumerateArray();
|
|
|
|
|
|
|
|
|
|
foreach (JsonElement timelineElement in timeline)
|
2022-05-17 18:17:05 -04:00
|
|
|
|
{
|
2022-11-25 13:48:49 -05:00
|
|
|
|
if (timelineElement.GetProperty("type").GetString() != "TimelineAddEntries")
|
|
|
|
|
continue;
|
2022-05-17 18:17:05 -04:00
|
|
|
|
|
2022-11-25 13:48:49 -05:00
|
|
|
|
|
|
|
|
|
foreach (JsonElement tweet in timelineElement.GetProperty("entries").EnumerateArray())
|
|
|
|
|
{
|
2022-11-26 14:34:09 -05:00
|
|
|
|
if (tweet.GetProperty("content").GetProperty("entryType").GetString() != "TimelineTimelineItem")
|
|
|
|
|
continue;
|
|
|
|
|
|
2022-11-25 14:48:10 -05:00
|
|
|
|
try
|
2022-11-26 16:42:15 -05:00
|
|
|
|
{
|
|
|
|
|
var extractedTweet = Extract(tweet);
|
2022-11-25 15:05:05 -05:00
|
|
|
|
extractedTweets.Add(extractedTweet);
|
2022-11-26 16:42:15 -05:00
|
|
|
|
|
2022-11-25 14:48:10 -05:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2022-11-25 15:05:05 -05:00
|
|
|
|
_logger.LogError("Tried getting timeline from user " + username + ", but got error: \n" + e.Message + e.StackTrace + e.Source
|
|
|
|
|
+ JsonObject.Create(tweet).ToJsonString(new JsonSerializerOptions { WriteIndented = true }));
|
2022-11-25 14:48:10 -05:00
|
|
|
|
|
|
|
|
|
}
|
2022-11-25 13:48:49 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-05-13 11:29:28 -04:00
|
|
|
|
|
2022-11-25 13:48:49 -05:00
|
|
|
|
return extractedTweets.ToArray();
|
2022-05-08 19:19:09 -04:00
|
|
|
|
}
|
|
|
|
|
|
2022-11-26 16:42:15 -05:00
|
|
|
|
private ExtractedTweet Extract(JsonElement tweet)
|
2022-05-08 19:19:09 -04:00
|
|
|
|
{
|
|
|
|
|
|
2022-11-26 16:42:15 -05:00
|
|
|
|
JsonElement retweet;
|
|
|
|
|
TwitterUser OriginalAuthor;
|
2022-11-26 16:58:35 -05:00
|
|
|
|
JsonElement inReplyToPostIdElement;
|
|
|
|
|
JsonElement inReplyToUserElement;
|
|
|
|
|
string inReplyToUser = null;
|
|
|
|
|
long? inReplyToPostId = null;
|
|
|
|
|
|
|
|
|
|
bool isReply = tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.TryGetProperty("in_reply_to_status_id_str", out inReplyToPostIdElement);
|
|
|
|
|
tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.TryGetProperty("in_reply_to_screen_name", out inReplyToUserElement);
|
|
|
|
|
if (isReply)
|
|
|
|
|
{
|
|
|
|
|
inReplyToPostId = Int64.Parse(inReplyToPostIdElement.GetString());
|
|
|
|
|
inReplyToUser = inReplyToUserElement.GetString();
|
|
|
|
|
}
|
2022-11-26 16:42:15 -05:00
|
|
|
|
bool isRetweet = tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.TryGetProperty("retweeted_status_result", out retweet);
|
|
|
|
|
string MessageContent;
|
|
|
|
|
if (!isRetweet)
|
2022-05-13 11:29:28 -04:00
|
|
|
|
{
|
2022-11-26 16:42:15 -05:00
|
|
|
|
MessageContent = tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.GetProperty("full_text").GetString();
|
|
|
|
|
OriginalAuthor = null;
|
2022-05-14 11:19:35 -04:00
|
|
|
|
}
|
2022-11-26 16:42:15 -05:00
|
|
|
|
else
|
2022-05-14 11:19:35 -04:00
|
|
|
|
{
|
2022-11-26 16:42:15 -05:00
|
|
|
|
MessageContent = tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.GetProperty("retweeted_status_result").GetProperty("result")
|
|
|
|
|
.GetProperty("legacy").GetProperty("full_text").GetString();
|
|
|
|
|
string OriginalAuthorUsername = tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.GetProperty("retweeted_status_result").GetProperty("result")
|
|
|
|
|
.GetProperty("core").GetProperty("user_results").GetProperty("result")
|
|
|
|
|
.GetProperty("legacy").GetProperty("screen_name").GetString();
|
|
|
|
|
OriginalAuthor = _twitterUserService.GetUser(OriginalAuthorUsername);
|
2022-05-13 11:29:28 -04:00
|
|
|
|
}
|
|
|
|
|
|
2022-11-26 16:42:15 -05:00
|
|
|
|
string creationTime = tweet.GetProperty("content").GetProperty("itemContent")
|
|
|
|
|
.GetProperty("tweet_results").GetProperty("result").GetProperty("legacy")
|
|
|
|
|
.GetProperty("created_at").GetString().Replace(" +0000", "");
|
2022-05-08 19:19:09 -04:00
|
|
|
|
var extractedTweet = new ExtractedTweet
|
|
|
|
|
{
|
2022-11-26 16:42:15 -05:00
|
|
|
|
Id = Int64.Parse(tweet.GetProperty("sortIndex").GetString()),
|
2022-11-26 16:58:35 -05:00
|
|
|
|
InReplyToStatusId = inReplyToPostId,
|
|
|
|
|
InReplyToAccount = inReplyToUser,
|
2022-11-26 16:42:15 -05:00
|
|
|
|
MessageContent = MessageContent,
|
|
|
|
|
CreatedAt = DateTime.ParseExact(creationTime, "ddd MMM dd HH:mm:ss yyyy", System.Globalization.CultureInfo.InvariantCulture),
|
2022-11-26 16:58:35 -05:00
|
|
|
|
IsReply = isReply,
|
2022-05-08 19:19:09 -04:00
|
|
|
|
IsThread = false,
|
2022-11-26 16:42:15 -05:00
|
|
|
|
IsRetweet = isRetweet,
|
|
|
|
|
Media = null,
|
2022-05-09 20:31:18 -04:00
|
|
|
|
RetweetUrl = "https://t.co/123",
|
2022-11-26 16:42:15 -05:00
|
|
|
|
OriginalAuthor = OriginalAuthor,
|
2022-05-08 19:19:09 -04:00
|
|
|
|
};
|
2022-11-26 16:42:15 -05:00
|
|
|
|
|
2022-05-08 19:19:09 -04:00
|
|
|
|
return extractedTweet;
|
2022-11-26 16:42:15 -05:00
|
|
|
|
|
2020-07-18 23:35:19 -04:00
|
|
|
|
}
|
2022-05-13 11:29:28 -04:00
|
|
|
|
private string GetMediaType(string mediaType, string mediaUrl)
|
|
|
|
|
{
|
|
|
|
|
switch (mediaType)
|
|
|
|
|
{
|
|
|
|
|
case "photo":
|
|
|
|
|
var pExt = Path.GetExtension(mediaUrl);
|
|
|
|
|
switch (pExt)
|
|
|
|
|
{
|
|
|
|
|
case ".jpg":
|
|
|
|
|
case ".jpeg":
|
|
|
|
|
return "image/jpeg";
|
|
|
|
|
case ".png":
|
|
|
|
|
return "image/png";
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
case "animated_gif":
|
|
|
|
|
var vExt = Path.GetExtension(mediaUrl);
|
|
|
|
|
switch (vExt)
|
|
|
|
|
{
|
|
|
|
|
case ".gif":
|
|
|
|
|
return "image/gif";
|
|
|
|
|
case ".mp4":
|
|
|
|
|
return "video/mp4";
|
|
|
|
|
}
|
|
|
|
|
return "image/gif";
|
|
|
|
|
case "video":
|
|
|
|
|
return "video/mp4";
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2020-03-21 18:58:23 -04:00
|
|
|
|
}
|
2022-05-07 18:54:06 +00:00
|
|
|
|
}
|