代码拉取完成,页面将自动刷新
#!/usr/bin/env python
"""
// -------------------------------------------------------------
// author Giga
// project qeeqbox/social-analyzer
// email gigaqeeq@gmail.com
// description app.py (CLI)
// licensee AGPL-3.0
// -------------------------------------------------------------
// contributors list qeeqbox/social-analyzer/graphs/contributors
// -------------------------------------------------------------
"""
from logging import getLogger, DEBUG, Formatter, Handler, addLevelName
from logging.handlers import RotatingFileHandler
from sys import platform
from os import path, system
from time import time, sleep
from argparse import ArgumentParser
from json import load, dumps
from uuid import uuid4
from functools import wraps
from re import sub as resub
from re import findall, IGNORECASE
from re import compile as recompile
from re import search as research
from contextlib import suppress
from concurrent.futures import ThreadPoolExecutor, as_completed
from random import randint
from tempfile import mkdtemp
from urllib.parse import unquote, urlparse
from urllib3.exceptions import InsecureRequestWarning
from bs4 import BeautifulSoup
from tld import get_fld
from requests import get, packages, Session
from termcolor import colored
from langdetect import detect
if platform == "win32":
system("color")
packages.urllib3.disable_warnings(category=InsecureRequestWarning)
WEBSITES_ENTRIES = []
SHARED_DETECTIONS = []
GENERIC_DETECTION = []
LOG = getLogger("social-analyzer")
SITES_PATH = path.join(path.dirname(__file__), "data", "sites.json")
LANGUAGES_PATH = path.join(path.dirname(__file__), "data", "languages.json")
STRINGS_PAGES = recompile('captcha-info|Please enable cookies|Completing the CAPTCHA', IGNORECASE)
STRINGS_TITLES = recompile('not found|blocked|attention required|cloudflare', IGNORECASE)
STRINGS_META = recompile(r'regionsAllowed|width|height|color|rgba\(|charset|viewport|refresh|equiv', IGNORECASE)
LANGUAGES_JSON = {}
WORKERS = 15
CUSTOM_MESSAGE = 51
WAF = True
with open(LANGUAGES_PATH) as f:
LANGUAGES_JSON = load(f)
def delete_keys(in_object, keys):
'''
delete specific keys from object
'''
for key in keys:
with suppress(Exception):
del in_object[key]
return in_object
def clean_up_item(in_object, keys_str):
'''
delete specific keys from object (user input)
'''
with suppress(Exception):
del in_object["image"]
if keys_str == "" or keys_str is None:
with suppress(Exception):
pass
else:
for key in in_object.copy():
if key not in keys_str:
with suppress(Exception):
del in_object[key]
return in_object
def get_language_by_guessing(text):
'''
guess language by text, this needs long text
'''
with suppress(Exception):
lang = detect(text)
if lang and lang != "":
return LANGUAGES_JSON[lang] + " (Maybe)"
return "unavailable"
def get_language_by_parsing(source):
'''
guess language by parsing the lang tag
'''
with suppress(Exception):
lang = BeautifulSoup(source, "html.parser").find("html", attrs={"lang": True})["lang"]
if lang and lang != "":
return LANGUAGES_JSON[lang]
return "unavailable"
def check_errors(on_off=None):
'''
wrapper function for debugging
'''
def decorator(func):
@wraps(func)
def wrapper(*args, **kwargs):
if on_off:
try:
return func(*args, **kwargs)
except Exception as err:
pass
# print(e)
else:
return func(*args, **kwargs)
return wrapper
return decorator
class CustomHandler(Handler):
'''
custom stream handler
'''
def __init__(self, argv=None):
'''
int, user choices needed
'''
Handler.__init__(self)
self.argv = argv
def emit(self, record):
'''
emit, based on user choices
'''
if self.argv.output != "json":
if record.levelname == "CUSTOM":
for item in record.msg:
with suppress(Exception):
if item == record.msg[0]:
print("-----------------------")
for key, value in item.items():
if key == "metadata" or key == "extracted":
if (self.argv.metadata and key == "metadata") or (self.argv.extract and key == "extracted"):
with suppress(Exception):
for idx, _item in enumerate(value):
empty_string = key + " " + str(idx)
empty_string = colored(empty_string.ljust(13, ' '), 'blue') + ": "
for _item_key, _item_value in _item.items():
if self.argv.trim and _item_key == "content" and len(_item_value) > 50:
empty_string += "{} : {} ".format(colored(_item_key, 'blue'), colored(_item_value[:50] + "..", 'yellow'))
else:
empty_string += "{} : {} ".format(colored(_item_key, 'blue'), colored(_item_value, 'yellow'))
print("{}".format(empty_string))
else:
print(colored(key.ljust(13, ' '), 'blue'), colored(value, 'yellow'), sep=": ")
print("-----------------------")
else:
print(record.msg)
@check_errors(True)
def setup_logger(uuid=None, file=False, argv=None):
'''
setup a logger for logs in the temp folder
'''
temp_folder = mkdtemp()
if argv.output != "json":
print('[!] Temporary Logs Directory {}'.format(temp_folder))
LOG.setLevel(DEBUG)
LOG.addHandler(CustomHandler(argv))
addLevelName(CUSTOM_MESSAGE, "CUSTOM")
if file and uuid:
fh = RotatingFileHandler(path.join(temp_folder, uuid))
fh.setFormatter(Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s"))
LOG.addHandler(fh)
@check_errors(True)
def init_detections(detections):
'''
load websites_entries, shared_detections and generic_detection
'''
temp_list = []
with open(SITES_PATH, encoding='utf-8') as file:
for item in load(file)[detections]:
item["selected"] = "false"
temp_list.append(item)
return temp_list
def get_website(site):
'''
extract domain from website
'''
temp_value = get_fld(site, fix_protocol=True)
temp_value = temp_value.replace(".{username}", "").replace("{username}.", "")
return temp_value
def list_all_websites():
'''
list all the available websites' entries
'''
if len(WEBSITES_ENTRIES) > 0:
for site in WEBSITES_ENTRIES:
temp_value = get_fld(site["url"], fix_protocol=True)
temp_value = temp_value.replace(".{username}", "").replace("{username}.", "")
LOG.info(temp_value)
@check_errors(True)
def find_username_normal(req):
'''
main find usernames logic using ThreadPoolExecutor
'''
resutls = []
def fetch_url(site, username, options):
'''
this runs for every website entry
'''
sleep(randint(1, 99) / 100)
LOG.info("[Checking] " + get_fld(site["url"]))
source = ""
detection_level = {
"extreme": {
"fast": "normal",
"slow": "normal,advanced,ocr",
"detections": "true",
"count": 1,
"found": 2
},
"high": {
"fast": "normal",
"slow": "normal,advanced,ocr",
"detections": "true,false",
"count": 2,
"found": 1
},
"current": "high"
}
headers = {
"User-Agent": "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:86.0) Gecko/20100101 Firefox/86.0",
}
with suppress(Exception):
session = Session()
session.headers.update(headers)
response = session.get(site["url"].replace("{username}", username), timeout=5, verify=False)
source = response.text
content = response.content
answer = dict((k.lower(), v.lower()) for k, v in response.headers.items())
session.close()
temp_profile = {}
temp_detected = {}
detections_count = 0
def check_url(url):
'''
check if url is okay
'''
with suppress(Exception):
result = urlparse(url)
if result.scheme == "http" or result.scheme == "https":
return all([result.scheme, result.netloc])
return False
def merge_dicts(temp_dict):
'''
'''
result = {}
for item in temp_dict:
for key, value in item.items():
if key in result:
result[key] += value
else:
result[key] = value
return result
def detect_logic(detections):
'''
check for detections in website entry
'''
detections_count = 0
temp_detected = []
temp_found = "false"
temp_profile = {
"found": 0,
"image": "",
"link": "",
"rate": "",
"status": "",
"title": "unavailable",
"language": "unavailable",
"text": "unavailable",
"type": "unavailable",
"extracted": "unavailable",
"metadata": "unavailable",
"good": "",
"method": ""
}
for detection in detections:
temp_found = "false"
if detection["type"] in detection_level[detection_level["current"]]["fast"] and source != "":
detections_count += 1
if detection["string"].replace("{username}", username).lower() in source.lower():
temp_found = "true"
if detection["return"] == temp_found:
temp_profile["found"] += 1
return temp_profile, temp_detected, detections_count
def detect():
'''
main detect logic
'''
temp_profile_all = []
temp_detected_all = []
detections_count_all = 0
for detection in site["detections"]:
detections_ = []
if detection["type"] == "shared":
detections_ = next(item for item in SHARED_DETECTIONS if item["name"] == detection['name'])
if len(detections_) > 0:
val1, val2, val3 = detect_logic(detections_["detections"])
temp_profile_all.append(val1)
detections_count_all += val3
val1, val2, val3 = detect_logic(site["detections"])
temp_profile_all.append(val1)
detections_count_all += val3
return merge_dicts(temp_profile_all), temp_detected_all, detections_count_all
temp_profile, temp_detected, detections_count = detect()
if temp_profile["found"] >= detection_level[detection_level["current"]]["found"] and detections_count >= detection_level[detection_level["current"]]["count"]:
temp_profile["good"] = "true"
soup = None
with suppress(Exception):
soup = BeautifulSoup(content, "html.parser")
with suppress(Exception):
temp_text_arr = []
temp_text_list = []
soup = BeautifulSoup(content, "html.parser")
for item in soup.stripped_strings:
if item not in temp_text_list:
temp_text_list.append(item)
temp_text_arr.append(repr(item).replace("'", ""))
temp_profile["text"] = " ".join(temp_text_arr)
temp_profile["text"] = resub(r"\s\s+", " ", temp_profile["text"])
with suppress(Exception):
temp_profile["language"] = get_language_by_parsing(source)
if temp_profile["language"] == "unavailable":
temp_profile["language"] = get_language_by_guessing(temp_profile["text"])
with suppress(Exception):
temp_profile["title"] = BeautifulSoup(source, "html.parser").title.string
temp_profile["title"] = resub(r"\s\s+", " ", temp_profile["title"])
with suppress(Exception):
temp_matches = []
temp_matches_list = []
if "extract" in site:
for item in site["extract"]:
matches = findall(item["regex"], source)
for match in matches:
if item["type"] == "link":
if check_url(unquote(match)):
parsed = "{}:({})".format(item["type"], unquote(match))
if parsed not in temp_matches:
temp_matches.append(parsed)
temp_matches_list.append({"name": item["type"], "value": unquote(match)})
if len(temp_matches_list) > 0:
temp_profile["extracted"] = temp_matches_list
else:
del temp_profile["extracted"]
temp_profile["text"] = temp_profile["text"].replace("\n", "").replace("\t", "").replace("\r", "").strip()
temp_profile["title"] = temp_profile["title"].replace("\n", "").replace("\t", "").replace("\r", "").strip()
if temp_profile["text"] == "":
temp_profile["text"] = "unavailable"
if temp_profile["title"] == "":
temp_profile["title"] = "unavailable"
if WAF:
with suppress(Exception):
if 'cf-ray' in answer:
temp_profile["text"] = "filtered"
temp_profile["title"] = "filtered"
elif "server" in answer:
if "cloudflare" in answer["server"]:
temp_profile["text"] = "filtered"
temp_profile["title"] = "filtered"
if research(STRINGS_PAGES, temp_profile["text"]):
temp_profile["text"] = "filtered"
temp_profile["title"] = "filtered"
if research(STRINGS_TITLES, temp_profile["title"]):
temp_profile["text"] = "filtered"
temp_profile["title"] = "filtered"
with suppress(Exception):
if detections_count != 0:
temp_value = round(((temp_profile["found"] / detections_count) * 100), 2)
temp_profile["rate"] = "%" + str(temp_value)
if temp_value >= 100.00:
temp_profile["status"] = "good"
elif temp_value >= 50.00 and temp_value < 100.00:
temp_profile["status"] = "maybe"
else:
temp_profile["status"] = "bad"
# copied from qeeqbox osint (pypi) project (currently in-progress)
with suppress(Exception):
if temp_profile["status"] == "good":
temp_meta_list = []
temp_for_checking = []
soup = BeautifulSoup(content, "lxml")
for meta in soup.find_all('meta'):
if meta not in temp_for_checking and not research(STRINGS_META, str(meta)):
temp_for_checking.append(meta)
temp_mata_item = {}
add = True
if meta.has_attr("property"):
temp_mata_item.update({"property": meta["property"]})
if meta.has_attr("content"):
temp_mata_item.update({"content": meta["content"].replace("\n", "").replace("\t", "").replace("\r", "").strip()})
if meta.has_attr("itemprop"):
temp_mata_item.update({"itemprop": meta["itemprop"]})
if meta.has_attr("name"):
temp_mata_item.update({"name": meta["name"]})
with suppress(Exception):
if "property" in temp_mata_item:
for i, item in enumerate(temp_meta_list.copy()):
if "property" in item:
if temp_mata_item["property"] == item["property"]:
temp_meta_list[i]["content"] += ", " + temp_mata_item["content"]
add = False
elif "name" in temp_mata_item:
for i, item in enumerate(temp_meta_list.copy()):
if "name" in item:
if temp_mata_item["name"] == item["name"]:
temp_meta_list[i]["content"] += ", " + temp_mata_item["content"]
add = False
elif "itemprop" in temp_mata_item:
for i, item in enumerate(temp_meta_list.copy()):
if "itemprop" in item:
if temp_mata_item["itemprop"] == item["itemprop"]:
temp_meta_list[i]["content"] += ", " + temp_mata_item["content"]
add = False
if len(temp_mata_item) > 0 and add:
temp_meta_list.append(temp_mata_item)
if len(temp_meta_list) > 0:
temp_profile["metadata"] = temp_meta_list
temp_profile["link"] = site["url"].replace("{username}", req["body"]["string"])
temp_profile["type"] = site["type"]
if "FindUserProfilesFast" in options and "GetUserProfilesFast" not in options:
temp_profile["method"] = "find"
elif "GetUserProfilesFast" in options and "FindUserProfilesFast" not in options:
temp_profile["method"] = "get"
elif "FindUserProfilesFast" in options and "GetUserProfilesFast" in options:
temp_profile["method"] = "all"
copy_temp_profile = temp_profile.copy()
return 1, site["url"], copy_temp_profile
return None, site["url"], []
for i in range(3):
WEBSITES_ENTRIES[:] = [d for d in WEBSITES_ENTRIES if d.get('selected') == "true"]
if len(WEBSITES_ENTRIES) > 0:
with ThreadPoolExecutor(max_workers=WORKERS) as executor:
future_fetch_url = (executor.submit(fetch_url, site, req["body"]["string"], req["body"]["options"]) for site in WEBSITES_ENTRIES)
for future in as_completed(future_fetch_url):
with suppress(Exception):
good, site, data = future.result()
if good:
WEBSITES_ENTRIES[:] = [d for d in WEBSITES_ENTRIES if d.get('url') != site]
resutls.append(data)
else:
LOG.info("[Waiting to retry] " + get_website(site))
WEBSITES_ENTRIES[:] = [d for d in WEBSITES_ENTRIES if d.get('selected') == "true"]
if len(WEBSITES_ENTRIES) > 0:
for site in WEBSITES_ENTRIES:
temp_profile = {"link": "",
"method": "failed"}
temp_profile["link"] = site["url"].replace("{username}", req["body"]["string"])
resutls.append(temp_profile)
return resutls
@check_errors(True)
def check_user_cli(argv):
'''
main cli logic
'''
temp_detected = {"detected": [], "unknown": [], "failed": []}
temp_options = "GetUserProfilesFast,FindUserProfilesFast"
if argv.method != "":
if argv.method == "find":
temp_options = "FindUserProfilesFast"
if argv.method == "get":
temp_options = "GetUserProfilesFast"
req = {"body": {"uuid": str(uuid4()), "string": argv.username, "options": temp_options}}
setup_logger(uuid=req["body"]["uuid"], file=True, argv=argv)
if argv.websites == "all":
for site in WEBSITES_ENTRIES:
site["selected"] = "true"
else:
for site in WEBSITES_ENTRIES:
for temp in argv.websites.split(" "):
if temp in site["url"]:
site["selected"] = "true"
resutls = find_username_normal(req)
for item in resutls:
if item is not None:
if item["method"] == "all":
if item["good"] == "true":
item = delete_keys(item, ["method", "good"])
item = clean_up_item(item, argv.options)
temp_detected["detected"].append(item)
else:
item = delete_keys(item, ["found", "rate", "status", "method", "good", "text", "extracted", "metadata"])
item = clean_up_item(item, argv.options)
temp_detected["unknown"].append(item)
elif item["method"] == "find":
if item["good"] == "true":
item = delete_keys(item, ["method", "good"])
item = clean_up_item(item, argv.options)
temp_detected["detected"].append(item)
elif item["method"] == "get":
item = delete_keys(item, ["found", "rate", "status", "method", "good", "text", "extracted", "metadata"])
item = clean_up_item(item, argv.options)
temp_detected["unknown"].append(item)
else:
item = delete_keys(item, ["found", "rate", "status", "method", "good", "text", "title", "language", "rate", "extracted", "metadata"])
item = clean_up_item(item, argv.options)
temp_detected["failed"].append(item)
with suppress(Exception):
if len(temp_detected["detected"]) == 0:
del temp_detected["detected"]
else:
if "all" in argv.profiles or "detected" in argv.profiles:
if argv.filter == "all":
pass
else:
temp_detected["detected"] = [item for item in temp_detected["detected"] if item['status'] in argv.filter]
if len(temp_detected["detected"]) > 0:
temp_detected["detected"] = sorted(temp_detected["detected"], key=lambda k: float(k['rate'].strip('%')), reverse=True)
else:
del temp_detected["detected"]
else:
del temp_detected["detected"]
if len(temp_detected["unknown"]) == 0:
del temp_detected["unknown"]
else:
if "all" in argv.profiles or "unknown" in argv.profiles:
pass
else:
del temp_detected["unknown"]
if len(temp_detected["failed"]) == 0:
del temp_detected["failed"]
else:
if "all" in argv.profiles or "failed" in argv.profiles:
pass
else:
del temp_detected["failed"]
if argv.output == "pretty" or argv.output == "":
if 'detected' in temp_detected:
LOG.info("[Detected] {} Profile[s]".format(len(temp_detected['detected'])))
LOG.log(CUSTOM_MESSAGE, temp_detected['detected'])
if 'unknown' in temp_detected:
LOG.info("[unknown] {} Profile[s]".format(len(temp_detected['unknown'])))
LOG.log(CUSTOM_MESSAGE, temp_detected['unknown'])
if 'failed' in temp_detected:
LOG.info("[failed] {} Profile[s]".format(len(temp_detected['failed'])))
LOG.log(CUSTOM_MESSAGE, temp_detected['failed'])
if argv.output == "json":
print(dumps(temp_detected, sort_keys=True, indent=None))
def msg():
'''
welcome message
'''
if path.basename(__file__) == "__main__.py":
return """python -m social-analyzer --cli --mode 'fast' --username 'johndoe' --websites 'youtube pinterest tumblr' --output 'pretty'"""
return """python3 app.py --cli --mode 'fast' --username 'johndoe' --websites 'youtube pinterest tumblr' --output 'pretty'"""
WEBSITES_ENTRIES = init_detections("websites_entries")
SHARED_DETECTIONS = init_detections("shared_detections")
GENERIC_DETECTION = init_detections("generic_detection")
ARG_PARSER = ArgumentParser(description="Qeeqbox/social-analyzer - API and Web App for analyzing & finding a person's profile across 300+ social media websites (Detections are updated regularly)", usage=msg())
ARG_PARSER._action_groups.pop()
ARG_PARSER_REQUIRED = ARG_PARSER.add_argument_group("Required Arguments")
ARG_PARSER_REQUIRED.add_argument("--cli", action="store_true", help="Turn this CLI on", required=True)
ARG_PARSER_REQUIRED.add_argument("--username", help="E.g. johndoe, john_doe or johndoe9999", metavar="", required=True)
ARG_PARSER_OPTIONAL = ARG_PARSER.add_argument_group("Optional Arguments")
ARG_PARSER_OPTIONAL.add_argument("--websites", help="Website or websites separated by space E.g. youtube, tiktok or tumblr", metavar="", default="all")
ARG_PARSER_OPTIONAL.add_argument("--mode", help="Analysis mode E.g.fast -> FindUserProfilesFast, slow -> FindUserProfilesSlow or special -> FindUserProfilesSpecial", metavar="", default="fast")
ARG_PARSER_OPTIONAL.add_argument("--output", help="Show the output in the following format: json -> json output for integration or pretty -> prettify the output", metavar="", default="pretty")
ARG_PARSER_OPTIONAL.add_argument("--options", help="Show the following when a profile is found: link, rate, title or text", metavar="", default="")
ARG_PARSER_OPTIONAL.add_argument("--method", help="find -> show detected profiles, get -> show all profiles regardless detected or not, both -> combine find & get", metavar="", default="all")
ARG_PARSER_OPTIONAL.add_argument("--filter", help="filter detected profiles by good, maybe or bad, you can do combine them with comma (good,bad) or use all", metavar="", default="all")
ARG_PARSER_OPTIONAL.add_argument("--profiles", help="filter profiles by detected, unknown or failed, you can do combine them with comma (detected,failed) or use all", metavar="", default="all")
ARG_PARSER_OPTIONAL.add_argument("--extract", action="store_true", help="Extract profiles, urls & patterns if possible", required=False)
ARG_PARSER_OPTIONAL.add_argument("--metadata", action="store_true", help="Extract metadata if possible (pypi QeeqBox OSINT)", required=False)
ARG_PARSER_OPTIONAL.add_argument("--trim", action="store_true", help="Trim long strings", required=False)
ARG_PARSER_LIST = ARG_PARSER.add_argument_group("Listing websites & detections")
ARG_PARSER_LIST.add_argument("--list", action="store_true", help="List all available websites")
ARGV = ARG_PARSER.parse_args()
if __name__ == "__main__":
if ARGV.output != "json":
print("[!] Detections are updated very often, make sure to get the most up-to-date ones")
if ARGV.cli:
if ARGV.list:
setup_logger()
list_all_websites()
elif ARGV.mode == "fast":
if ARGV.username != "" and ARGV.websites != "":
check_user_cli(ARGV)
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。