2018-03-04 21:22:34 +08:00
|
|
|
import hashlib
|
|
|
|
import binascii
|
2018-05-15 18:49:23 +08:00
|
|
|
import copy
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-05-15 20:26:46 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
from pymongo import MongoClient
|
2018-03-07 15:56:32 +08:00
|
|
|
|
2018-05-15 20:26:46 +08:00
|
|
|
class mongo(object):
|
|
|
|
db = MongoClient().monkeyisland
|
|
|
|
else:
|
2018-04-01 20:24:40 +08:00
|
|
|
from cc.database import mongo
|
|
|
|
|
2018-03-04 21:22:34 +08:00
|
|
|
DsRole_RoleStandaloneWorkstation = 0
|
|
|
|
DsRole_RoleMemberWorkstation = 1
|
|
|
|
DsRole_RoleStandaloneServer = 2
|
|
|
|
DsRole_RoleMemberServer = 3
|
|
|
|
DsRole_RoleBackupDomainController = 4
|
|
|
|
DsRole_RolePrimaryDomainController = 5
|
|
|
|
|
2018-04-30 23:40:11 +08:00
|
|
|
SidTypeUser = 1
|
|
|
|
SidTypeGroup = 2
|
|
|
|
SidTypeDomain = 3
|
|
|
|
SidTypeAlias = 4
|
|
|
|
SidTypeWellKnownGroup = 5
|
|
|
|
SidTypeDeletedAccount = 6
|
|
|
|
SidTypeInvalid = 7
|
|
|
|
SidTypeUnknown = 8
|
|
|
|
SidTypeComputer = 9
|
|
|
|
|
2018-05-15 16:10:32 +08:00
|
|
|
def is_group_sid_type(type):
|
|
|
|
return type in (SidTypeGroup, SidTypeAlias, SidTypeWellKnownGroup)
|
|
|
|
|
2018-03-04 21:22:34 +08:00
|
|
|
def myntlm(x):
|
|
|
|
hash = hashlib.new('md4', x.encode('utf-16le')).digest()
|
|
|
|
return str(binascii.hexlify(hash))
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
def cache(foo):
|
|
|
|
def hash(o):
|
|
|
|
if type(o) in (int, float, str, unicode):
|
2018-05-15 17:55:50 +08:00
|
|
|
return repr(o)
|
2018-04-03 15:47:10 +08:00
|
|
|
|
2018-04-30 23:15:10 +08:00
|
|
|
elif type(o) in (type(None),):
|
|
|
|
return "___None___"
|
|
|
|
|
2018-04-17 19:43:26 +08:00
|
|
|
elif type(o) in (list, tuple, set):
|
2018-04-03 15:47:10 +08:00
|
|
|
hashed = tuple([hash(x) for x in o])
|
|
|
|
|
|
|
|
if "NotHashable" in hashed:
|
|
|
|
return "NotHashable"
|
|
|
|
|
|
|
|
return hashed
|
|
|
|
|
|
|
|
elif type(o) == dict:
|
|
|
|
hashed_keys = tuple([hash(k) for k, v in o.iteritems()])
|
|
|
|
hashed_vals = tuple([hash(v) for k, v in o.iteritems()])
|
|
|
|
|
|
|
|
if "NotHashable" in hashed_keys or "NotHashable" in hashed_vals:
|
|
|
|
return "NotHashable"
|
|
|
|
|
|
|
|
return tuple(zip(hashed_keys, hashed_vals))
|
|
|
|
|
|
|
|
elif type(o) == Machine:
|
|
|
|
return o.monkey_guid
|
|
|
|
|
|
|
|
elif type(o) == PthMap:
|
|
|
|
return "PthMapSingleton"
|
|
|
|
|
2018-05-15 20:26:46 +08:00
|
|
|
elif type(o) == PassTheHashReport:
|
|
|
|
return "PassTheHashReportSingleton"
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
else:
|
2018-04-17 19:43:26 +08:00
|
|
|
assert False, "%s of type %s is not hashable" % (repr(o), type(o))
|
2018-04-03 15:47:10 +08:00
|
|
|
return "NotHashable"
|
|
|
|
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
hashed = (hash(args), hash(kwargs))
|
|
|
|
|
|
|
|
if "NotHashable" in hashed:
|
|
|
|
return foo(*args, **kwargs)
|
|
|
|
|
|
|
|
if not hasattr(foo, "_mycache_"):
|
|
|
|
foo._mycache_ = dict()
|
|
|
|
|
|
|
|
if hashed not in foo._mycache_.keys():
|
|
|
|
foo._mycache_[hashed] = foo(*args, **kwargs)
|
|
|
|
|
2018-05-15 18:49:23 +08:00
|
|
|
return copy.deepcopy(foo._mycache_[hashed])
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
return wrapper
|
|
|
|
|
2018-03-04 21:22:34 +08:00
|
|
|
class Machine(object):
|
|
|
|
def __init__(self, monkey_guid):
|
|
|
|
self.monkey_guid = str(monkey_guid)
|
2018-03-06 21:37:50 +08:00
|
|
|
|
2018-04-17 18:30:55 +08:00
|
|
|
self.latest_system_info = mongo.db.telemetry.find({"telem_type":"system_info_collection", "monkey_guid": self.monkey_guid}).sort([("timestamp", -1)]).limit(1)
|
2018-03-06 21:37:50 +08:00
|
|
|
|
|
|
|
if self.latest_system_info.count() > 0:
|
|
|
|
self.latest_system_info = self.latest_system_info[0]
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-05-15 19:20:09 +08:00
|
|
|
def __eq__(self, other):
|
|
|
|
if isinstance(other, self.__class__):
|
|
|
|
return self.monkey_guid == other.monkey_guid
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetMimikatzOutput(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
if not doc:
|
|
|
|
return None
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
return doc["data"]["mimikatz"]
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetHostName(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for comp in doc["data"]["Win32_ComputerSystem"]:
|
|
|
|
return eval(comp["Name"])
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetIp(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for addr in doc["data"]["network_info"]["networks"]:
|
|
|
|
return str(addr["addr"])
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetDomainName(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for comp in doc["data"]["Win32_ComputerSystem"]:
|
|
|
|
return eval(comp["Domain"])
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return None
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetDomainRole(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for comp in doc["data"]["Win32_ComputerSystem"]:
|
|
|
|
return comp["DomainRole"]
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return None
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def IsDomainController(self):
|
|
|
|
return self.GetDomainRole() in (DsRole_RolePrimaryDomainController, DsRole_RoleBackupDomainController)
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-05-08 22:25:46 +08:00
|
|
|
def GetSidByUsername(self, username, domain=None):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for user in doc["data"]["Win32_UserAccount"]:
|
|
|
|
if eval(user["Name"]) != username:
|
|
|
|
continue
|
2018-04-30 23:40:11 +08:00
|
|
|
|
2018-04-30 23:42:30 +08:00
|
|
|
if user["SIDType"] != SidTypeUser:
|
2018-04-30 23:40:11 +08:00
|
|
|
continue
|
2018-05-08 22:25:46 +08:00
|
|
|
|
|
|
|
if domain and user["Domain"] != domain:
|
|
|
|
continue
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
return eval(user["SID"])
|
2018-03-11 15:58:51 +08:00
|
|
|
|
|
|
|
if not self.IsDomainController():
|
|
|
|
for dc in self.GetDomainControllers():
|
|
|
|
sid = dc.GetSidByUsername(username)
|
|
|
|
|
|
|
|
if sid != None:
|
|
|
|
return sid
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetUsernameBySid(self, sid):
|
2018-04-10 22:51:09 +08:00
|
|
|
info = self.GetSidInfo(sid)
|
|
|
|
|
|
|
|
if not info:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return info["Domain"] + "\\" + info["Username"]
|
|
|
|
|
|
|
|
@cache
|
|
|
|
def GetSidInfo(self, sid):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for user in doc["data"]["Win32_UserAccount"]:
|
|
|
|
if eval(user["SID"]) != sid:
|
|
|
|
continue
|
2018-04-30 23:40:11 +08:00
|
|
|
|
2018-04-30 23:42:30 +08:00
|
|
|
if user["SIDType"] != SidTypeUser:
|
2018-04-30 23:40:11 +08:00
|
|
|
continue
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-10 22:51:09 +08:00
|
|
|
return { "Domain": eval(user["Domain"]),
|
|
|
|
"Username": eval(user["Name"]),
|
|
|
|
"Disabled": user["Disabled"] == "true",
|
|
|
|
"PasswordRequired": user["PasswordRequired"] == "true",
|
|
|
|
"PasswordExpires": user["PasswordExpires"] == "true", }
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 16:32:57 +08:00
|
|
|
if not self.IsDomainController():
|
|
|
|
for dc in self.GetDomainControllers():
|
2018-04-10 22:51:09 +08:00
|
|
|
domain = dc.GetSidInfo(sid)
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-04-10 22:51:09 +08:00
|
|
|
if domain != None:
|
|
|
|
return domain
|
2018-03-06 16:32:57 +08:00
|
|
|
|
|
|
|
return None
|
2018-04-03 15:47:10 +08:00
|
|
|
|
2018-04-10 22:51:09 +08:00
|
|
|
@cache
|
2018-04-17 17:45:32 +08:00
|
|
|
def GetCriticalServicesInstalled(self):
|
2018-04-17 17:29:28 +08:00
|
|
|
def IsNameOfCriticalService(name):
|
2018-04-30 21:19:58 +08:00
|
|
|
services = ("W3svc", "MSExchangeServiceHost", "MSSQLServer", "dns")
|
2018-04-17 17:49:14 +08:00
|
|
|
services = map(str.lower, services)
|
2018-04-17 17:45:32 +08:00
|
|
|
|
|
|
|
if not name:
|
|
|
|
return False
|
|
|
|
|
2018-04-17 17:29:28 +08:00
|
|
|
name = name.lower()
|
|
|
|
|
2018-04-30 21:44:43 +08:00
|
|
|
return name in services
|
|
|
|
#for ser in services:
|
|
|
|
# if ser in name:
|
|
|
|
# return True
|
2018-04-17 17:29:28 +08:00
|
|
|
|
|
|
|
return False
|
2018-04-10 22:51:09 +08:00
|
|
|
|
2018-04-17 17:29:28 +08:00
|
|
|
doc = self.latest_system_info
|
|
|
|
found = []
|
2018-04-17 17:45:32 +08:00
|
|
|
|
|
|
|
if self.IsDomainController():
|
|
|
|
found.append("Domain Controller")
|
2018-04-17 17:29:28 +08:00
|
|
|
|
|
|
|
for product in doc["data"]["Win32_Product"]:
|
|
|
|
service_name = eval(product["Name"])
|
|
|
|
|
|
|
|
if not IsNameOfCriticalService(service_name):
|
|
|
|
continue
|
|
|
|
|
|
|
|
found.append(service_name)
|
|
|
|
|
|
|
|
for service in doc["data"]["Win32_Service"]:
|
|
|
|
service_name = eval(service["Name"])
|
|
|
|
|
|
|
|
if not IsNameOfCriticalService(service_name):
|
|
|
|
continue
|
2018-04-30 21:44:43 +08:00
|
|
|
|
|
|
|
if eval(service["State"]) != "Running":
|
|
|
|
continue
|
2018-04-17 17:29:28 +08:00
|
|
|
|
|
|
|
found.append(service_name)
|
|
|
|
|
|
|
|
return found
|
|
|
|
|
2018-04-17 17:47:52 +08:00
|
|
|
@cache
|
|
|
|
def IsCriticalServer(self):
|
|
|
|
return len(self.GetCriticalServicesInstalled()) > 0
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 23:45:48 +08:00
|
|
|
def GetUsernamesBySecret(self, secret):
|
2018-03-06 20:52:39 +08:00
|
|
|
sam = self.GetLocalSecrets()
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
names = set()
|
|
|
|
|
|
|
|
for username, user_secret in sam.iteritems():
|
2018-03-06 16:32:57 +08:00
|
|
|
if secret == user_secret:
|
2018-03-07 23:45:48 +08:00
|
|
|
names.add(username)
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
return names
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 23:45:48 +08:00
|
|
|
def GetSidsBySecret(self, secret):
|
|
|
|
usernames = self.GetUsernamesBySecret(secret)
|
|
|
|
return set(map(self.GetSidByUsername, usernames))
|
2018-03-06 21:37:50 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetGroupSidByGroupName(self, group_name):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for group in doc["data"]["Win32_Group"]:
|
|
|
|
if eval(group["Name"]) != group_name:
|
|
|
|
continue
|
2018-05-15 16:10:32 +08:00
|
|
|
|
|
|
|
if not is_group_sid_type(group["SIDType"]):
|
2018-04-30 23:40:11 +08:00
|
|
|
continue
|
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
return eval(group["SID"])
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetUsersByGroupSid(self, sid):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
users = dict()
|
2018-04-17 18:30:55 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for group_user in doc["data"]["Win32_GroupUser"]:
|
|
|
|
if eval(group_user["GroupComponent"]["SID"]) != sid:
|
|
|
|
continue
|
2018-04-30 23:40:11 +08:00
|
|
|
|
2018-05-15 16:10:32 +08:00
|
|
|
if not is_group_sid_type(group_user["GroupComponent"]["SIDType"]):
|
2018-04-30 23:40:11 +08:00
|
|
|
continue
|
2018-03-06 21:37:50 +08:00
|
|
|
|
|
|
|
if "PartComponent" not in group_user.keys():
|
|
|
|
continue
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-05-08 22:25:46 +08:00
|
|
|
if type(group_user["PartComponent"]) in (str, unicode):
|
|
|
|
# PartComponent is an id to Win32_UserAccount table
|
|
|
|
|
|
|
|
wmi_id = group_user["PartComponent"]
|
|
|
|
|
|
|
|
if "cimv2:Win32_UserAccount" not in wmi_id:
|
|
|
|
continue
|
|
|
|
|
|
|
|
# u'\\\\WIN-BFA01FFQFLS\\root\\cimv2:Win32_UserAccount.Domain="MYDOMAIN",Name="WIN-BFA01FFQFLS$"'
|
|
|
|
username = wmi_id.split('cimv2:Win32_UserAccount.Domain="')[1].split('",Name="')[0]
|
|
|
|
domain = wmi_id.split('cimv2:Win32_UserAccount.Domain="')[1].split('",Name="')[1][:-1]
|
|
|
|
|
|
|
|
sid = self.GetSidByUsername(username, domain)
|
|
|
|
users[sid] = username
|
|
|
|
|
|
|
|
else:
|
|
|
|
if group_user["PartComponent"]["SIDType"] != SidTypeUser:
|
|
|
|
continue
|
2018-04-30 23:40:11 +08:00
|
|
|
|
2018-05-08 22:25:46 +08:00
|
|
|
users[eval(group_user["PartComponent"]["SID"])] = eval(group_user["PartComponent"]["Name"])
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return users
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetDomainControllersMonkeyGuidByDomainName(self, domain_name):
|
|
|
|
cur = mongo.db.telemetry.find({"telem_type":"system_info_collection", "data.Win32_ComputerSystem.Domain":"u'%s'" % (domain_name,)})
|
|
|
|
|
|
|
|
GUIDs = set()
|
|
|
|
|
|
|
|
for doc in cur:
|
2018-03-06 16:32:57 +08:00
|
|
|
if not Machine(doc["monkey_guid"]).IsDomainController():
|
|
|
|
continue
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 16:32:57 +08:00
|
|
|
GUIDs.add(doc["monkey_guid"])
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return GUIDs
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetLocalAdmins(self):
|
2018-04-17 18:12:57 +08:00
|
|
|
admins = self.GetUsersByGroupSid(self.GetGroupSidByGroupName("Administrators"))
|
|
|
|
|
|
|
|
#debug = self.GetUsersByGroupSid(self.GetGroupSidByGroupName("Users"))
|
|
|
|
#admins.update(debug)
|
|
|
|
|
|
|
|
return admins
|
|
|
|
|
|
|
|
@cache
|
|
|
|
def GetLocalAdminSids(self):
|
|
|
|
return set(self.GetLocalAdmins().keys())
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 03:53:22 +08:00
|
|
|
def GetLocalSids(self):
|
|
|
|
doc = self.latest_system_info
|
|
|
|
|
|
|
|
SIDs = set()
|
|
|
|
|
|
|
|
for user in doc["data"]["Win32_UserAccount"]:
|
2018-04-30 23:42:30 +08:00
|
|
|
if user["SIDType"] != SidTypeUser:
|
2018-04-30 23:40:11 +08:00
|
|
|
continue
|
|
|
|
|
2018-03-07 03:53:22 +08:00
|
|
|
SIDs.add(eval(user["SID"]))
|
|
|
|
|
|
|
|
return SIDs
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetLocalAdminNames(self):
|
2018-04-17 18:12:57 +08:00
|
|
|
return set(self.GetLocalAdmins().values())
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def GetSam(self):
|
2018-03-06 20:52:39 +08:00
|
|
|
if not self.GetMimikatzOutput():
|
|
|
|
return {}
|
|
|
|
|
|
|
|
mimikatz = self.GetMimikatzOutput()
|
|
|
|
|
|
|
|
if mimikatz.count("\n42.") != 2:
|
|
|
|
return {}
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
try:
|
|
|
|
sam_users = mimikatz.split("\n42.")[1].split("\nSAMKey :")[1].split("\n\n")[1:]
|
2018-03-06 20:52:39 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
sam = {}
|
|
|
|
|
|
|
|
for sam_user_txt in sam_users:
|
|
|
|
sam_user = dict([map(unicode.strip, line.split(":")) for line in filter(lambda l: l.count(":") == 1, sam_user_txt.splitlines())])
|
|
|
|
|
|
|
|
ntlm = sam_user["NTLM"]
|
|
|
|
if "[hashed secret]" not in ntlm:
|
|
|
|
continue
|
|
|
|
|
|
|
|
sam[sam_user["User"]] = ntlm.replace("[hashed secret]", "").strip()
|
|
|
|
|
|
|
|
return sam
|
|
|
|
|
|
|
|
except:
|
|
|
|
return {}
|
|
|
|
|
|
|
|
@cache
|
2018-03-06 20:52:39 +08:00
|
|
|
def GetNtds(self):
|
|
|
|
if not self.GetMimikatzOutput():
|
|
|
|
return {}
|
|
|
|
|
|
|
|
mimikatz = self.GetMimikatzOutput()
|
|
|
|
|
|
|
|
if mimikatz.count("\n42.") != 2:
|
|
|
|
return {}
|
|
|
|
|
|
|
|
ntds_users = mimikatz.split("\n42.")[2].split("\nRID :")[1:]
|
|
|
|
ntds = {}
|
|
|
|
|
|
|
|
for ntds_user_txt in ntds_users:
|
|
|
|
user = ntds_user_txt.split("User :")[1].splitlines()[0].replace("User :", "").strip()
|
|
|
|
ntlm = ntds_user_txt.split("* Primary\n NTLM :")[1].splitlines()[0].replace("NTLM :", "").strip()
|
|
|
|
ntlm = ntlm.replace("[hashed secret]", "").strip()
|
|
|
|
|
|
|
|
if ntlm:
|
|
|
|
ntds[user] = ntlm
|
|
|
|
|
|
|
|
return ntds
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 20:52:39 +08:00
|
|
|
def GetLocalSecrets(self):
|
|
|
|
sam = self.GetSam()
|
|
|
|
ntds = self.GetNtds()
|
|
|
|
|
|
|
|
secrets = sam.copy()
|
|
|
|
secrets.update(ntds)
|
|
|
|
|
|
|
|
return secrets
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def GetLocalAdminSecrets(self):
|
2018-03-07 23:45:48 +08:00
|
|
|
return set(self.GetLocalAdminCreds().values())
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 23:45:48 +08:00
|
|
|
def GetLocalAdminCreds(self):
|
2018-03-06 16:32:57 +08:00
|
|
|
admin_names = self.GetLocalAdminNames()
|
2018-03-06 20:52:39 +08:00
|
|
|
sam = self.GetLocalSecrets()
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
admin_creds = dict()
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
for username, secret in sam.iteritems():
|
|
|
|
if username not in admin_names:
|
2018-03-04 21:22:34 +08:00
|
|
|
continue
|
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
admin_creds[username] = secret
|
2018-03-06 20:52:39 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
return admin_creds
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetCachedSecrets(self):
|
2018-03-07 23:45:48 +08:00
|
|
|
return set(self.GetCachedCreds().values())
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 23:45:48 +08:00
|
|
|
def GetCachedCreds(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
creds = dict()
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
if not self.GetMimikatzOutput():
|
|
|
|
return {}
|
|
|
|
|
|
|
|
mimikatz = self.GetMimikatzOutput()
|
|
|
|
|
|
|
|
for user in mimikatz.split("\n42.")[0].split("Authentication Id")[1:]:
|
|
|
|
username = None
|
|
|
|
secret = None
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
for line in user.splitlines():
|
|
|
|
if "User Name" in line:
|
|
|
|
username = line.split(":")[1].strip()
|
|
|
|
|
|
|
|
if ("NTLM" in line or "Password" in line) and "[hashed secret]" in line:
|
|
|
|
secret = line.split(":")[1].replace("[hashed secret]", "").strip()
|
|
|
|
|
|
|
|
if username and secret:
|
|
|
|
creds[username] = secret
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
return creds
|
2018-03-06 16:32:57 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def GetDomainControllers(self):
|
2018-03-04 21:22:34 +08:00
|
|
|
domain_name = self.GetDomainName()
|
|
|
|
DCs = self.GetDomainControllersMonkeyGuidByDomainName(domain_name)
|
2018-03-06 16:32:57 +08:00
|
|
|
return map(Machine, DCs)
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def GetDomainAdminsOfMachine(self):
|
|
|
|
DCs = self.GetDomainControllers()
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
domain_admins = set()
|
|
|
|
|
2018-03-06 16:32:57 +08:00
|
|
|
for dc in DCs:
|
2018-04-30 23:15:10 +08:00
|
|
|
domain_admins |= set(dc.GetUsersByGroupSid(self.GetGroupSidByGroupName("Domain Admins")).keys())
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return domain_admins
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetAdmins(self):
|
2018-04-17 18:12:57 +08:00
|
|
|
return self.GetLocalAdminSids() | self.GetDomainAdminsOfMachine()
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetAdminNames(self):
|
|
|
|
return set(map(lambda x: self.GetUsernameBySid(x), self.GetAdmins()))
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetCachedSids(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
SIDs = set()
|
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for username in doc["data"]["credentials"]:
|
2018-03-11 15:58:51 +08:00
|
|
|
sid = self.GetSidByUsername(username)
|
|
|
|
|
|
|
|
if not sid:
|
|
|
|
sid = "__USERNAME__" + username
|
|
|
|
|
|
|
|
SIDs.add(sid)
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
return SIDs
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetCachedUsernames(self):
|
2018-03-06 21:37:50 +08:00
|
|
|
doc = self.latest_system_info
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 23:31:35 +08:00
|
|
|
names = set()
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 21:37:50 +08:00
|
|
|
for username in doc["data"]["credentials"]:
|
2018-03-06 23:31:35 +08:00
|
|
|
names.add(username)
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-06 23:31:35 +08:00
|
|
|
return names
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-05-15 20:26:46 +08:00
|
|
|
class PassTheHashReport(object):
|
2018-03-04 21:22:34 +08:00
|
|
|
def __init__(self):
|
|
|
|
self.vertices = self.GetAllMachines()
|
2018-04-03 15:47:10 +08:00
|
|
|
|
2018-03-04 21:22:34 +08:00
|
|
|
self.edges = set()
|
2018-03-07 15:56:32 +08:00
|
|
|
self.machines = map(Machine, self.vertices)
|
2018-04-03 15:47:10 +08:00
|
|
|
|
2018-03-06 16:32:57 +08:00
|
|
|
self.GenerateEdgesBySid() # Useful for non-cached domain users
|
2018-03-04 21:22:34 +08:00
|
|
|
self.GenerateEdgesBySamHash() # This will add edges based only on password hash without caring about username
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GetAllMachines(self):
|
|
|
|
cur = mongo.db.telemetry.find({"telem_type":"system_info_collection"})
|
|
|
|
|
|
|
|
GUIDs = set()
|
|
|
|
|
|
|
|
for doc in cur:
|
|
|
|
GUIDs.add(doc["monkey_guid"])
|
|
|
|
|
|
|
|
return GUIDs
|
2018-04-03 15:47:10 +08:00
|
|
|
|
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def ReprSidList(self, sid_list, attacker, victim):
|
|
|
|
label = set()
|
|
|
|
|
|
|
|
for sid in sid_list:
|
|
|
|
username = Machine(victim).GetUsernameBySid(sid)
|
|
|
|
|
|
|
|
#if not username:
|
|
|
|
# username = Machine(attacker).GetUsernameBySid(sid)
|
|
|
|
|
|
|
|
if username:
|
|
|
|
label.add(username)
|
|
|
|
|
|
|
|
return ",\n".join(label)
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 16:32:57 +08:00
|
|
|
def ReprSecretList(self, secret_list, victim):
|
|
|
|
label = set()
|
|
|
|
|
|
|
|
for secret in secret_list:
|
2018-03-07 23:45:48 +08:00
|
|
|
label |= Machine(victim).GetUsernamesBySecret(secret)
|
2018-03-06 16:32:57 +08:00
|
|
|
|
|
|
|
return ",\n".join(label)
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GenerateEdgesBySid(self):
|
|
|
|
for attacker in self.vertices:
|
2018-05-15 17:55:50 +08:00
|
|
|
cached = self.GetCachedSids(Machine(attacker))
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
for victim in self.vertices:
|
|
|
|
if attacker == victim:
|
|
|
|
continue
|
|
|
|
|
|
|
|
admins = Machine(victim).GetAdmins()
|
2018-04-03 15:47:10 +08:00
|
|
|
|
2018-03-04 21:22:34 +08:00
|
|
|
if len(cached & admins) > 0:
|
2018-03-06 16:32:57 +08:00
|
|
|
label = self.ReprSidList(cached & admins, attacker, victim)
|
|
|
|
self.edges.add((attacker, victim, label))
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GenerateEdgesBySamHash(self):
|
|
|
|
for attacker in self.vertices:
|
2018-03-07 23:45:48 +08:00
|
|
|
cached_creds = set(Machine(attacker).GetCachedCreds().items())
|
2018-03-04 21:22:34 +08:00
|
|
|
|
|
|
|
for victim in self.vertices:
|
|
|
|
if attacker == victim:
|
|
|
|
continue
|
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
admin_creds = set(Machine(victim).GetLocalAdminCreds().items())
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-03-07 23:45:48 +08:00
|
|
|
if len(cached_creds & admin_creds) > 0:
|
|
|
|
label = self.ReprSecretList(set(dict(cached_creds & admin_creds).values()), victim)
|
2018-03-06 16:32:57 +08:00
|
|
|
self.edges.add((attacker, victim, label))
|
2018-03-04 21:22:34 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def GenerateEdgesByUsername(self):
|
|
|
|
for attacker in self.vertices:
|
|
|
|
cached = Machine(attacker).GetCachedUsernames()
|
|
|
|
|
|
|
|
for victim in self.vertices:
|
|
|
|
if attacker == victim:
|
|
|
|
continue
|
|
|
|
|
|
|
|
admins = Machine(victim).GetAdminNames()
|
|
|
|
|
|
|
|
if len(cached & admins) > 0:
|
|
|
|
self.edges.add((attacker, victim))
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-04 21:22:34 +08:00
|
|
|
def Print(self):
|
|
|
|
print map(lambda x: Machine(x).GetIp(), self.vertices)
|
|
|
|
print map(lambda x: (Machine(x[0]).GetIp(), Machine(x[1]).GetIp()), self.edges)
|
2018-03-06 21:37:50 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-11 15:18:53 +08:00
|
|
|
def GetPossibleAttackCountBySid(self, sid):
|
|
|
|
return len(self.GetPossibleAttacksBySid(sid))
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
|
|
|
def GetPossibleAttacksByAttacker(self, attacker):
|
|
|
|
attacks = set()
|
|
|
|
|
|
|
|
cached_creds = set(Machine(attacker).GetCachedCreds().items())
|
|
|
|
|
|
|
|
for victim in self.vertices:
|
|
|
|
if attacker == victim:
|
|
|
|
continue
|
|
|
|
|
|
|
|
admin_creds = set(Machine(victim).GetLocalAdminCreds().items())
|
|
|
|
|
|
|
|
if len(cached_creds & admin_creds) > 0:
|
|
|
|
curr_attacks = dict(cached_creds & admin_creds)
|
|
|
|
attacks.add((attacker, victim, curr_attacks))
|
|
|
|
|
|
|
|
return attacks
|
|
|
|
|
|
|
|
@cache
|
2018-03-11 15:18:53 +08:00
|
|
|
def GetPossibleAttacksBySid(self, sid):
|
|
|
|
attacks = set()
|
|
|
|
|
|
|
|
for attacker in self.vertices:
|
2018-04-03 15:47:10 +08:00
|
|
|
tmp = self.GetPossibleAttacksByAttacker(attacker)
|
2018-03-11 15:18:53 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
for _, victim, curr_attacks in tmp:
|
|
|
|
for username, secret in curr_attacks.iteritems():
|
|
|
|
if Machine(victim).GetSidByUsername(username) == sid:
|
|
|
|
attacks.add((attacker, victim))
|
2018-03-11 15:18:53 +08:00
|
|
|
|
|
|
|
return attacks
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetSecretBySid(self, sid):
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
|
|
|
for user, user_secret in m.GetLocalSecrets().iteritems():
|
2018-03-06 23:31:35 +08:00
|
|
|
if m.GetSidByUsername(user) == sid:
|
|
|
|
return user_secret
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 15:56:32 +08:00
|
|
|
def GetVictimCountBySid(self, sid):
|
2018-03-07 22:47:29 +08:00
|
|
|
return len(self.GetVictimsBySid(sid))
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 15:56:32 +08:00
|
|
|
def GetVictimCountByMachine(self, attacker):
|
|
|
|
return len(self.GetVictimsByAttacker(attacker))
|
2018-03-06 23:31:35 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 23:45:48 +08:00
|
|
|
def GetAttackCountBySecret(self, secret):
|
|
|
|
return len(self.GetAttackersBySecret(secret))
|
2018-03-07 04:29:17 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 04:29:17 +08:00
|
|
|
def GetAllUsernames(self):
|
|
|
|
names = set()
|
|
|
|
|
|
|
|
for sid in self.GetAllSids():
|
|
|
|
names.add(self.GetUsernameBySid(sid))
|
|
|
|
|
|
|
|
return names
|
2018-03-06 21:37:50 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 03:53:22 +08:00
|
|
|
def GetAllSids(self):
|
|
|
|
SIDs = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-07 03:53:22 +08:00
|
|
|
SIDs |= m.GetLocalSids()
|
|
|
|
|
|
|
|
return SIDs
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetAllSecrets(self):
|
2018-03-07 03:53:22 +08:00
|
|
|
secrets = set()
|
2018-03-06 21:37:50 +08:00
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 21:37:50 +08:00
|
|
|
for secret in m.GetLocalAdminSecrets():
|
2018-03-07 15:56:32 +08:00
|
|
|
secrets.add(secret)
|
2018-03-06 21:37:50 +08:00
|
|
|
|
|
|
|
return secrets
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetUsernameBySid(self, sid):
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 21:37:50 +08:00
|
|
|
username = m.GetUsernameBySid(sid)
|
|
|
|
|
|
|
|
if username:
|
|
|
|
return username
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-10 22:51:09 +08:00
|
|
|
@cache
|
|
|
|
def GetSidInfo(self, sid):
|
|
|
|
for m in self.machines:
|
|
|
|
info = m.GetSidInfo(sid)
|
|
|
|
|
|
|
|
if info:
|
|
|
|
return info
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetSidsBySecret(self, secret):
|
2018-03-06 21:37:50 +08:00
|
|
|
SIDs = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-07 23:45:48 +08:00
|
|
|
SIDs |= m.GetSidsBySecret(secret)
|
2018-03-06 21:37:50 +08:00
|
|
|
|
2018-03-06 23:31:35 +08:00
|
|
|
return SIDs
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetAllDomainControllers(self):
|
|
|
|
DCs = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 23:31:35 +08:00
|
|
|
if m.IsDomainController():
|
|
|
|
DCs.add(m)
|
2018-03-07 15:56:32 +08:00
|
|
|
|
|
|
|
return DCs
|
2018-03-06 23:31:35 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetSidsByUsername(self, username):
|
|
|
|
SIDs = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 23:31:35 +08:00
|
|
|
sid = m.GetSidByUsername(username)
|
|
|
|
if sid:
|
|
|
|
SIDs.add(sid)
|
|
|
|
|
|
|
|
return SIDs
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 15:56:32 +08:00
|
|
|
def GetVictimsBySid(self, sid):
|
2018-03-06 23:31:35 +08:00
|
|
|
machines = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 23:31:35 +08:00
|
|
|
if sid in m.GetAdmins():
|
|
|
|
machines.add(m)
|
|
|
|
|
|
|
|
return machines
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 15:56:32 +08:00
|
|
|
def GetVictimsBySecret(self, secret):
|
2018-03-06 23:31:35 +08:00
|
|
|
machines = set()
|
|
|
|
|
|
|
|
SIDs = self.GetSidsBySecret(secret)
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 23:31:35 +08:00
|
|
|
if len(SIDs & m.GetAdmins()) > 0:
|
|
|
|
machines.add(m)
|
|
|
|
|
|
|
|
return machines
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetAttackersBySecret(self, secret):
|
|
|
|
machines = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in self.machines:
|
2018-03-06 23:31:35 +08:00
|
|
|
if secret in m.GetCachedSecrets():
|
|
|
|
machines.add(m)
|
|
|
|
|
|
|
|
return machines
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-06 23:31:35 +08:00
|
|
|
def GetAttackersByVictim(self, victim):
|
2018-04-17 19:59:06 +08:00
|
|
|
if type(victim) != unicode:
|
|
|
|
victim = victim.monkey_guid
|
2018-03-07 15:36:04 +08:00
|
|
|
|
2018-04-17 19:59:06 +08:00
|
|
|
attackers = set()
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for atck, vic, _ in self.edges:
|
2018-03-07 15:36:04 +08:00
|
|
|
if vic == victim:
|
|
|
|
attackers.add(atck)
|
|
|
|
|
2018-04-17 19:59:06 +08:00
|
|
|
return set(map(Machine, attackers))
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-05-15 18:07:19 +08:00
|
|
|
@cache
|
|
|
|
def GetAttackersBySid(self, sid):
|
|
|
|
machines = set()
|
|
|
|
|
|
|
|
for m in self.machines:
|
|
|
|
if sid in self.GetCachedSids(m):
|
|
|
|
machines.add(m)
|
|
|
|
|
|
|
|
return machines
|
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 15:56:32 +08:00
|
|
|
def GetVictimsByAttacker(self, attacker):
|
2018-04-17 20:02:49 +08:00
|
|
|
if type(attacker) != unicode:
|
|
|
|
attacker = attacker.monkey_guid
|
2018-04-17 20:00:22 +08:00
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
victims = set()
|
|
|
|
|
|
|
|
for atck, vic, _ in self.edges:
|
|
|
|
if atck == attacker:
|
|
|
|
victims.add(vic)
|
|
|
|
|
2018-04-17 20:00:22 +08:00
|
|
|
return set(map(Machine, victims))
|
2018-03-07 22:47:29 +08:00
|
|
|
|
2018-04-03 15:47:10 +08:00
|
|
|
@cache
|
2018-03-07 22:47:29 +08:00
|
|
|
def GetInPathCountByVictim(self, victim, already_processed=None):
|
|
|
|
if type(victim) != unicode:
|
|
|
|
victim = victim.monkey_guid
|
|
|
|
|
|
|
|
if not already_processed:
|
|
|
|
already_processed = set([victim])
|
|
|
|
|
|
|
|
count = 0
|
|
|
|
|
|
|
|
for atck, vic, _ in self.edges:
|
|
|
|
if atck == vic:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if vic != victim:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if atck in already_processed:
|
|
|
|
continue
|
|
|
|
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
already_processed.add(atck)
|
|
|
|
count += self.GetInPathCountByVictim(atck, already_processed)
|
|
|
|
|
|
|
|
return count
|
2018-03-07 15:56:32 +08:00
|
|
|
|
2018-04-17 17:45:32 +08:00
|
|
|
@cache
|
|
|
|
def GetCritialServers(self):
|
|
|
|
machines = set()
|
|
|
|
|
|
|
|
for m in self.machines:
|
|
|
|
if m.IsCriticalServer():
|
|
|
|
machines.add(m)
|
|
|
|
|
|
|
|
return machines
|
|
|
|
|
2018-04-17 20:54:03 +08:00
|
|
|
@cache
|
|
|
|
def GetNonCritialServers(self):
|
2018-04-17 20:57:44 +08:00
|
|
|
return set(self.machines) - self.GetCritialServers()
|
2018-04-17 20:54:03 +08:00
|
|
|
|
2018-05-15 17:55:50 +08:00
|
|
|
@cache
|
|
|
|
def GetCachedSids(self, m):
|
|
|
|
sids = set()
|
|
|
|
tmp = m.GetCachedSids()
|
|
|
|
|
|
|
|
for sid in tmp:
|
|
|
|
if sid.startswith("__USERNAME__"):
|
|
|
|
|
|
|
|
s = self.GetSidsByUsername(sid[len("__USERNAME__"):])
|
|
|
|
if len(s) == 1:
|
|
|
|
sids.add(s.pop())
|
|
|
|
else:
|
|
|
|
sids.add(sid)
|
|
|
|
|
|
|
|
else:
|
|
|
|
sids.add(sid)
|
|
|
|
|
|
|
|
return sids
|
|
|
|
|
2018-04-17 20:54:03 +08:00
|
|
|
@cache
|
2018-04-17 19:43:26 +08:00
|
|
|
def GetThreateningUsersByVictim(self, victim):
|
2018-04-17 18:58:47 +08:00
|
|
|
threatening_users = set()
|
|
|
|
|
2018-04-17 19:43:26 +08:00
|
|
|
for attacker in self.GetAttackersByVictim(victim):
|
2018-05-15 17:55:50 +08:00
|
|
|
threatening_users |= (self.GetCachedSids(attacker) & victim.GetAdmins())
|
2018-04-17 18:58:47 +08:00
|
|
|
|
|
|
|
return threatening_users
|
|
|
|
|
2018-04-30 22:48:21 +08:00
|
|
|
@cache
|
|
|
|
def GetSharedAdmins(self, m):
|
|
|
|
shared_admins = set()
|
|
|
|
|
2018-04-30 22:50:38 +08:00
|
|
|
for other in self.machines:
|
2018-04-30 22:48:21 +08:00
|
|
|
if m == other:
|
|
|
|
continue
|
|
|
|
|
|
|
|
shared_admins |= (m.GetLocalAdminSids() & other.GetLocalAdminSids())
|
|
|
|
|
2018-04-30 22:58:15 +08:00
|
|
|
shared_admins -= m.GetDomainAdminsOfMachine()
|
2018-04-30 22:48:21 +08:00
|
|
|
return shared_admins
|
|
|
|
|
2018-03-07 03:53:22 +08:00
|
|
|
def main():
|
2018-05-15 20:26:46 +08:00
|
|
|
pth = PassTheHashReport()
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-05-15 19:28:51 +08:00
|
|
|
print """<style>table, td {border: ridge;} .hidden * {color:white; border:white;}</style>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
print "<h1>Pass The Hash Report</h1>"
|
|
|
|
|
|
|
|
print "<h2>Duplicated Passwords</h2>"
|
|
|
|
print "<h3>How many users share each secret?</h3>"
|
2018-04-03 15:47:10 +08:00
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
dups = dict(map(lambda x: (x, len(pth.GetSidsBySecret(x))), pth.GetAllSecrets()))
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<table>"""
|
2018-04-30 21:53:48 +08:00
|
|
|
print """<tr><th>User Count</th><th>Users That Share This Password</th></tr>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
for secret, count in sorted(dups.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-03 15:47:10 +08:00
|
|
|
if count <= 1:
|
|
|
|
continue
|
2018-04-30 21:53:48 +08:00
|
|
|
print """<tr><td>{count}</td>""".format(secret=secret, count=count)
|
2018-04-17 18:12:57 +08:00
|
|
|
print """<td><ul>"""
|
|
|
|
for sid in pth.GetSidsBySecret(secret):
|
|
|
|
print """<li><a href="#{sid}">{username}</a></li>""".format(sid=sid, username=pth.GetUsernameBySid(sid))
|
|
|
|
print """</ul></td></tr>"""
|
2018-03-07 16:01:43 +08:00
|
|
|
print """</table>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-04-30 22:48:21 +08:00
|
|
|
|
|
|
|
print "<h2>Local Admin Uniqueness</h2>"
|
|
|
|
print "<h3>We argue that each machine should have it's own distinct set of local admins</h3>"
|
2018-04-17 18:58:47 +08:00
|
|
|
|
2018-04-30 22:48:21 +08:00
|
|
|
dups = dict(map(lambda x: (x, len(pth.GetSharedAdmins(x))), pth.machines))
|
2018-04-17 18:58:47 +08:00
|
|
|
|
2018-04-30 22:48:21 +08:00
|
|
|
print """<table>"""
|
|
|
|
print """<tr><th>Ip</th><th>Hostname</th><th>Domain</th><th>Critical Services Installed</th><th>Shared User Count</th><th>Shared Users</th></tr>"""
|
2018-04-30 22:50:38 +08:00
|
|
|
for m, count in sorted(dups.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-30 22:48:21 +08:00
|
|
|
if count <= 0:
|
|
|
|
continue
|
|
|
|
|
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td>{hostname}</td><td>{domain}</td>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName(), count=count)
|
|
|
|
|
|
|
|
print """<td><ul>"""
|
|
|
|
for service_name in m.GetCriticalServicesInstalled():
|
|
|
|
print """<li>{service_name}</li>""".format(service_name=service_name)
|
|
|
|
print """</ul></td>"""
|
|
|
|
|
|
|
|
print """<td>{count}</td>""".format(count=count)
|
|
|
|
|
|
|
|
print """<td><ul>"""
|
|
|
|
|
2018-04-30 23:15:10 +08:00
|
|
|
for sid in pth.GetSharedAdmins(m):
|
2018-04-30 22:48:21 +08:00
|
|
|
print """<li><a href="#{sid}">{username}</a></li>""".format(sid=sid, username=pth.GetUsernameBySid(sid))
|
|
|
|
|
|
|
|
print """</ul></td></tr>"""
|
|
|
|
print """</table>"""
|
|
|
|
|
2018-04-17 18:58:47 +08:00
|
|
|
print "<h2>Strong Users That Threat Critical Servers</h2>"
|
|
|
|
print "<h3>Administrators of critical servers that we could find thier secret cached somewhere</h3>"
|
|
|
|
|
|
|
|
threatening = dict(map(lambda x: (x, len(pth.GetThreateningUsersByVictim(x))), pth.GetCritialServers()))
|
|
|
|
|
|
|
|
print """<table>"""
|
2018-04-30 21:28:29 +08:00
|
|
|
print """<tr><th>Critical Server</th><th>Hostname</th><th>Domain</th><th>Critical Services Installed</th><th>Threatening User Count</th><th>Threatening Users</th></tr>"""
|
2018-04-17 18:58:47 +08:00
|
|
|
for m, count in sorted(threatening.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
|
|
|
if count <= 0:
|
|
|
|
continue
|
2018-04-30 21:48:42 +08:00
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td>{hostname}</td><td>{domain}</td>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName(), count=count)
|
2018-04-30 21:28:29 +08:00
|
|
|
|
|
|
|
print """<td><ul>"""
|
|
|
|
for service_name in m.GetCriticalServicesInstalled():
|
|
|
|
print """<li>{service_name}</li>""".format(service_name=service_name)
|
|
|
|
print """</ul></td>"""
|
|
|
|
|
2018-04-30 21:48:42 +08:00
|
|
|
print """<td>{count}</td>""".format(count=count)
|
|
|
|
|
2018-04-17 20:54:03 +08:00
|
|
|
print """<td><ul>"""
|
|
|
|
|
|
|
|
for sid in pth.GetThreateningUsersByVictim(m):
|
2018-05-15 19:20:09 +08:00
|
|
|
print """<li><a href="#{sid}">{username}</a> attackers:<ul>""".format(sid=sid, username=pth.GetUsernameBySid(sid))
|
|
|
|
|
|
|
|
for mm in pth.GetAttackersBySid(sid):
|
|
|
|
if m == mm:
|
|
|
|
continue
|
|
|
|
print """<li><a href="#{ip}">{ip}</a></li>""".format(ip=mm.GetIp())
|
|
|
|
print """</ul></li>"""
|
2018-04-17 20:54:03 +08:00
|
|
|
|
|
|
|
print """</ul></td></tr>"""
|
|
|
|
print """</table>"""
|
|
|
|
|
|
|
|
|
|
|
|
print "<h2>Strong Users That Threat NonCritical Servers</h2>"
|
|
|
|
print "<h3>Administrators of non-critical servers that we could find thier secret cached somewhere</h3>"
|
|
|
|
|
|
|
|
threatening = dict(map(lambda x: (x, len(pth.GetThreateningUsersByVictim(x))), pth.GetNonCritialServers()))
|
|
|
|
|
|
|
|
print """<table>"""
|
2018-04-30 21:51:07 +08:00
|
|
|
print """<tr><th>Critical Server</th><th>Hostname</th><th>Domain</th><th>Threatening User Count</th><th>Threatening Users</th></tr>"""
|
2018-04-17 20:54:03 +08:00
|
|
|
for m, count in sorted(threatening.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
|
|
|
if count <= 0:
|
|
|
|
continue
|
2018-04-30 21:48:42 +08:00
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td>{hostname}</td><td>{domain}</td>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName(), count=count)
|
2018-04-30 21:28:29 +08:00
|
|
|
|
|
|
|
|
2018-04-30 21:48:42 +08:00
|
|
|
print """<td>{count}</td>""".format(count=count)
|
|
|
|
|
2018-04-17 18:58:47 +08:00
|
|
|
print """<td><ul>"""
|
|
|
|
|
|
|
|
for sid in pth.GetThreateningUsersByVictim(m):
|
2018-05-15 19:20:09 +08:00
|
|
|
print """<li><a href="#{sid}">{username}</a> attackers:<ul>""".format(sid=sid, username=pth.GetUsernameBySid(sid))
|
|
|
|
|
|
|
|
for mm in pth.GetAttackersBySid(sid):
|
|
|
|
if m == mm:
|
|
|
|
continue
|
|
|
|
print """<li><a href="#{ip}">{ip}</a></li>""".format(ip=mm.GetIp())
|
|
|
|
print """</ul></li>"""
|
2018-04-17 18:58:47 +08:00
|
|
|
|
|
|
|
print """</ul></td></tr>"""
|
|
|
|
print """</table>"""
|
2018-04-30 21:19:58 +08:00
|
|
|
|
2018-05-15 19:28:51 +08:00
|
|
|
print """<div class="hidden">"""
|
2018-04-17 18:58:47 +08:00
|
|
|
|
2018-03-07 03:53:22 +08:00
|
|
|
print "<h2>Cached Passwords</h2>"
|
2018-03-07 23:45:48 +08:00
|
|
|
print "<h3>On how many machines each secret is cached (possible attacker count)?</h3>"
|
|
|
|
cache_counts = dict(map(lambda x: (x, pth.GetAttackCountBySecret(x)), pth.GetAllSecrets()))
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<table>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
print """<tr><th>Secret</th><th>Machine Count</th></tr>"""
|
|
|
|
for secret, count in sorted(cache_counts.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-03 15:47:10 +08:00
|
|
|
if count <= 0:
|
|
|
|
continue
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<tr><td><a href="#{secret}">{secret}</a></td><td>{count}</td>""".format(secret=secret, count=count)
|
|
|
|
print """</table>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
|
|
|
|
print "<h2>User's Creds</h2>"
|
2018-03-11 15:18:53 +08:00
|
|
|
print "<h3>To how many machines each user is able to connect with admin rights</h3>"
|
2018-03-07 15:56:32 +08:00
|
|
|
attackable_counts = dict(map(lambda x: (x, pth.GetVictimCountBySid(x)), pth.GetAllSids()))
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<table>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
print """<tr><th>SID</th><th>Username</th><th>Machine Count</th></tr>"""
|
|
|
|
for sid, count in sorted(attackable_counts.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-03 15:47:10 +08:00
|
|
|
if count <= 1:
|
|
|
|
continue
|
2018-03-07 22:48:03 +08:00
|
|
|
print """<tr><td><a href="#{sid}">{sid}</a></td><td>{username}</td><td>{count}</td>""".format(sid=sid, username=pth.GetUsernameBySid(sid), count=count)
|
2018-03-07 16:01:43 +08:00
|
|
|
print """</table>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-03-11 15:18:53 +08:00
|
|
|
print "<h2>Actual Possible Attacks By SID</h2>"
|
|
|
|
print "<h3>How many attacks possible using each SID (aka len(attacker->victim pairs))</h3>"
|
|
|
|
possible_attacks_by_sid = dict(map(lambda x: (x, pth.GetPossibleAttackCountBySid(x)), pth.GetAllSids()))
|
|
|
|
|
|
|
|
print """<table>"""
|
|
|
|
print """<tr><th>SID</th><th>Username</th><th>Machine Count</th></tr>"""
|
|
|
|
for sid, count in sorted(possible_attacks_by_sid.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-03 15:47:10 +08:00
|
|
|
if count <= 1:
|
|
|
|
continue
|
2018-03-11 15:18:53 +08:00
|
|
|
print """<tr><td><a href="#{sid}">{sid}</a></td><td>{username}</td><td>{count}</td>""".format(sid=sid, username=pth.GetUsernameBySid(sid), count=count)
|
|
|
|
print """</table>"""
|
|
|
|
|
2018-03-07 03:53:22 +08:00
|
|
|
print "<h2>Machine's Creds</h2>"
|
|
|
|
print "<h3>To how many machines each machine is able to directly connect with admin rights?</h3>"
|
2018-03-07 15:56:32 +08:00
|
|
|
attackable_counts = dict(map(lambda m: (m, pth.GetVictimCountByMachine(m)), pth.machines))
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<table>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
print """<tr><th>Attacker Ip</th><th>Attacker Hostname</th><th>Domain Name</th><th>Victim Machine Count</th></tr>"""
|
|
|
|
for m, count in sorted(attackable_counts.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-03 15:47:10 +08:00
|
|
|
if count <= 1:
|
|
|
|
continue
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td>{hostname}</td><td>{domain}</td><td>{count}</td>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName(), count=count)
|
|
|
|
print """</table>"""
|
2018-03-07 03:55:56 +08:00
|
|
|
|
|
|
|
print "<h2>Domain Controllers</h2>"
|
|
|
|
print "<h3>List of domain controllers (we count them as critical points, so they are listed here)</h3>"
|
2018-03-07 22:47:29 +08:00
|
|
|
DCs = dict(map(lambda m: (m, pth.GetInPathCountByVictim(m)), pth.GetAllDomainControllers()))
|
|
|
|
|
|
|
|
print """<table>"""
|
|
|
|
print """<tr><th>DC Ip</th><th>DC Hostname</th><th>Domain Name</th><th>In-Path Count</th></tr>"""
|
|
|
|
for m, path_count in sorted(DCs.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td><a href="#{ip}">{hostname}</a></td><td>{domain}</td><td>{path_count}</td></tr>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName(), path_count=path_count)
|
|
|
|
print """</table>"""
|
|
|
|
|
|
|
|
print "<h2>Most Vulnerable Machines</h2>"
|
|
|
|
print "<h3>List all machines in the network sorted by the potincial to attack them</h3>"
|
|
|
|
all_machines = dict(map(lambda m: (m, pth.GetInPathCountByVictim(m)), pth.machines))
|
2018-03-07 03:55:56 +08:00
|
|
|
|
2018-03-07 16:01:43 +08:00
|
|
|
print """<table>"""
|
2018-03-07 22:47:29 +08:00
|
|
|
print """<tr><th>Ip</th><th>Hostname</th><th>Domain Name</th><th>In-Path Count</th></tr>"""
|
|
|
|
for m, path_count in sorted(all_machines.iteritems(), key=lambda (k,v): (v,k), reverse=True):
|
2018-04-03 16:01:53 +08:00
|
|
|
if count <= 0:
|
|
|
|
continue
|
2018-03-07 22:47:29 +08:00
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td><a href="#{ip}">{hostname}</a></td><td>{domain}</td><td>{path_count}</td></tr>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName(), path_count=path_count)
|
2018-03-07 16:01:43 +08:00
|
|
|
print """</table>"""
|
2018-03-07 04:29:17 +08:00
|
|
|
|
2018-04-17 17:45:32 +08:00
|
|
|
print "<h2>Critical Servers</h2>"
|
|
|
|
print "<h3>List of all machines identified as critical servers</h3>"
|
|
|
|
critical_servers = pth.GetCritialServers()
|
|
|
|
|
|
|
|
print """<table>"""
|
|
|
|
print """<tr><th>Ip</th><th>Hostname</th><th>Domain Name</th></tr>"""
|
|
|
|
for m in critical_servers:
|
|
|
|
print """<tr><td><a href="#{ip}">{ip}</a></td><td><a href="#{ip}">{hostname}</a></td><td>{domain}</td></tr>""".format(ip=m.GetIp(), hostname=m.GetHostName(), domain=m.GetDomainName())
|
|
|
|
print """</table>"""
|
|
|
|
|
2018-03-07 04:29:17 +08:00
|
|
|
print "<hr />"
|
|
|
|
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in pth.machines:
|
2018-03-07 04:29:17 +08:00
|
|
|
print """<a name="{ip}"><h2>Machine '{ip}'</h2></a>
|
2018-03-07 15:56:32 +08:00
|
|
|
<h3>Hostname '{hostname}'</h3>""".format(ip=m.GetIp(), hostname=m.GetHostName())
|
2018-03-07 04:29:17 +08:00
|
|
|
|
|
|
|
print """<h3>Cached SIDs</h3>"""
|
2018-03-07 15:36:04 +08:00
|
|
|
print """<h4>SIDs cached on this machine</h4>"""
|
2018-03-07 04:29:17 +08:00
|
|
|
print """<ul>"""
|
2018-05-15 17:55:50 +08:00
|
|
|
for sid in pth.GetCachedSids(m):
|
2018-03-11 15:58:51 +08:00
|
|
|
print """<li><a href="#{sid}">{username} ({sid})</a></li>""".format(username=pth.GetUsernameBySid(sid), sid=sid)
|
2018-03-07 04:29:17 +08:00
|
|
|
print """</ul>"""
|
|
|
|
|
|
|
|
print """<h3>Possible Attackers</h3>"""
|
2018-03-07 15:36:04 +08:00
|
|
|
print """<h4>Machines that can attack this machine</h4>"""
|
|
|
|
print """<ul>"""
|
|
|
|
for attacker in pth.GetAttackersByVictim(m):
|
|
|
|
print """<li><a href="#{ip}">{ip} ({hostname})</a></li>""".format(ip=attacker.GetIp(), hostname=attacker.GetHostName())
|
|
|
|
print """</ul>"""
|
|
|
|
|
2018-03-07 04:29:17 +08:00
|
|
|
|
|
|
|
print """<h3>Admins</h3>"""
|
2018-03-07 15:36:04 +08:00
|
|
|
print """<h4>Users that have admin rights on this machine</h4>"""
|
2018-03-07 04:29:17 +08:00
|
|
|
print """<ul>"""
|
|
|
|
for sid in m.GetAdmins():
|
|
|
|
print """<li><a href="#{sid}">{username} ({sid})</a></li>""".format(username=m.GetUsernameBySid(sid), sid=sid)
|
|
|
|
print """</ul>"""
|
2018-04-17 17:29:28 +08:00
|
|
|
|
|
|
|
print """<h3>Installed Critical Services</h3>"""
|
|
|
|
print """<h4>List of crtical services found installed on machine</h4>"""
|
|
|
|
print """<ul>"""
|
2018-04-17 17:45:32 +08:00
|
|
|
for service_name in m.GetCriticalServicesInstalled():
|
2018-04-17 17:29:28 +08:00
|
|
|
print """<li>{service_name}</li>""".format(service_name=service_name)
|
|
|
|
print """</ul>"""
|
|
|
|
|
|
|
|
|
2018-03-07 04:29:17 +08:00
|
|
|
|
|
|
|
print "<hr />"
|
|
|
|
|
|
|
|
for username in pth.GetAllUsernames():
|
|
|
|
print """<a name="{username}"><h2>User '{username}'</h2></a>""".format(username=username)
|
|
|
|
|
|
|
|
print """<h3>Matching SIDs</h3>"""
|
|
|
|
print """<ul>"""
|
2018-03-07 15:56:32 +08:00
|
|
|
for sid in pth.GetSidsByUsername(username):
|
2018-03-11 15:58:51 +08:00
|
|
|
print """<li><a href="#{sid}">{username} ({sid})</a></li>""".format(username=pth.GetUsernameBySid(sid), sid=sid)
|
2018-03-07 04:29:17 +08:00
|
|
|
print """</ul>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
|
2018-03-07 04:29:17 +08:00
|
|
|
print "<hr />"
|
|
|
|
|
|
|
|
for sid in pth.GetAllSids():
|
|
|
|
print """<a name="{sid}"><h2>SID '{sid}'</h2></a>
|
2018-03-07 22:48:03 +08:00
|
|
|
<h3>Username: '<a href="#{username}">{username}</a>'</h3>
|
2018-04-10 22:51:09 +08:00
|
|
|
<h3>Domain: {domain}</h3>
|
2018-03-07 22:48:03 +08:00
|
|
|
<h3>Secret: '<a href="#{secret}">{secret}</a>'</h3>
|
2018-04-10 22:51:09 +08:00
|
|
|
""".format(username=pth.GetUsernameBySid(sid), sid=sid, secret=pth.GetSecretBySid(sid), domain=pth.GetSidInfo(sid)["Domain"])
|
2018-03-07 04:29:17 +08:00
|
|
|
|
2018-05-15 19:20:09 +08:00
|
|
|
print """<h3>Machines the sid is local admin on</h3>"""
|
2018-03-07 04:29:17 +08:00
|
|
|
print """<ul>"""
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in pth.GetVictimsBySid(sid):
|
2018-03-07 04:29:17 +08:00
|
|
|
print """<li><a href="#{ip}">{ip} ({hostname})</a></li>""".format(ip=m.GetIp(), hostname=m.GetHostName())
|
|
|
|
print """</ul>"""
|
2018-05-15 18:07:19 +08:00
|
|
|
|
2018-05-15 19:20:09 +08:00
|
|
|
print """<h3>Machines the sid is in thier cache</h3>"""
|
2018-05-15 18:07:19 +08:00
|
|
|
print """<ul>"""
|
|
|
|
for m in pth.GetAttackersBySid(sid):
|
|
|
|
print """<li><a href="#{ip}">{ip} ({hostname})</a></li>""".format(ip=m.GetIp(), hostname=m.GetHostName())
|
|
|
|
print """</ul>"""
|
2018-03-07 04:29:17 +08:00
|
|
|
|
|
|
|
for secret in pth.GetAllSecrets():
|
|
|
|
print """<a name="{secret}"><h2>Secret '{secret}'</h2></a>""".format(secret=secret)
|
|
|
|
|
|
|
|
print """<h3>SIDs that use that secret</h3>"""
|
|
|
|
print """<ul>"""
|
|
|
|
for sid in pth.GetSidsBySecret(secret):
|
2018-03-07 16:03:55 +08:00
|
|
|
print """<li><a href="#{sid}">{username} ({sid})</a></li>""".format(username=pth.GetUsernameBySid(sid), sid=sid)
|
2018-03-07 04:29:17 +08:00
|
|
|
print """</ul>"""
|
|
|
|
|
|
|
|
print """<h3>Attackable Machines with that secret</h3>"""
|
|
|
|
print """<ul>"""
|
2018-03-07 15:56:32 +08:00
|
|
|
for m in pth.GetVictimsBySecret(secret):
|
2018-03-07 04:29:17 +08:00
|
|
|
print """<li><a href="#{ip}">{hostname}</a></li>""".format(ip=m.GetIp(), hostname=m.GetHostName())
|
|
|
|
print """</ul>"""
|
|
|
|
|
|
|
|
print """<h3>Machines that have this secret cached and can use it to attack other machines</h3>"""
|
|
|
|
print """<ul>"""
|
|
|
|
for m in pth.GetAttackersBySecret(secret):
|
|
|
|
print """<li><a href="#{ip}">{hostname}</a></li>""".format(ip=m.GetIp(), hostname=m.GetHostName())
|
|
|
|
print """</ul>"""
|
|
|
|
|
2018-05-15 19:28:51 +08:00
|
|
|
print """</div>"""
|
2018-03-07 03:53:22 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|