@@ -43,16 +43,26 @@ def main():
43
43
44
44
# Get the values of the configuration variables, using default values if not available
45
45
max_ip = int (config .get ('DEFAULT' , 'max_ip' , fallback = DEFAULT_MAX_IP ))
46
- max_ping = int (config .get ('DEFAULT' , 'max_ping' , fallback = DEFAULT_MAX_PING ))
47
- max_jitter = int (config .get ('DEFAULT' , 'max_jitter' , fallback = DEFAULT_MAX_JITTER ))
48
- max_latency = int (config .get ('DEFAULT' , 'max_latency' , fallback = DEFAULT_MAX_LATENCY ))
49
- ip_include = config .get ('DEFAULT' , 'ip_include' , fallback = DEFAULT_IP_INCLUDE )
50
- ip_exclude = config .get ('DEFAULT' , 'ip_exclude' , fallback = DEFAULT_IP_EXCLUDE )
51
- test_size = config .get ('DEFAULT' , 'test_size' , fallback = DEFAULT_DOWNLOAD_SIZE_KB )
52
- min_download_speed = config .get ('DEFAULT' , 'min_download_speed' , fallback = DEFAULT_MIN_DOWNLOAD_SPEED )
53
- min_upload_speed = config .get ('DEFAULT' , 'min_upload_speed' , fallback = DEFAULT_MIN_UPLOAD_SPEED )
54
- default_upload_results = config .get ('DEFAULT' , 'upload_results' , fallback = 'no' )
55
- default_delete_existing = config .get ('DEFAULT' , 'delete_existing' , fallback = 'yes' )
46
+ max_ping = int (config .get ('DEFAULT' , 'max_ping' ,
47
+ fallback = DEFAULT_MAX_PING ))
48
+ max_jitter = int (config .get ('DEFAULT' , 'max_jitter' ,
49
+ fallback = DEFAULT_MAX_JITTER ))
50
+ max_latency = int (config .get ('DEFAULT' , 'max_latency' ,
51
+ fallback = DEFAULT_MAX_LATENCY ))
52
+ ip_include = config .get ('DEFAULT' , 'ip_include' ,
53
+ fallback = DEFAULT_IP_INCLUDE )
54
+ ip_exclude = config .get ('DEFAULT' , 'ip_exclude' ,
55
+ fallback = DEFAULT_IP_EXCLUDE )
56
+ test_size = config .get ('DEFAULT' , 'test_size' ,
57
+ fallback = DEFAULT_DOWNLOAD_SIZE_KB )
58
+ min_download_speed = config .get (
59
+ 'DEFAULT' , 'min_download_speed' , fallback = DEFAULT_MIN_DOWNLOAD_SPEED )
60
+ min_upload_speed = config .get (
61
+ 'DEFAULT' , 'min_upload_speed' , fallback = DEFAULT_MIN_UPLOAD_SPEED )
62
+ default_upload_results = config .get (
63
+ 'DEFAULT' , 'upload_results' , fallback = 'no' )
64
+ default_delete_existing = config .get (
65
+ 'DEFAULT' , 'delete_existing' , fallback = 'yes' )
56
66
default_email = config .get ('DEFAULT' , 'email' , fallback = '' )
57
67
default_zone_id = config .get ('DEFAULT' , 'zone_id' , fallback = '' )
58
68
default_api_key = config .get ('DEFAULT' , 'api_key' , fallback = '' )
@@ -76,12 +86,18 @@ def main():
76
86
max_ip = input (f"Enter max IP [{ max_ip } ]: " ) or max_ip
77
87
max_ping = input (f"Enter max ping [{ max_ping } ]: " ) or max_ping
78
88
max_jitter = input (f"Enter max jitter [{ max_jitter } ]: " ) or max_jitter
79
- max_latency = input (f"Enter max latency [{ max_latency } ]: " ) or max_latency
80
- ip_include = input (f"Enter IPs to include (comma seperated, '-' to ignore) [{ ip_include } ]: " ) or ip_include
81
- ip_exclude = input (f"Enter IPs to exclude (comma seperated, '-' to ignore) [{ ip_exclude } ]: " ) or ip_exclude
82
- test_size = input (f"Enter test data size in KB [{ test_size } ]: " ) or test_size
83
- min_download_speed = input (f"Enter minimum download speed (Mbps) [{ min_download_speed } ]: " ) or min_download_speed
84
- min_upload_speed = input (f"Enter minimum upload speed (Mbps) [{ min_upload_speed } ]: " ) or min_upload_speed
89
+ max_latency = input (
90
+ f"Enter max latency [{ max_latency } ]: " ) or max_latency
91
+ ip_include = input (
92
+ f"Enter IPs to include (comma seperated, '-' to ignore) [{ ip_include } ]: " ) or ip_include
93
+ ip_exclude = input (
94
+ f"Enter IPs to exclude (comma seperated, '-' to ignore) [{ ip_exclude } ]: " ) or ip_exclude
95
+ test_size = input (
96
+ f"Enter test data size in KB [{ test_size } ]: " ) or test_size
97
+ min_download_speed = input (
98
+ f"Enter minimum download speed (Mbps) [{ min_download_speed } ]: " ) or min_download_speed
99
+ min_upload_speed = input (
100
+ f"Enter minimum upload speed (Mbps) [{ min_upload_speed } ]: " ) or min_upload_speed
85
101
86
102
# Clear the include regex in case "-" provided by the user
87
103
if ip_include == '-' :
@@ -104,33 +120,41 @@ def main():
104
120
api_key = default_api_key
105
121
subdomain = default_subdomain
106
122
107
-
108
123
# Prompt the user for whether they want to upload the result to their Cloudflare subdomain
109
- upload_results = input (f"Do you want to upload the result to your Cloudflare subdomain (yes/no) [{ default_upload_results } ]? " ) or default_upload_results
124
+ upload_results = input (
125
+ f"Do you want to upload the result to your Cloudflare subdomain (yes/no) [{ default_upload_results } ]? " ) or default_upload_results
110
126
111
127
# Code block to execute if upload_results is 'y' or 'yes'
112
128
if upload_results .lower () in ["y" , "yes" ]:
113
- delete_existing = input (f"Do you want to delete extisting records of given subdomain before uploading the result to your Cloudflare (yes/no) [{ default_delete_existing } ]? " ) or default_delete_existing
114
- email = input (f"Cloudflare email [{ default_email } ]: " ) or default_email
115
- zone_id = input (f"Cloudflare zone ID [{ default_zone_id } ]: " ) or default_zone_id
116
- api_key = input (f"Cloudflare API key [{ default_api_key } ]: " ) or default_api_key
129
+ delete_existing = input (
130
+ f"Do you want to delete extisting records of given subdomain before uploading the result to your Cloudflare (yes/no) [{ default_delete_existing } ]? " ) or default_delete_existing
131
+ email = input (
132
+ f"Cloudflare email [{ default_email } ]: " ) or default_email
133
+ zone_id = input (
134
+ f"Cloudflare zone ID [{ default_zone_id } ]: " ) or default_zone_id
135
+ api_key = input (
136
+ f"Cloudflare API key [{ default_api_key } ]: " ) or default_api_key
117
137
118
138
# Prompt user to enter subdomain to modify
119
- subdomain = input (f"Subdomain to modify (i.e ip.my-domain.com) [{ default_subdomain } ]: " ) or default_subdomain
139
+ subdomain = input (
140
+ f"Subdomain to modify (i.e ip.my-domain.com) [{ default_subdomain } ]: " ) or default_subdomain
120
141
121
142
# Check if provided credentials are correct and retry if they are not
122
143
while not validateCloudflareCredentials (email , api_key , zone_id ):
123
144
print ("Invalid cloudflare credentials, please try again." )
124
- email = input (f"Cloudflare email [{ default_email } ]: " ) or default_email
125
- zone_id = input (f"Cloudflare zone ID [{ default_zone_id } ]: " ) or default_zone_id
126
- api_key = input (f"Cloudflare API key [{ default_api_key } ]: " ) or default_api_key
127
-
145
+ email = input (
146
+ f"Cloudflare email [{ default_email } ]: " ) or default_email
147
+ zone_id = input (
148
+ f"Cloudflare zone ID [{ default_zone_id } ]: " ) or default_zone_id
149
+ api_key = input (
150
+ f"Cloudflare API key [{ default_api_key } ]: " ) or default_api_key
128
151
129
152
# Use regular expression to validate subdomain format
130
153
while not re .match (r"^[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,}$" , subdomain ):
131
154
# If subdomain is invalid, prompt user to try again
132
155
print ("Invalid subdomain, please try again." )
133
- subdomain = input (f"Subdomain to modify (i.e ip.my-domain.com) [{ default_subdomain } ]: " ) or default_subdomain
156
+ subdomain = input (
157
+ f"Subdomain to modify (i.e ip.my-domain.com) [{ default_subdomain } ]: " ) or default_subdomain
134
158
135
159
# Update config variable with given data from user
136
160
config ['DEFAULT' ] = {
@@ -157,11 +181,13 @@ def main():
157
181
158
182
# Convert IP ranges to include (provided by user in a comma-seperated string) to Regular Expression
159
183
if ip_include :
160
- include_regex = re .compile ('|' .join (['^' + re .escape (c ).replace ('.' , '\\ .' ) + '\\ .' for c in ip_include .split (',' )]))
184
+ include_regex = re .compile ('|' .join (
185
+ ['^' + re .escape (c ).replace ('.' , '\\ .' ) + '\\ .' for c in ip_include .split (',' )]))
161
186
162
187
# Convert IP ranges to exclude (provided by user in a comma-seperated string) to Regular Expression
163
188
if ip_exclude :
164
- exclude_regex = re .compile ('|' .join (['^' + re .escape (c ).replace ('.' , '\\ .' ) + '\\ .' for c in ip_exclude .split (',' )]))
189
+ exclude_regex = re .compile ('|' .join (
190
+ ['^' + re .escape (c ).replace ('.' , '\\ .' ) + '\\ .' for c in ip_exclude .split (',' )]))
165
191
166
192
# Get IPv4 CIDR blocks of Cloudflare Network from related function
167
193
cidr_list = getCIDRv4Ranges ()
@@ -188,12 +214,11 @@ def main():
188
214
# Convert CIDR block to IP addresses and add them to IP List
189
215
ip_list = ip_list + processCIDR (cidr )
190
216
191
-
192
217
# Shuffling the IP list in order to test different ip in different ranges by random
193
218
print (f"\n Shuffling the IPs..." , end = '' )
194
219
random .shuffle (ip_list )
195
220
196
- # Preparation is done
221
+ # Preparation is done
197
222
print ("Done." )
198
223
except KeyboardInterrupt :
199
224
# Print proper message and exit the script in case user pressed CTRL+C
@@ -216,7 +241,8 @@ def main():
216
241
openssl_is_active = False
217
242
218
243
# Start testing clean IPs
219
- selectd_ip_list , total_test = curses .wrapper (startTest , ip_list = ip_list , config = config )
244
+ selectd_ip_list , total_test = curses .wrapper (
245
+ startTest , ip_list = ip_list , config = config )
220
246
221
247
print (f"\n { total_test } of { len (ip_list )} matched IPs have peen tested." )
222
248
print (f"{ len (selectd_ip_list )} IP(s) found:" )
@@ -240,17 +266,20 @@ def main():
240
266
# Check if user wanted to delete existing records of given subdomain
241
267
if delete_existing .lower () in ["y" , "yes" ]:
242
268
# Get existing records of the given subdomain
243
- existing_records = getCloudflareExistingRecords (email , api_key , zone_id , subdomain )
269
+ existing_records = getCloudflareExistingRecords (
270
+ email , api_key , zone_id , subdomain )
244
271
print ("Deleting existing records..." , end = '' , flush = True )
245
- #Delete all existing records of the given subdomain
272
+ # Delete all existing records of the given subdomain
246
273
for record in existing_records :
247
- deleteCloudflareExistingRecord (email , api_key , zone_id , record ["id" ])
274
+ deleteCloudflareExistingRecord (
275
+ email , api_key , zone_id , record ["id" ])
248
276
print ("Done." )
249
277
250
278
print ("Adding new A Record(s) for selected IP(s):" )
251
279
for el in selectd_ip_list :
252
280
print (el .ip , end = '' , flush = True )
253
- addNewCloudflareRecord (email , api_key , zone_id , subdomain , el .ip )
281
+ addNewCloudflareRecord (
282
+ email , api_key , zone_id , subdomain , el .ip )
254
283
print (" Done." )
255
284
print ("All records have been added to your subdomain." )
256
285
except Exception as e :
@@ -279,17 +308,22 @@ def startTest(stdscr: curses.window, ip_list: Pattern[AnyStr], config: configpar
279
308
280
309
# Creating `selected-ips.csv` file to output results
281
310
with open ('selected-ips.csv' , 'w' ) as csv_file :
282
- csv_file .write ("#,IP,Ping (ms),Jitter (ms),Latency (ms),Upload (Mbps),Download (Mbps)\n " )
311
+ csv_file .write (
312
+ "#,IP,Ping (ms),Jitter (ms),Latency (ms),Upload (Mbps),Download (Mbps)\n " )
283
313
284
314
# Creating `selected-ips.csv` file to output results
285
315
with open ('selected-ips.txt' , 'w' ) as txt_file :
286
316
txt_file .write ("" )
287
317
288
318
# Print out table header if it was the first record
289
- stdscr .addstr (3 , 0 , "|---|---------------|----------|---------|---------|----------|------------|" )
290
- stdscr .addstr (4 , 0 , "| # | IP | Ping(ms) | Jit(ms) | Lat(ms) | Up(Mbps) | Down(Mbps) |" )
291
- stdscr .addstr (5 , 0 , "|---|---------------|----------|---------|---------|----------|------------|" )
292
- stdscr .addstr (6 , 0 , "|---|---------------|----------|---------|---------|----------|------------|" )
319
+ stdscr .addstr (
320
+ 3 , 0 , "|---|---------------|----------|---------|---------|----------|------------|" )
321
+ stdscr .addstr (
322
+ 4 , 0 , "| # | IP | Ping(ms) | Jit(ms) | Lat(ms) | Up(Mbps) | Down(Mbps) |" )
323
+ stdscr .addstr (
324
+ 5 , 0 , "|---|---------------|----------|---------|---------|----------|------------|" )
325
+ stdscr .addstr (
326
+ 6 , 0 , "|---|---------------|----------|---------|---------|----------|------------|" )
293
327
294
328
# Loop through IP adresses to check their ping, latency and download/upload speed
295
329
for ip in ip_list :
@@ -344,7 +378,8 @@ def startTest(stdscr: curses.window, ip_list: Pattern[AnyStr], config: configpar
344
378
stdscr .refresh ()
345
379
346
380
# Calculate download speed of selected ip using related function
347
- download_speed = getDownloadSpeed (ip , test_size , min_download_speed )
381
+ download_speed = getDownloadSpeed (
382
+ ip , test_size , min_download_speed )
348
383
# Ignore the IP if download speed dosn't match the minimum required speed
349
384
350
385
stdscr .move (1 , 0 )
@@ -362,21 +397,24 @@ def startTest(stdscr: curses.window, ip_list: Pattern[AnyStr], config: configpar
362
397
# Insert a new line at the cursor position, shifting the existing lines down
363
398
stdscr .insertln ()
364
399
# Print out the IP and related info as well as ping, latency and download/upload speed
365
- stdscr .addstr (f"|{ successful_no :3d} |{ ip :15s} |{ ping :7d} |{ jitter :6d} |{ latency :6d} |{ upload_speed :7.2f} |{ download_speed :9.2f} |" )
400
+ stdscr .addstr (
401
+ f"|{ successful_no :3d} |{ ip :15s} |{ ping :7d} |{ jitter :6d} |{ latency :6d} |{ upload_speed :7.2f} |{ download_speed :9.2f} |" )
366
402
stdscr .refresh ()
367
403
368
- selectd_ip_list .append (IPInfo (ip , ping , jitter , latency , upload_speed , download_speed ))
404
+ selectd_ip_list .append (
405
+ IPInfo (ip , ping , jitter , latency , upload_speed , download_speed ))
369
406
370
407
with open ('selected-ips.csv' , 'a' ) as csv_file :
371
- csv_file .write (f"{ successful_no } ,{ ip } ,{ ping } ,{ jitter } ,{ latency } ,{ upload_speed } ,{ download_speed } \n " )
408
+ csv_file .write (
409
+ f"{ successful_no } ,{ ip } ,{ ping } ,{ jitter } ,{ latency } ,{ upload_speed } ,{ download_speed } \n " )
372
410
with open ('selected-ips.txt' , 'a' ) as txt_file :
373
411
txt_file .write (f"{ ip } \n " )
374
412
375
413
except KeyboardInterrupt :
376
414
print ("\n \n Request cancelled by user!" )
377
415
sys .exit (0 )
378
416
except requests .exceptions .RequestException as e :
379
- print ("\r " , end = '' , flush = True ) # Nothing to do
417
+ print ("\r " , end = '' , flush = True ) # Nothing to do
380
418
381
419
# Exit the loop if we found required number of clean IP addresses
382
420
if len (selectd_ip_list ) >= max_ip :
@@ -443,7 +481,8 @@ def getPing(ip, acceptable_ping):
443
481
# Calculate spent time for fallback
444
482
duration = int ((time .time () - start_time ) * 1000 )
445
483
# Calculate the ping in milliseconds
446
- ping = int (response_time * 1000 ) if response_time is not None and response_time > 0 else duration
484
+ ping = int (
485
+ response_time * 1000 ) if response_time is not None and response_time > 0 else duration
447
486
except Exception as e :
448
487
ping = - 1
449
488
@@ -474,7 +513,8 @@ def getLatencyAndJitter(ip, acceptable_latency):
474
513
headers = {'Host' : 'speed.cloudflare.com' }
475
514
# Set the parameters for the download request
476
515
if openssl_is_active :
477
- params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " , 'alpn' : 'h2,http/1.1' , 'utls' : 'random' }
516
+ params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " ,
517
+ 'alpn' : 'h2,http/1.1' , 'utls' : 'random' }
478
518
else :
479
519
params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " }
480
520
@@ -486,7 +526,8 @@ def getLatencyAndJitter(ip, acceptable_latency):
486
526
# Start the timer for the download request
487
527
start_time = time .time ()
488
528
# Send the download request and get the response
489
- response = requests .get (url , headers = headers , params = params , timeout = timeout )
529
+ response = requests .get (
530
+ url , headers = headers , params = params , timeout = timeout )
490
531
# Calculate the latency in milliseconds
491
532
current_latency = int ((time .time () - start_time ) * 1000 )
492
533
latency = latency + current_latency
@@ -504,7 +545,6 @@ def getLatencyAndJitter(ip, acceptable_latency):
504
545
latency = 99999
505
546
jitter = - 1
506
547
507
-
508
548
# Return latency in milliseconds
509
549
return latency , jitter
510
550
@@ -535,15 +575,17 @@ def getDownloadSpeed(ip, size, min_speed):
535
575
headers = {'Host' : 'speed.cloudflare.com' }
536
576
# Set the parameters for the download request
537
577
if openssl_is_active :
538
- params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " , 'alpn' : 'h2,http/1.1' , 'utls' : 'random' }
578
+ params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " ,
579
+ 'alpn' : 'h2,http/1.1' , 'utls' : 'random' }
539
580
else :
540
581
params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " }
541
582
542
583
try :
543
584
# Start the timer for the download request
544
585
start_time = time .time ()
545
586
# Send the download request and get the response
546
- response = requests .get (url , headers = headers , params = params , timeout = timeout )
587
+ response = requests .get (url , headers = headers ,
588
+ params = params , timeout = timeout )
547
589
# Calculate the download time
548
590
download_time = time .time () - start_time
549
591
# Calculate the download speed in Mbps
@@ -578,9 +620,11 @@ def getUploadSpeed(ip, size, min_speed):
578
620
timeout = upload_size / min_speed_bytes
579
621
# Set the URL, headers, and parameters for the request
580
622
url = 'https://speed.cloudflare.com/__up'
581
- headers = {'Content-Type' : 'multipart/form-data' , 'Host' : 'speed.cloudflare.com' }
623
+ headers = {'Content-Type' : 'multipart/form-data' ,
624
+ 'Host' : 'speed.cloudflare.com' }
582
625
if openssl_is_active :
583
- params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " , 'alpn' : 'h2,http/1.1' , 'utls' : 'random' }
626
+ params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " ,
627
+ 'alpn' : 'h2,http/1.1' , 'utls' : 'random' }
584
628
else :
585
629
params = {'resolve' : f"speed.cloudflare.com:443:{ ip } " }
586
630
@@ -590,7 +634,8 @@ def getUploadSpeed(ip, size, min_speed):
590
634
try :
591
635
# Send the request and measure the upload time
592
636
start_time = time .time ()
593
- response = requests .post (url , headers = headers , params = params , files = files , timeout = timeout )
637
+ response = requests .post (url , headers = headers ,
638
+ params = params , files = files , timeout = timeout )
594
639
upload_time = time .time () - start_time
595
640
# Calculate the upload speed in Mbps
596
641
upload_speed = round (upload_size / upload_time * 8 / 1000000 , 2 )
@@ -728,7 +773,8 @@ def processRegex(cidr: str, include_reg: Pattern[AnyStr], exclude_reg: Pattern[A
728
773
# Check if openssl is installed or not
729
774
def has_openssl ():
730
775
try :
731
- openssl = subprocess .check_call (["openssl" , "version" ], stdout = subprocess .PIPE )
776
+ openssl = subprocess .check_call (
777
+ ["openssl" , "version" ], stdout = subprocess .PIPE )
732
778
return True
733
779
except :
734
780
return False
@@ -755,7 +801,6 @@ def getCIDRv4Ranges():
755
801
return getCIDRv4Ranges ()
756
802
757
803
758
-
759
804
# Call the main function
760
805
if __name__ == '__main__' :
761
806
main ()
0 commit comments