This repository has been archived on 2022-08-14. You can view files and clone it, but cannot push or open issues or pull requests.
feurBot/main.py

286 lines
13 KiB
Python
Raw Normal View History

2021-08-03 14:17:00 +02:00
from dotenv import load_dotenv
from os import environ
from tweepy import OAuthHandler, API, StreamListener, Stream
from re import sub, findall
2021-08-03 19:19:30 +02:00
from random import choice
2021-08-03 20:27:06 +02:00
from datetime import datetime
2021-08-03 19:59:32 +02:00
from pytz import timezone
2021-08-04 14:04:52 +02:00
from queue import Queue
2021-08-07 03:12:51 +02:00
from json import loads
2021-08-03 14:17:00 +02:00
def load(variables) -> dict:
"""Load environment variables."""
2021-08-03 14:17:00 +02:00
keys = {}
load_dotenv() # load .env file
for var in variables:
try:
2021-08-05 20:26:25 +02:00
if var == "VERBOSE": # check is VERBOSE is set
try:
res = bool(environ[var])
except:
2021-08-05 20:26:25 +02:00
res = False # if not its False
elif var == "WHITELIST": # check if WHITELIST is set
try:
res = list(set(environ[var].split(",")) - {""})
except:
res = [] # if not its an empty list
else:
res = environ[var]
if var == "PSEUDOS":
2021-08-05 18:43:13 +02:00
res = list(set(res.split(",")) - {""}) # create a list for the channels and remove blank channels and doubles
keys[var] = res
2021-08-03 14:17:00 +02:00
except KeyError:
2021-08-04 18:30:49 +02:00
print(f"Veuillez définir la variable d'environnement {var} (fichier .env supporté)")
2021-08-03 14:17:00 +02:00
exit(1)
return keys
def cleanTweet(tweet: str) -> str:
2021-11-02 10:39:51 +01:00
"""Remove all unwanted elements from the tweet."""
2021-08-06 02:51:36 +02:00
tweet = tweet.lower() # convert to lower case
tweet = sub(r"(https?:\/\/\S+|www.\S+)", " ", tweet) # remove URLs
2022-05-12 15:21:01 +02:00
hashtagMatch = findall(r"#\S+", tweet) # check all hashtags
if len(hashtagMatch) < 3: # if less than 3
tweet = sub(r"#\S+", " ", tweet) # remove them
2021-08-05 19:52:54 +02:00
else:
2022-05-12 15:21:01 +02:00
return "" # too much hashtags, ignoring tweet
2021-08-06 02:51:36 +02:00
tweet = sub(r"@\S+", " ", tweet) # remove usernames
tweet = sub(r" *?[^\w\s]+", " ", tweet) # remove everything who is not a letter or a number or a space
2022-05-12 15:21:01 +02:00
tweet = sub(r"\S+(?=si|ci)", " ", tweet) # remove element of the word only if the last syllable can be matched (so more words will be answered without adding them manually)
tweet = sub(r"(?<=ui)i+|(?<=na)a+(?<!n)|(?<=quoi)i+|(?<=no)o+(?<!n)|(?<=hei)i+(?<!n)|(?<=si)i+", "", tweet) # remove key smashing in certains words
2021-11-02 10:39:51 +01:00
return tweet.strip()
2021-08-03 14:17:00 +02:00
class Listener(StreamListener):
2021-08-05 18:50:33 +02:00
"""Watch for tweets that match criteria in real-time."""
2021-08-04 14:04:52 +02:00
def __init__(self, api = None, users = None, q = Queue()):
2021-08-03 14:17:00 +02:00
super(Listener, self).__init__()
2021-08-04 14:04:52 +02:00
self.q = q
2021-08-03 14:17:00 +02:00
self.api = api
self.users = users
self.listOfFriendsID = getFriendsID(api, users)
2021-08-05 18:51:14 +02:00
def on_connect(self):
2021-08-05 18:47:48 +02:00
print(f"Scroll sur Twitter avec les abonnements de @{', @'.join(self.users)} comme timeline...")
2021-11-02 10:39:51 +01:00
2021-08-05 18:41:30 +02:00
def on_disconnect(notice):
notice = notice["disconnect"]
print(f"Déconnexion (code {notice['code']}).", end = " ")
2021-08-05 18:43:13 +02:00
if len(notice["reason"]) > 0:
2021-08-05 18:41:30 +02:00
print(f"Raison : {notice['reason']}")
2021-08-03 14:17:00 +02:00
def on_status(self, status):
2021-08-05 20:26:25 +02:00
if status._json["user"]["id"] in self.listOfFriendsID and status._json["user"]["screen_name"] not in keys["WHITELIST"]: # verification of the author of the tweet
2021-08-04 16:00:05 +02:00
if seniority(status._json["created_at"]): # verification of the age of the tweet
if not hasattr(status, "retweeted_status"): # ignore Retweet
if "extended_tweet" in status._json:
tweet = cleanTweet(status.extended_tweet["full_text"])
else:
tweet = cleanTweet(status.text)
lastWord = tweet.split()[-1:][0]
if keys["VERBOSE"]:
2021-08-05 19:52:54 +02:00
infoLastWord = f"dernier mot : \"{lastWord}\"" if len(lastWord) > 0 else "tweet ignoré car trop de hashtags"
print(f"Tweet trouvé de {status._json['user']['screen_name']} ({infoLastWord})...", end = " ")
if lastWord in universalBase: # check if the last word found is a supported word
answer = None
for mot in base.items():
if lastWord in mot[1]:
2021-08-06 01:53:33 +02:00
if mot[0] == "bon":
2021-08-06 01:55:59 +02:00
if datetime.now().hour in range(7, 17): # between 7am and 5pm
2021-08-06 01:53:33 +02:00
answer = answers[mot[0]][0] # jour
else:
answer = answers[mot[0]][1] # soir
else:
answer = answers[mot[0]]
if answer == None:
if keys["VERBOSE"]:
print(f"{errorMessage} Aucune réponse trouvée.")
else:
if keys["VERBOSE"]:
print(f"Envoie d'un {answer[0]}...", end = " ")
try: # send answer
self.api.update_status(status = choice(answer), in_reply_to_status_id = status._json["id"], auto_populate_reply_metadata = True)
print(f"{status._json['user']['screen_name']} s'est fait {answer[0]} !")
except Exception as error:
2021-08-07 03:12:51 +02:00
error = loads(error.response.text)["errors"][0]
if error["code"] == 385:
error["message"] = "Tweet supprimé ou auteur en privé/bloqué."
print(f"{errorMessage[:-2]} ({error['code']}) ! {error['message']}")
2021-08-04 18:30:49 +02:00
else:
if keys["VERBOSE"]:
print("Annulation car le dernier mot n'est pas intéressant.")
2021-08-04 14:05:20 +02:00
2021-08-04 14:04:52 +02:00
def do_stuff(self):
while True:
self.q.get()
self.q.task_done()
2021-08-03 19:59:32 +02:00
def on_error(self, status_code):
print(f"{errorMessage[:-2]} ({status_code}) !", end = " ")
if status_code == 413:
2021-08-05 18:30:43 +02:00
if keys["VERBOSE"]:
print("La liste des mots est trop longue (triggerWords).")
elif status_code == 420:
2021-08-05 18:30:43 +02:00
if keys["VERBOSE"]:
print("Déconnecter du flux.")
else:
print("\n")
return False
def getFriendsID(api, users: list) -> list:
"""Get all friends of choosen users."""
liste = []
for user in users:
liste.extend(api.friends_ids(user))
2021-08-04 01:06:16 +02:00
return list(set(liste))
def seniority(date: str) -> bool:
"""Return True only if the given string date is less than one day old."""
2021-08-05 20:26:25 +02:00
datetimeObject = datetime.strptime(date, "%a %b %d %H:%M:%S +0000 %Y") # convert String format to datetime format
2022-05-12 15:21:01 +02:00
datetimeObject = datetimeObject.replace(tzinfo = timezone("UTC")) # Twitter give us an UTC time
age = datetime.now(timezone("UTC")) - datetimeObject # time now in UTC minus the time we got to get the age of the date
return False if age.days >= 1 else True # False if older than a day
2021-08-03 14:17:00 +02:00
def permute(array: list) -> list:
"""Retrieves all possible combinations for the given list and returns the result as a list."""
2021-08-03 23:33:53 +02:00
quoiListe = []
for text in array: # all element of the list
if text.lower() not in quoiListe:
2021-08-05 20:26:25 +02:00
quoiListe.append(text.lower()) # word fully in lowercase
if text.upper() not in quoiListe:
2021-08-05 20:26:25 +02:00
quoiListe.append(text.upper()) # word fully in uppercase
if text.capitalize() not in quoiListe:
2021-08-05 20:26:25 +02:00
quoiListe.append(text.capitalize()) # word with the first letter in uppercase
2021-08-03 23:33:53 +02:00
return quoiListe
2021-08-04 18:30:49 +02:00
def createBaseTrigger(lists) -> list:
2021-08-04 16:00:05 +02:00
"""Merges all given lists into one."""
listing = []
for liste in lists:
listing.extend(liste)
return list(set(listing))
def createBaseAnswers(word) -> list:
2021-08-04 16:00:05 +02:00
"""Generates default answers for a given word."""
2021-08-04 15:45:11 +02:00
return [word, f"({word})", word.upper(), f"{word} lol"]
def main(accessToken: str, accessTokenSecret: str, consumerKey: str, consumerSecret: str, users: list):
2021-08-03 14:17:00 +02:00
"""Main method."""
auth = OAuthHandler(consumerKey, consumerSecret)
auth.set_access_token(accessToken, accessTokenSecret)
2021-11-02 10:39:51 +01:00
2021-08-03 23:11:17 +02:00
api = API(auth_handler = auth, wait_on_rate_limit = True)
2021-08-03 14:17:00 +02:00
2021-08-05 17:04:20 +02:00
if keys["VERBOSE"]:
try:
api.verify_credentials()
print(f"Authentification réussie en tant que", end = " ")
except:
print("Erreur d'authentification.")
exit(1)
print(f"@{api.me()._json['screen_name']}.")
2021-11-02 10:39:51 +01:00
2021-08-05 20:26:25 +02:00
if keys['WHITELIST'] == []:
whitelist = "Aucun"
else:
whitelist = f"@{', @'.join(keys['WHITELIST'])}"
2021-08-05 21:47:44 +02:00
print(f"Liste des comptes ignorés : {whitelist}.")
2021-08-05 17:04:20 +02:00
listener = Listener(api, users)
2021-08-03 19:19:30 +02:00
stream = Stream(auth = api.auth, listener = listener)
2021-08-04 14:08:10 +02:00
stream.filter(track = triggerWords, languages = ["fr"], stall_warnings = True, is_async = True)
2021-08-03 14:17:00 +02:00
2021-08-05 18:43:13 +02:00
if __name__ == "__main__":
2021-08-03 14:17:00 +02:00
"""
2021-08-03 19:19:30 +02:00
TOKEN is the Access Token available in the Authentication Tokens section under Access Token and Secret sub-heading.
TOKEN_SECRET is the Access Token Secret available in the Authentication Tokens section under Access Token and Secret sub-heading.
CONSUMER_KEY is the API Key available in the Consumer Keys section.
CONSUMER_SECRET is the API Secret Key available in the Consumer Keys section.
2021-08-03 14:17:00 +02:00
--
2021-08-03 23:11:17 +02:00
PSEUDO is the PSEUDO of the account you want to listen to snipe.
2021-08-03 14:17:00 +02:00
"""
2021-08-04 18:30:49 +02:00
errorMessage = "Une erreur survient !" # error message
2021-08-05 20:26:25 +02:00
base = { # words to detect in lowercase
2021-08-04 19:40:03 +02:00
"quoi": ["quoi", "koi", "quoient"],
2021-08-06 01:57:53 +02:00
"oui": ["oui", "ui", "wi"],
2021-08-04 18:33:55 +02:00
"non": ["non", "nn"],
2021-08-05 12:32:05 +02:00
"nan": ["nan"],
2022-05-17 22:17:33 +02:00
"hein": ["hein", "1", "un"],
2021-08-06 02:51:36 +02:00
"ci": ["ci", "si"],
2021-08-05 12:32:05 +02:00
"con": ["con"],
"ok": ["ok", "okay", "oké", "k"],
2021-08-05 12:32:05 +02:00
"ouais": ["ouais", "oué"],
"comment": ["comment"],
2021-08-05 20:35:09 +02:00
"mais": ["mais", ""],
2021-08-05 22:16:29 +02:00
"fort": ["fort"],
2021-08-06 01:44:12 +02:00
"coup": ["coup", "cou"],
2021-08-07 01:25:31 +02:00
"ça": ["ça", "sa"],
2021-08-06 01:57:53 +02:00
"bon": ["bon"],
2021-08-07 01:25:31 +02:00
"qui": ["qui", "ki"],
"sur": ["sur", "sûr"],
"pas": ["pas", "pa"],
2021-08-07 10:39:38 +02:00
"ka": ["ka", "kha"],
2022-01-26 10:09:38 +01:00
"fais": ["fais", "fait"],
2022-05-12 15:29:19 +02:00
"tant": ["tant", "temps", "tend", "tends"],
"et": ["et"],
"la": ["la", ""],
"tki": ["tki"],
"moi": ["moi", "mwa"],
2022-07-20 01:53:52 +02:00
"toi": ["toi", "toit"],
"top": ["top"],
"jour": ["jour", "bonjour"]
2021-08-04 18:30:49 +02:00
}
2021-08-05 20:26:25 +02:00
answers = { # creation of answers
2021-08-04 18:30:49 +02:00
"quoi": createBaseAnswers("feur") + [
2021-08-05 12:32:05 +02:00
"https://twitter.com/Myshawii/status/1423219640025722880/video/1",
2021-08-04 18:30:49 +02:00
"feur (-isson)",
2021-08-05 12:32:05 +02:00
"https://twitter.com/Myshawii/status/1423219684552417281/video/1",
2021-11-02 10:39:51 +01:00
"feur (-isson -ictalope -diatre -uil)",
"https://twitter.com/Myshawii/status/1455469162202075138/video/1"
2022-05-12 15:29:19 +02:00
] + createBaseAnswers("feuse"),
2022-02-07 08:18:12 +01:00
"oui": createBaseAnswers("stiti") + createBaseAnswers("fi"),
2021-08-04 18:33:55 +02:00
"non": createBaseAnswers("bril"),
2021-08-05 12:32:05 +02:00
"nan": createBaseAnswers("cy"),
2022-05-17 22:17:33 +02:00
"hein": createBaseAnswers("deux") + createBaseAnswers("bécile") + [
2021-08-05 12:32:05 +02:00
"2"
],
2022-05-12 15:29:19 +02:00
"ci": createBaseAnswers("tron") + createBaseAnswers("prine"),
2021-08-05 12:32:05 +02:00
"con": createBaseAnswers("combre"),
"ok": createBaseAnswers("sur glace"),
"ouais": createBaseAnswers("stern"),
2022-05-12 15:29:19 +02:00
"comment": createBaseAnswers("tateur") + createBaseAnswers("trice") + createBaseAnswers("dant Cousteau"),
2021-08-07 01:33:18 +02:00
"mais": createBaseAnswers("on") + [
"on (-dulation)"
],
2021-08-05 21:04:32 +02:00
"fort": createBaseAnswers("boyard") + [
"boyard (-ennes)"
2021-08-05 22:16:29 +02:00
],
"coup": createBaseAnswers("teau"),
2021-08-07 01:25:31 +02:00
"ça": createBaseAnswers("pristi") + createBaseAnswers("perlipopette"),
2021-08-06 01:57:53 +02:00
"bon": [createBaseAnswers("jour"), createBaseAnswers("soir")],
2021-08-07 01:25:31 +02:00
"qui": createBaseAnswers("wi") + createBaseAnswers("mono"),
"sur": createBaseAnswers("prise"),
"pas": createBaseAnswers("nini"),
2022-05-12 15:29:19 +02:00
"ka": createBaseAnswers("pitaine") + createBaseAnswers("pitulation"),
2022-01-26 10:09:38 +01:00
"fais": createBaseAnswers("rtile"),
2022-05-12 15:29:19 +02:00
"tant": createBaseAnswers("gente"),
"et": createBaseAnswers("eint") + createBaseAnswers("ain"),
"la": createBaseAnswers("vabo") + createBaseAnswers("vande"),
"tki": createBaseAnswers("la"),
"moi": createBaseAnswers("tié") + createBaseAnswers("sson"),
2022-07-20 01:53:52 +02:00
"toi": createBaseAnswers("lette"),
"top": createBaseAnswers("inambour"),
"jour": createBaseAnswers("nal")
2021-08-04 18:30:49 +02:00
}
2021-08-04 19:40:03 +02:00
2021-08-05 20:26:25 +02:00
universalBase = createBaseTrigger(list(base.values())) # creation of a list of all the words
2021-08-04 19:40:03 +02:00
2021-08-05 20:26:25 +02:00
triggerWords = permute(universalBase) # creation of a list of all the words (upper and lower case)
2021-08-04 19:40:03 +02:00
2022-05-12 15:21:01 +02:00
# Loading environment variables and launching the bot
2021-08-05 20:26:25 +02:00
keys = load(["TOKEN", "TOKEN_SECRET", "CONSUMER_KEY", "CONSUMER_SECRET", "PSEUDOS", "VERBOSE", "WHITELIST"])
2021-08-06 10:41:03 +02:00
print("") # just a newline
main(keys["TOKEN"], keys["TOKEN_SECRET"], keys["CONSUMER_KEY"], keys["CONSUMER_SECRET"], keys["PSEUDOS"])