-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
118 lines (91 loc) · 4.16 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
import python_weather
import instagrapi
import asyncio
import os
import logging
from random import sample, choice
from dotenv import load_dotenv
from enum import Enum
from instagrapi.exceptions import LoginRequired
from utils import get_logger
load_dotenv()
logger = get_logger()
INSTAGRAM_USERNAME = os.environ.get("INSTAGRAM_USERNAME")
INSTAGRAM_PASSWORD = os.environ.get("INSTAGRAM_PASSWORD")
class WeatherType(Enum):
RAIN_LIGHT = 1
RAIN_MEDIUM = 2
RAIN_HEAVY = 3
async def main() -> None:
logger.info("Getting Brittany cities.")
with open("cities.txt", "r", encoding="utf-8") as f:
cities = f.read().split("\n")
logger.info("Getting 50 random cities from Brittany.")
random_cities = sample(cities, 50)
logger.info("Getting precipitation for each city :")
cities_precipitation = {}
async with python_weather.Client(unit=python_weather.METRIC) as weather_client:
for city in random_cities:
logger.info(" Getting weather data for %s." % city)
weather = await weather_client.get(f"{city}, Bretagne, France", locale=python_weather.Locale.FRENCH)
if weather.current.precipitation >= 4:
cities_precipitation[city] = WeatherType.RAIN_HEAVY
elif weather.current.precipitation >= 2:
cities_precipitation[city] = WeatherType.RAIN_MEDIUM
elif weather.current.precipitation > 0:
cities_precipitation[city] = WeatherType.RAIN_LIGHT
else:
continue
if len(cities_precipitation) == 0:
logger.info("No rain detected on any of the cities, exiting.")
logging.shutdown()
return
logger.info("Rain detected on %d cities, selecting one randomly." % len(cities_precipitation))
city = choice(list(cities_precipitation.keys()))
rain_type = cities_precipitation[city]
logger.info(f"{rain_type.name} detected on {city}, posting to Instagram the corresponding picture and caption.")
logger.info("Logging in to Instagram.")
instagram_client = login_instagram()
with open(f"texts/{rain_type.name.lower()}.txt", "r", encoding="utf-8") as _caption_file:
caption = _caption_file.read()
instagram_client.photo_upload(f"images/generated/{rain_type.name.lower()}_{city.replace(' ', '_')}.jpg", caption)
logger.info("Media posted.")
instagram_client.dump_settings("instagram_session.json")
logger.info("Session saved, exiting.")
logging.shutdown()
def login_instagram() -> instagrapi.Client:
instagram_client = instagrapi.Client()
session = instagram_client.load_settings("instagram_session.json")
login_via_session = False
login_via_pw = False
if session:
try:
instagram_client.set_settings(session)
instagram_client.login(INSTAGRAM_USERNAME, INSTAGRAM_PASSWORD)
try:
instagram_client.get_timeline_feed()
except LoginRequired:
logger.info("Session is invalid, need to login via username and password")
old_session = instagram_client.get_settings()
instagram_client.set_settings({})
instagram_client.set_uuids(old_session["uuids"])
instagram_client.login(INSTAGRAM_USERNAME, INSTAGRAM_PASSWORD)
login_via_session = True
except Exception as e:
logger.error("Couldn't login user using session information: %s" % e)
if not login_via_session:
try:
logger.info("Attempting to login via username and password. username: %s" % INSTAGRAM_USERNAME)
if instagram_client.login(INSTAGRAM_USERNAME, INSTAGRAM_PASSWORD):
login_via_pw = True
except Exception as e:
logger.error("Couldn't login user using username and password: %s" % e)
if not login_via_pw and not login_via_session:
raise Exception("Couldn't login user with either password or session")
else:
instagram_client.dump_settings("instagram_session.json")
return instagram_client
if __name__ == "__main__":
if os.name == "nt":
asyncio.set_event_loop_policy(asyncio.WindowsSelectorEventLoopPolicy())
asyncio.run(main())