3
3
import time
4
4
import os
5
5
import logging
6
- from typing import List , Tuple
6
+ from typing import List
7
7
from dotenv import load_dotenv
8
8
9
9
# Load environment variables from .env file
10
10
load_dotenv ()
11
11
12
12
# Configure logging
13
13
logging .basicConfig (
14
- level = logging .INFO ,
14
+ level = logging .INFO , # Set to INFO for general logs; use DEBUG for more verbosity
15
15
format = '%(asctime)s [%(levelname)s] %(message)s' ,
16
16
handlers = [
17
17
logging .FileHandler ("follow_users.log" ),
28
28
logging .error ("GitHub token not found. Please set GITHUB_TOKEN in the environment variables." )
29
29
exit (1 )
30
30
31
- # Semaphore to limit concurrent requests
32
- SEM = asyncio .Semaphore (5 ) # Adjust the number based on your needs and testing
31
+ # Semaphore to limit concurrent requests (set to 1 for sequential processing)
32
+ SEM = asyncio .Semaphore (1 )
33
33
34
34
# Function to read usernames from a file
35
35
def read_usernames (file_path : str ) -> List [str ]:
@@ -72,47 +72,25 @@ def write_last_line(file_path: str, line_number: int) -> None:
72
72
logging .exception (f"An error occurred while writing to '{ file_path } ': { e } " )
73
73
74
74
# Asynchronous function to follow a user on GitHub
75
- async def follow_user (session : aiohttp .ClientSession , username : str ) -> Tuple [ int , str ] :
75
+ async def follow_user (session : aiohttp .ClientSession , username : str , line_number : int ) -> None :
76
76
url = f'https://api.github.com/user/following/{ username } '
77
- async with SEM : # Limit concurrency
77
+ async with SEM : # Ensure sequential processing
78
78
try :
79
79
async with session .put (url ) as response :
80
80
status = response .status
81
81
text = await response .text ()
82
82
83
- # Log rate limit headers
84
- rate_limit_remaining = response .headers .get ('X-RateLimit-Remaining' )
85
- rate_limit_reset = response .headers .get ('X-RateLimit-Reset' )
86
- if rate_limit_remaining and rate_limit_reset :
87
- logging .debug (f"Rate Limit Remaining: { rate_limit_remaining } " )
88
- logging .debug (f"Rate Limit Reset Time: { rate_limit_reset } " )
83
+ if status == 204 :
84
+ logging .info (f"Line { line_number + 1 } : Successfully followed '{ username } '." )
85
+ elif status == 404 :
86
+ logging .warning (f"Line { line_number + 1 } : User '{ username } ' not found." )
87
+ elif status == 403 or status == 429 :
88
+ logging .error (f"Line { line_number + 1 } : Rate limit exceeded or forbidden access." )
89
+ else :
90
+ logging .error (f"Line { line_number + 1 } : Failed to follow '{ username } ': { status } , { text } " )
89
91
90
- return status , text
91
92
except Exception as e :
92
- logging .exception (f"Error following user '{ username } ': { e } " )
93
- return 0 , str (e )
94
-
95
- # Function to handle rate limiting based on GitHub's response headers
96
- async def handle_rate_limit (headers : dict ):
97
- rate_limit_remaining = headers .get ('X-RateLimit-Remaining' )
98
- rate_limit_reset = headers .get ('X-RateLimit-Reset' )
99
-
100
- if rate_limit_remaining is not None and rate_limit_reset is not None :
101
- rate_limit_remaining = int (rate_limit_remaining )
102
- rate_limit_reset = int (rate_limit_reset )
103
-
104
- if rate_limit_remaining == 0 :
105
- current_time = int (time .time ())
106
- sleep_duration = rate_limit_reset - current_time + 5 # Add a buffer of 5 seconds
107
- if sleep_duration > 0 :
108
- reset_time_str = time .strftime ('%Y-%m-%d %H:%M:%S' , time .localtime (rate_limit_reset ))
109
- logging .warning (f"Rate limit reached. Sleeping until { reset_time_str } ({ sleep_duration } seconds)." )
110
- await asyncio .sleep (sleep_duration )
111
-
112
- # Function to check and handle rate limits after each request
113
- async def check_rate_limit_after_request (response : aiohttp .ClientResponse ):
114
- headers = response .headers
115
- await handle_rate_limit (headers )
93
+ logging .exception (f"Line { line_number + 1 } : Error following user '{ username } ': { e } " )
116
94
117
95
# Main asynchronous function
118
96
async def main ():
@@ -129,27 +107,15 @@ async def main():
129
107
130
108
async with aiohttp .ClientSession (headers = headers ) as session :
131
109
for i , username in enumerate (usernames [last_line :], start = last_line ):
132
- status_code , response_text = await follow_user (session , username )
133
-
134
- if status_code == 204 :
135
- logging .info (f"Line { i + 1 } : Successfully followed '{ username } '." )
136
- write_last_line (LAST_LINE_FILE , i + 1 )
137
- elif status_code == 404 :
138
- logging .warning (f"Line { i + 1 } : User '{ username } ' not found." )
139
- write_last_line (LAST_LINE_FILE , i + 1 )
140
- elif status_code == 403 :
141
- if 'rate limit' in response_text .lower ():
142
- logging .error (f"Line { i + 1 } : Rate limit exceeded." )
143
- # Extract headers from the last response
144
- await handle_rate_limit (session ._connector ._session .headers )
145
- else :
146
- logging .error (f"Line { i + 1 } : Forbidden access when trying to follow '{ username } '." )
147
- else :
148
- logging .error (f"Line { i + 1 } : Failed to follow '{ username } ': { status_code } , { response_text } " )
110
+ await follow_user (session , username , i )
149
111
150
- # Optional: Dynamic sleep based on remaining rate limit
151
- # This example uses a fixed sleep; you can adjust it based on rate limits
152
- await asyncio .sleep (1 ) # Adjust as needed
112
+ # Wait for 10 seconds before processing the next user
113
+ if i < total_usernames - 1 :
114
+ #logging.info("Waiting for 10 seconds before following the next user...")
115
+ await asyncio .sleep (10 )
116
+
117
+ # Update the last processed line
118
+ write_last_line (LAST_LINE_FILE , i + 1 )
153
119
154
120
logging .info ("Finished processing all usernames." )
155
121
@@ -160,5 +126,3 @@ async def main():
160
126
logging .info ("Script interrupted by user." )
161
127
except Exception as e :
162
128
logging .exception (f"An unexpected error occurred: { e } " )
163
-
164
- ###test!
0 commit comments