fully functional?!
This commit is contained in:
+3
-3
@@ -142,8 +142,8 @@ cython_debug/
|
|||||||
# VS Code files
|
# VS Code files
|
||||||
.vscode
|
.vscode
|
||||||
|
|
||||||
# project-specific
|
# project-specific (secret.ini: can't ignore existing file?)
|
||||||
/secrets.ini
|
secrets.ini
|
||||||
/queue.txt
|
queue.txt
|
||||||
/img.png
|
/img.png
|
||||||
/src/img.png
|
/src/img.png
|
||||||
+30
-3
@@ -1,7 +1,34 @@
|
|||||||
## The bot's listen mode
|
## The bot's listen mode
|
||||||
# Continuously listen for cross-company interactions.
|
# Continuously listen for cross-company interactions.
|
||||||
|
|
||||||
import twapi
|
import tweepy
|
||||||
|
from talenttweet import TalentTweet
|
||||||
|
|
||||||
async def run():
|
from twapi import TwAPI
|
||||||
pass
|
import api_secrets
|
||||||
|
import talent_lists as tl
|
||||||
|
|
||||||
|
async def on_response(resp):
|
||||||
|
print(resp)
|
||||||
|
print(resp.data)
|
||||||
|
ttweet = TalentTweet.create_from_v2api_response(resp)
|
||||||
|
await TwAPI.instance.post_ttweet(ttweet)
|
||||||
|
|
||||||
|
def run():
|
||||||
|
sc = tweepy.StreamingClient(api_secrets.bearer_token())
|
||||||
|
|
||||||
|
# clear rules
|
||||||
|
rules_resp = sc.get_rules()
|
||||||
|
if rules_resp.data:
|
||||||
|
sc.delete_rules(rules_resp.data)
|
||||||
|
|
||||||
|
# create new rules
|
||||||
|
for rule in tl.get_twitter_rules():
|
||||||
|
sc.add_rules(tweepy.StreamRule(rule))
|
||||||
|
|
||||||
|
sc.on_response=on_response
|
||||||
|
sc.filter(
|
||||||
|
expansions=TwAPI.TWEET_EXPANSIONS,
|
||||||
|
media_fields=TwAPI.TWEET_MEDIA_FIELDS,
|
||||||
|
tweet_fields=TwAPI.TWEET_FIELDS
|
||||||
|
)
|
||||||
@@ -51,6 +51,7 @@ async def async_main():
|
|||||||
case 'c' | 'catchup':
|
case 'c' | 'catchup':
|
||||||
print('RUNNING IN CATCH-UP MODE\n')
|
print('RUNNING IN CATCH-UP MODE\n')
|
||||||
await catchup.run(PROGRAM_ARGS)
|
await catchup.run(PROGRAM_ARGS)
|
||||||
|
await listen.run()
|
||||||
case 'd' | 'delete-all':
|
case 'd' | 'delete-all':
|
||||||
print('WARNING: SELF-DESTRUCT MODE')
|
print('WARNING: SELF-DESTRUCT MODE')
|
||||||
await self_destruct()
|
await self_destruct()
|
||||||
|
|||||||
+15
-1
@@ -17,7 +17,7 @@ def __create_dict(file, _dict):
|
|||||||
if len(words) == 2 and line[0] != '#':
|
if len(words) == 2 and line[0] != '#':
|
||||||
name, id = line.split()
|
name, id = line.split()
|
||||||
talents[int(id)] = name
|
talents[int(id)] = name
|
||||||
# name = util.get_username_online(id) # attempt to get updated name
|
name = util.get_username_online(id, default=name) # attempt to get updated name
|
||||||
talents[int(id)] = name
|
talents[int(id)] = name
|
||||||
_dict[int(id)] = name
|
_dict[int(id)] = name
|
||||||
def init():
|
def init():
|
||||||
@@ -38,3 +38,17 @@ def init():
|
|||||||
|
|
||||||
test_talents = holo_en
|
test_talents = holo_en
|
||||||
|
|
||||||
|
def get_twitter_rules():
|
||||||
|
global talents
|
||||||
|
rules = list()
|
||||||
|
|
||||||
|
names = list(talents.values())
|
||||||
|
curr_rule = f'from:{names[0]}'
|
||||||
|
for name in list(talents.values())[1:]:
|
||||||
|
test_rule = curr_rule + f' OR from:{name}'
|
||||||
|
if len(test_rule) > 512:
|
||||||
|
rules.append(curr_rule)
|
||||||
|
curr_rule = f'from:{name}'
|
||||||
|
else:
|
||||||
|
curr_rule = test_rule
|
||||||
|
return rules
|
||||||
+22
-5
@@ -72,26 +72,43 @@ class TalentTweet:
|
|||||||
date_time = datetime.datetime.strptime(tweet.datetime, '%Y-%m-%d %H:%M:%S %Z')
|
date_time = datetime.datetime.strptime(tweet.datetime, '%Y-%m-%d %H:%M:%S %Z')
|
||||||
return TalentTweet(tweet_id=tweet.id, author_id=tweet.user_id, date_time=date_time, mrq=(mentions, reply_to, quoted_id))
|
return TalentTweet(tweet_id=tweet.id, author_id=tweet.user_id, date_time=date_time, mrq=(mentions, reply_to, quoted_id))
|
||||||
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
async def create_from_id(id):
|
def create_from_v2api_response(resp):
|
||||||
resp = await TwAPI.instance.get_tweet_response(id)
|
|
||||||
tweet = resp.data
|
tweet = resp.data
|
||||||
mrq = TwAPI.get_mrq(tweet, resp)
|
mrq = TwAPI.get_mrq(tweet, resp)
|
||||||
|
rt_target = None
|
||||||
|
rt_author_id = None
|
||||||
|
|
||||||
|
# check if is RT
|
||||||
|
if tweet.referenced_tweets is not None and len(tweet.referenced_tweets) > 0:
|
||||||
|
for ref in tweet.referenced_tweets:
|
||||||
|
if ref.type == 'retweeted':
|
||||||
|
rt_target = ref.id
|
||||||
|
for incl_tweet in resp.includes['tweets']:
|
||||||
|
if incl_tweet.id == ref.id:
|
||||||
|
rt_author_id = incl_tweet.author_id
|
||||||
|
|
||||||
return TalentTweet(
|
return TalentTweet(
|
||||||
tweet_id=tweet.id,
|
tweet_id=tweet.id,
|
||||||
author_id=tweet.author_id,
|
author_id=tweet.author_id,
|
||||||
date_time=tweet.created_at,
|
date_time=tweet.created_at,
|
||||||
mrq=mrq
|
mrq=mrq,
|
||||||
|
rt_target=rt_target,
|
||||||
|
rt_author_id=rt_author_id
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, tweet_id: int, author_id: int,date_time: datetime.datetime, mrq: tuple):
|
@staticmethod
|
||||||
|
async def create_from_id(id):
|
||||||
|
resp = await TwAPI.instance.get_tweet_response(id)
|
||||||
|
return TalentTweet.create_from_v2api_response(resp)
|
||||||
|
|
||||||
|
def __init__(self, tweet_id: int, author_id: int, date_time: datetime.datetime, mrq: tuple, rt_target: int=None, rt_author_id: int=None):
|
||||||
self.tweet_id, self.author_id = tweet_id, author_id
|
self.tweet_id, self.author_id = tweet_id, author_id
|
||||||
self.date_time = date_time
|
self.date_time = date_time
|
||||||
self.mentions = tuple(int(x) for x in mrq[0])
|
self.mentions = tuple(int(x) for x in mrq[0])
|
||||||
self.reply_to = int(mrq[1]) if mrq[1] is not None else None
|
self.reply_to = int(mrq[1]) if mrq[1] is not None else None
|
||||||
self.quote_retweeted = int(mrq[2]) if mrq[2] is not None else None
|
self.quote_retweeted = int(mrq[2]) if mrq[2] is not None else None
|
||||||
|
self.rt_target, self.rt_author_id = rt_target, rt_author_id
|
||||||
|
|
||||||
# all users involved, except for the author
|
# all users involved, except for the author
|
||||||
self.all_parties = {self.reply_to, self.quote_retweeted}
|
self.all_parties = {self.reply_to, self.quote_retweeted}
|
||||||
|
|||||||
+15
-10
@@ -3,7 +3,6 @@ import datetime
|
|||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
import tweepy
|
import tweepy
|
||||||
import twint
|
|
||||||
|
|
||||||
import api_secrets
|
import api_secrets
|
||||||
import talenttweet as tt
|
import talenttweet as tt
|
||||||
@@ -13,7 +12,7 @@ class TwAPI:
|
|||||||
tweets_fetched = 0
|
tweets_fetched = 0
|
||||||
instance = None
|
instance = None
|
||||||
TWEET_MEDIA_FIELDS = ['url']
|
TWEET_MEDIA_FIELDS = ['url']
|
||||||
TWEET_FIELDS = ['created_at', 'in_reply_to_user_id']
|
TWEET_FIELDS = ['created_at', 'in_reply_to_user_id', 'referenced_tweets']
|
||||||
TWEET_EXPANSIONS = ['entities.mentions.username', 'referenced_tweets.id.author_id']
|
TWEET_EXPANSIONS = ['entities.mentions.username', 'referenced_tweets.id.author_id']
|
||||||
|
|
||||||
# Returns a tuple of user IDs:(reply_to, qrt, {mentions})
|
# Returns a tuple of user IDs:(reply_to, qrt, {mentions})
|
||||||
@@ -171,10 +170,12 @@ class TwAPI:
|
|||||||
|
|
||||||
REPLY = '{0} replied to {1}!\n'
|
REPLY = '{0} replied to {1}!\n'
|
||||||
QUOTE_TWEET = '{0} quote tweeted {1}!\n'
|
QUOTE_TWEET = '{0} quote tweeted {1}!\n'
|
||||||
MENTION = '{0} tweeted!\n'
|
TWEET = '{0} tweeted!\n'
|
||||||
|
RETWEET = '{0} retweeted {1}!\n'
|
||||||
|
|
||||||
def create_text():
|
def create_text():
|
||||||
author_username = f'@/{util.get_username_online(ttweet.author_id)}'
|
author_username = f'@/{util.get_username_local(ttweet.author_id)}'
|
||||||
|
mention_ids = set()
|
||||||
ret = str()
|
ret = str()
|
||||||
if is_catchup:
|
if is_catchup:
|
||||||
# ret += '[catch-up tweet]\n'
|
# ret += '[catch-up tweet]\n'
|
||||||
@@ -182,8 +183,11 @@ class TwAPI:
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
# Tweet types
|
# Tweet types
|
||||||
if ttweet.reply_to is not None: # reply (w/ qrt; push it into mentions)
|
if ttweet.rt_target is not None: # standalone tweet
|
||||||
reply_username = f'@/{util.get_username_online(ttweet.reply_to)}'
|
ret += RETWEET.format(author_username, f'@/{util.get_username(ttweet.rt_author_id)}')
|
||||||
|
mention_ids.clear()
|
||||||
|
elif ttweet.reply_to is not None: # reply (w/ qrt; push it into mentions)
|
||||||
|
reply_username = f'@/{util.get_username_local(ttweet.reply_to)}'
|
||||||
ret += REPLY.format(author_username, reply_username)
|
ret += REPLY.format(author_username, reply_username)
|
||||||
|
|
||||||
mention_ids = set(ttweet.mentions)
|
mention_ids = set(ttweet.mentions)
|
||||||
@@ -191,16 +195,16 @@ class TwAPI:
|
|||||||
try: mention_ids.remove(None)
|
try: mention_ids.remove(None)
|
||||||
except: pass
|
except: pass
|
||||||
elif ttweet.quote_retweeted is not None: # standalone qrt
|
elif ttweet.quote_retweeted is not None: # standalone qrt
|
||||||
quoted_username = f'@/{util.get_username_online(ttweet.quote_retweeted)}'
|
quoted_username = f'@/{util.get_username_local(ttweet.quote_retweeted)}'
|
||||||
ret += QUOTE_TWEET.format(author_username, quoted_username)
|
ret += QUOTE_TWEET.format(author_username, quoted_username)
|
||||||
elif len(ttweet.mentions) > 0: # standalone tweet w/ mentions
|
elif len(ttweet.mentions) > 0: # standalone tweet w/ mentions
|
||||||
ret += MENTION.format(author_username)
|
ret += TWEET.format(author_username)
|
||||||
else:
|
else:
|
||||||
raise ValueError(f'TalentTweet {ttweet.tweet_id} has insufficient other parties')
|
raise ValueError(f'TalentTweet {ttweet.tweet_id} has insufficient other parties')
|
||||||
|
|
||||||
# mention line
|
# mention line
|
||||||
if len(mention_ids) > 0:
|
if len(mention_ids) > 0:
|
||||||
mention_usernames = [f'@/{util.get_username_online(x)}' for x in mention_ids]
|
mention_usernames = [f'@/{util.get_username_local(x)}' for x in mention_ids]
|
||||||
ret += (
|
ret += (
|
||||||
'mentioning '
|
'mentioning '
|
||||||
f'{" ".join(mention_usernames)}\n'
|
f'{" ".join(mention_usernames)}\n'
|
||||||
@@ -210,11 +214,12 @@ class TwAPI:
|
|||||||
|
|
||||||
img_media_id_task = asyncio.create_task(self.get_ttweet_image_media_id(ttweet))
|
img_media_id_task = asyncio.create_task(self.get_ttweet_image_media_id(ttweet))
|
||||||
text = create_text()
|
text = create_text()
|
||||||
media_id = await img_media_id_task
|
|
||||||
try:
|
try:
|
||||||
print('posting main tweet')
|
print('posting main tweet')
|
||||||
twt_resp = await self.post_tweet(text)
|
twt_resp = await self.post_tweet(text)
|
||||||
twt_id = twt_resp.data['id']
|
twt_id = twt_resp.data['id']
|
||||||
|
print('waiting on reply img')
|
||||||
|
media_id = await img_media_id_task
|
||||||
print('posting reply tweet')
|
print('posting reply tweet')
|
||||||
await self.post_tweet(reply_to_tweet=twt_id, media_id=media_id,)
|
await self.post_tweet(reply_to_tweet=twt_id, media_id=media_id,)
|
||||||
print('successfully posted ttweet!')
|
print('successfully posted ttweet!')
|
||||||
|
|||||||
+34
-13
@@ -1,10 +1,12 @@
|
|||||||
## Shared utility functions.
|
## Shared utility functions.
|
||||||
|
|
||||||
import datetime
|
|
||||||
import os
|
import os
|
||||||
|
import traceback
|
||||||
|
import datetime
|
||||||
|
|
||||||
import pytz
|
import pytz
|
||||||
import twint
|
import twint
|
||||||
|
import twapi
|
||||||
from tweetcapture import TweetCapture
|
from tweetcapture import TweetCapture
|
||||||
|
|
||||||
import talent_lists
|
import talent_lists
|
||||||
@@ -60,21 +62,40 @@ def ttweet_to_url(ttweet):
|
|||||||
username = get_username_online(ttweet.author_id)
|
username = get_username_online(ttweet.author_id)
|
||||||
return f'https://twitter.com/{username}/status/{ttweet.tweet_id}'
|
return f'https://twitter.com/{username}/status/{ttweet.tweet_id}'
|
||||||
|
|
||||||
def get_username_local(user_id):
|
def get_username_local(id):
|
||||||
return talent_lists.talents.get(user_id, f'#{id}')
|
return talent_lists.talents.get(id, f'{id}')
|
||||||
|
|
||||||
def get_username_online(user_id):
|
# twint
|
||||||
c = twint.Config()
|
# May not work with short user IDs (ie. 1354241437)
|
||||||
c.User_id = user_id
|
# def get_username_online(id, default=None):
|
||||||
c.Store_object = True
|
# c = twint.Config()
|
||||||
c.Hide_output = True
|
# c.User_id = id
|
||||||
|
# c.Store_object = True
|
||||||
|
# c.Hide_output = True
|
||||||
|
# try:
|
||||||
|
# twint.output.users_list.clear()
|
||||||
|
# twint.run.Lookup(c)
|
||||||
|
# user = twint.output.users_list[0]
|
||||||
|
# return user.username
|
||||||
|
# except:
|
||||||
|
# return str(default) if default is not None else f'{id}'
|
||||||
|
|
||||||
|
# API v2 (tweepy)
|
||||||
|
# Short user IDs (ie. 1354241437) apparently don't work with twint
|
||||||
|
def get_username_online(id, default=None):
|
||||||
try:
|
try:
|
||||||
twint.output.users_list.clear()
|
resp = twapi.TwAPI.instance.client.get_user(id=id)
|
||||||
twint.run.Lookup(c)
|
return resp.data.username
|
||||||
user = twint.output.users_list[0]
|
|
||||||
return user.username
|
|
||||||
except:
|
except:
|
||||||
return f'#{user_id}'
|
print(f'Unhandled error retrieving username for {id}!')
|
||||||
|
traceback.print_exc()
|
||||||
|
return str(default) if default is not None else f'{id}'
|
||||||
|
|
||||||
|
## Attempt to pull username from local; pull from online if doesn't exist.
|
||||||
|
def get_username(id):
|
||||||
|
ret = talent_lists.talents.get(id, None)
|
||||||
|
if ret == None:
|
||||||
|
return get_username_online(id)
|
||||||
|
|
||||||
def get_user_id_local(username) -> int:
|
def get_user_id_local(username) -> int:
|
||||||
talent_usernames = list(talent_lists.talents.values())
|
talent_usernames = list(talent_lists.talents.values())
|
||||||
|
|||||||
Reference in New Issue
Block a user