@ -13,11 +13,12 @@ import os
import sys
import sys
import platform
import platform
import re
import re
from argparse import ArgumentParser , RawDescriptionHelpFormatter
from time import time
from concurrent . futures import ThreadPoolExecutor
import requests
import requests
from colorama import Back , Fore , Style , init
from argparse import ArgumentParser , RawDescriptionHelpFormatter
from concurrent . futures import ThreadPoolExecutor
from colorama import Fore , Style , init
from requests_futures . sessions import FuturesSession
from requests_futures . sessions import FuturesSession
from torrequest import TorRequest
from torrequest import TorRequest
@ -28,28 +29,75 @@ amount=0
# TODO: fix tumblr
# TODO: fix tumblr
# Extends FutureSession to add response time metric
# This is taken (almost) directly from here: https://github.com/ross/requests-futures#working-in-the-background
class ElapsedFuturesSession ( FuturesSession ) :
def request ( self , method , url , hooks = { } , * args , * * kwargs ) :
start = time ( )
def timing ( r , * args , * * kwargs ) :
elapsed_sec = time ( ) - start
r . elapsed = round ( elapsed_sec * 1000 )
try :
if isinstance ( hooks [ ' response ' ] , ( list , tuple ) ) :
# needs to be first so we don't time other hooks execution
hooks [ ' response ' ] . insert ( 0 , timing )
else :
hooks [ ' response ' ] = [ timing , hooks [ ' response ' ] ]
except KeyError :
hooks [ ' response ' ] = timing
return super ( ElapsedFuturesSession , self ) . request ( method , url , hooks = hooks , * args , * * kwargs )
def open_file ( fname ) :
def open_file ( fname ) :
return open ( fname , " a " )
return open ( fname , " a " )
def write_to_file ( url , f ) :
def write_to_file ( url , f ) :
f . write ( url + " \n " )
f . write ( url + " \n " )
def final_score ( amount , f ) :
def final_score ( amount , f ) :
f . write ( " Total: " + str ( amount ) + " \n " )
f . write ( " Total: " + str ( amount ) + " \n " )
def print_error ( err , errstr , var , debug = False ) :
def print_error ( err , errstr , var , verbose = False ) :
print ( Style . BRIGHT + Fore . WHITE + " [ " +
print ( Style . BRIGHT + Fore . WHITE + " [ " +
Fore . RED + " - " +
Fore . RED + " - " +
Fore . WHITE + " ] " +
Fore . WHITE + " ] " +
Fore . RED + f " { errstr } " +
Fore . RED + f " { errstr } " +
Fore . YELLOW + f " { err if debug else var } " )
Fore . YELLOW + f " { err if verbose else var } " )
def create_response_time ( response_time , verbose ) :
return " [ {} ms] " . format ( response_time ) if verbose else " "
def print_found ( social_network , url , response_time , verbose = False ) :
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
Fore . GREEN + " + " +
Fore . WHITE + " ] " +
create_response_time ( response_time , verbose ) +
Fore . GREEN + " {} : " ) . format ( social_network ) , url )
def print_not_found ( social_network , response_time , verbose = False ) :
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
Fore . RED + " - " +
Fore . WHITE + " ] " +
create_response_time ( response_time , verbose ) +
Fore . GREEN + " {} : " +
Fore . YELLOW + " Not Found! " ) . format ( social_network ) )
def get_response ( request_future , error_type , social_network , verbose = False ) :
def get_response ( request_future , error_type , social_network , verbose = False ) :
try :
try :
rsp = request_future . result ( )
rsp = request_future . result ( )
if rsp . status_code :
if rsp . status_code :
return rsp , error_type
return rsp , error_type , rsp . elapsed
except requests . exceptions . HTTPError as errh :
except requests . exceptions . HTTPError as errh :
print_error ( errh , " HTTP Error: " , social_network , verbose )
print_error ( errh , " HTTP Error: " , social_network , verbose )
except requests . exceptions . ConnectionError as errc :
except requests . exceptions . ConnectionError as errc :
@ -58,7 +106,7 @@ def get_response(request_future, error_type, social_network, verbose=False):
print_error ( errt , " Timeout Error: " , social_network , verbose )
print_error ( errt , " Timeout Error: " , social_network , verbose )
except requests . exceptions . RequestException as err :
except requests . exceptions . RequestException as err :
print_error ( err , " Unknown error: " , social_network , verbose )
print_error ( err , " Unknown error: " , social_network , verbose )
return None , " "
return None , " " , - 1
def sherlock ( username , site_data , verbose = False , tor = False , unique_tor = False ) :
def sherlock ( username , site_data , verbose = False , tor = False , unique_tor = False ) :
@ -119,8 +167,8 @@ def sherlock(username, site_data, verbose=False, tor=False, unique_tor=False):
underlying_request = TorRequest ( )
underlying_request = TorRequest ( )
underlying_session = underlying_request . session ( )
underlying_session = underlying_request . session ( )
# Create multi-threaded session for all requests
# Create multi-threaded session for all requests . Use our custom FuturesSession that exposes response time
session = FuturesSession( executor = executor , session = underlying_session )
session = Elapsed FuturesSession( executor = executor , session = underlying_session )
# Results from analysis of all sites
# Results from analysis of all sites
results_total = { }
results_total = { }
@ -193,7 +241,7 @@ def sherlock(username, site_data, verbose=False, tor=False, unique_tor=False):
# Retrieve future and ensure it has finished
# Retrieve future and ensure it has finished
future = net_info [ " request_future " ]
future = net_info [ " request_future " ]
r , error_type = get_response ( request_future = future ,
r , error_type , response_time = get_response ( request_future = future ,
error_type = error_type ,
error_type = error_type ,
social_network = social_network ,
social_network = social_network ,
verbose = verbose )
verbose = verbose )
@ -212,59 +260,35 @@ def sherlock(username, site_data, verbose=False, tor=False, unique_tor=False):
error = net_info . get ( " errorMsg " )
error = net_info . get ( " errorMsg " )
# Checks if the error message is in the HTML
# Checks if the error message is in the HTML
if not error in r . text :
if not error in r . text :
print_found ( social_network , url , response_time , verbose )
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
Fore . GREEN + " + " +
Fore . WHITE + " ] " +
Fore . GREEN + " {} : " ) . format ( social_network ) , url )
write_to_file ( url , f )
write_to_file ( url , f )
exists = " yes "
exists = " yes "
amount = amount + 1
amount = amount + 1
else :
else :
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
print_not_found ( social_network , response_time , verbose )
Fore . RED + " - " +
Fore . WHITE + " ] " +
Fore . GREEN + " {} : " +
Fore . YELLOW + " Not Found! " ) . format ( social_network ) )
exists = " no "
exists = " no "
elif error_type == " status_code " :
elif error_type == " status_code " :
# Checks if the status code of the response is 2XX
# Checks if the status code of the response is 2XX
if not r . status_code > = 300 or r . status_code < 200 :
if not r . status_code > = 300 or r . status_code < 200 :
print_found ( social_network , url , response_time , verbose )
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
Fore . GREEN + " + " +
Fore . WHITE + " ] " +
Fore . GREEN + " {} : " ) . format ( social_network ) , url )
write_to_file ( url , f )
write_to_file ( url , f )
exists = " yes "
exists = " yes "
amount = amount + 1
amount = amount + 1
else :
else :
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
print_not_found ( social_network , response_time , verbose )
Fore . RED + " - " +
Fore . WHITE + " ] " +
Fore . GREEN + " {} : " +
Fore . YELLOW + " Not Found! " ) . format ( social_network ) )
exists = " no "
exists = " no "
elif error_type == " response_url " :
elif error_type == " response_url " :
error = net_info . get ( " errorUrl " )
error = net_info . get ( " errorUrl " )
# Checks if the redirect url is the same as the one defined in data.json
# Checks if the redirect url is the same as the one defined in data.json
if not error in r . url :
if not error in r . url :
print_found ( social_network , url , response_time , verbose )
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
Fore . GREEN + " + " +
Fore . WHITE + " ] " +
Fore . GREEN + " {} : " ) . format ( social_network ) , url )
write_to_file ( url , f )
write_to_file ( url , f )
exists = " yes "
exists = " yes "
amount = amount + 1
amount = amount + 1
else :
else :
print ( ( Style . BRIGHT + Fore . WHITE + " [ " +
print_not_found ( social_network , response_time , verbose )
Fore . RED + " - " +
Fore . WHITE + " ] " +
Fore . GREEN + " {} : " +
Fore . YELLOW + " Not Found! " ) . format ( social_network ) )
exists = " no "
exists = " no "
elif error_type == " " :
elif error_type == " " :
@ -281,6 +305,7 @@ def sherlock(username, site_data, verbose=False, tor=False, unique_tor=False):
# Save results from request
# Save results from request
results_site [ ' http_status ' ] = http_status
results_site [ ' http_status ' ] = http_status
results_site [ ' response_text ' ] = response_text
results_site [ ' response_text ' ] = response_text
results_site [ ' response_time_ms ' ] = response_time
# Add this site's results into final dictionary with all of the other results.
# Add this site's results into final dictionary with all of the other results.
results_total [ social_network ] = results_site
results_total [ social_network ] = results_site
@ -393,7 +418,8 @@ def main():
' url_main ' ,
' url_main ' ,
' url_user ' ,
' url_user ' ,
' exists ' ,
' exists ' ,
' http_status '
' http_status ' ,
' response_time_ms '
]
]
)
)
for site in results :
for site in results :
@ -402,9 +428,11 @@ def main():
results [ site ] [ ' url_main ' ] ,
results [ site ] [ ' url_main ' ] ,
results [ site ] [ ' url_user ' ] ,
results [ site ] [ ' url_user ' ] ,
results [ site ] [ ' exists ' ] ,
results [ site ] [ ' exists ' ] ,
results [ site ] [ ' http_status ' ]
results [ site ] [ ' http_status ' ] ,
results [ site ] [ ' response_time_ms ' ]
]
]
)
)
if __name__ == " __main__ " :
if __name__ == " __main__ " :
main ( )
main ( )